2026-01-08 16:56:19 +01:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
|
#
|
|
|
|
|
# Rebuild a client's infrastructure from scratch
|
|
|
|
|
#
|
|
|
|
|
# Usage: ./scripts/rebuild-client.sh <client_name>
|
|
|
|
|
#
|
|
|
|
|
# This script will:
|
|
|
|
|
# 1. Destroy existing infrastructure (if exists)
|
|
|
|
|
# 2. Provision new VPS server
|
|
|
|
|
# 3. Deploy and configure all services
|
|
|
|
|
# 4. Configure SSO integration
|
|
|
|
|
#
|
|
|
|
|
# Result: Fully functional Authentik + Nextcloud with automated SSO
|
|
|
|
|
|
|
|
|
|
set -euo pipefail
|
|
|
|
|
|
|
|
|
|
# Colors for output
|
|
|
|
|
RED='\033[0;31m'
|
|
|
|
|
GREEN='\033[0;32m'
|
|
|
|
|
YELLOW='\033[1;33m'
|
|
|
|
|
BLUE='\033[0;34m'
|
|
|
|
|
NC='\033[0m' # No Color
|
|
|
|
|
|
|
|
|
|
# Script directory
|
|
|
|
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
|
|
|
PROJECT_ROOT="$(dirname "$SCRIPT_DIR")"
|
|
|
|
|
|
|
|
|
|
# Check arguments
|
|
|
|
|
if [ $# -ne 1 ]; then
|
|
|
|
|
echo -e "${RED}Error: Client name required${NC}"
|
|
|
|
|
echo "Usage: $0 <client_name>"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Example: $0 test"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
CLIENT_NAME="$1"
|
|
|
|
|
|
2026-01-17 20:04:29 +01:00
|
|
|
# Check if SSH key exists, generate if missing
|
|
|
|
|
SSH_KEY_FILE="$PROJECT_ROOT/keys/ssh/${CLIENT_NAME}"
|
|
|
|
|
if [ ! -f "$SSH_KEY_FILE" ]; then
|
|
|
|
|
echo -e "${YELLOW}SSH key not found for client: $CLIENT_NAME${NC}"
|
|
|
|
|
echo "Generating SSH key pair automatically..."
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Generate SSH key
|
|
|
|
|
"$SCRIPT_DIR/generate-client-keys.sh" "$CLIENT_NAME"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ SSH key generated${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# Check if secrets file exists, create from template if missing
|
2026-01-08 16:56:19 +01:00
|
|
|
SECRETS_FILE="$PROJECT_ROOT/secrets/clients/${CLIENT_NAME}.sops.yaml"
|
2026-01-17 20:04:29 +01:00
|
|
|
TEMPLATE_FILE="$PROJECT_ROOT/secrets/clients/template.sops.yaml"
|
|
|
|
|
|
2026-01-08 16:56:19 +01:00
|
|
|
if [ ! -f "$SECRETS_FILE" ]; then
|
2026-01-17 20:04:29 +01:00
|
|
|
echo -e "${YELLOW}Secrets file not found for client: $CLIENT_NAME${NC}"
|
|
|
|
|
echo "Creating from template and opening for editing..."
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Check if template exists
|
|
|
|
|
if [ ! -f "$TEMPLATE_FILE" ]; then
|
|
|
|
|
echo -e "${RED}Error: Template file not found: $TEMPLATE_FILE${NC}"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
2026-01-18 18:17:15 +01:00
|
|
|
# Copy template and decrypt to temporary file
|
|
|
|
|
if [ -z "${SOPS_AGE_KEY_FILE:-}" ]; then
|
|
|
|
|
export SOPS_AGE_KEY_FILE="$PROJECT_ROOT/keys/age-key.txt"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# Decrypt template to temp file
|
|
|
|
|
TEMP_PLAIN=$(mktemp)
|
|
|
|
|
sops -d "$TEMPLATE_FILE" > "$TEMP_PLAIN"
|
|
|
|
|
|
|
|
|
|
# Replace client name placeholders
|
|
|
|
|
sed -i '' "s/test/${CLIENT_NAME}/g" "$TEMP_PLAIN"
|
|
|
|
|
|
|
|
|
|
# Create unencrypted file in correct location (matching .sops.yaml regex)
|
|
|
|
|
# This is necessary because SOPS needs the file path to match creation rules
|
|
|
|
|
TEMP_SOPS="${SECRETS_FILE%.sops.yaml}-unenc.sops.yaml"
|
|
|
|
|
cat "$TEMP_PLAIN" > "$TEMP_SOPS"
|
|
|
|
|
|
|
|
|
|
# Encrypt in-place (SOPS finds creation rules because path matches regex)
|
|
|
|
|
sops --encrypt --in-place "$TEMP_SOPS"
|
|
|
|
|
|
|
|
|
|
# Rename to final name
|
|
|
|
|
mv "$TEMP_SOPS" "$SECRETS_FILE"
|
|
|
|
|
|
|
|
|
|
# Cleanup
|
|
|
|
|
rm "$TEMP_PLAIN"
|
|
|
|
|
|
|
|
|
|
echo -e "${GREEN}✓ Created secrets file with client-specific domains${NC}"
|
2026-01-17 20:04:29 +01:00
|
|
|
echo ""
|
|
|
|
|
|
2026-01-18 18:17:15 +01:00
|
|
|
# Open in SOPS for editing passwords
|
|
|
|
|
echo -e "${BLUE}Opening secrets file in SOPS for password generation...${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${YELLOW}IMPORTANT: Regenerate ALL passwords and tokens!${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Domains have been automatically configured:"
|
|
|
|
|
echo " ✓ client_name: $CLIENT_NAME"
|
|
|
|
|
echo " ✓ client_domain: ${CLIENT_NAME}.vrije.cloud"
|
|
|
|
|
echo " ✓ authentik_domain: auth.${CLIENT_NAME}.vrije.cloud"
|
|
|
|
|
echo " ✓ nextcloud_domain: nextcloud.${CLIENT_NAME}.vrije.cloud"
|
2026-01-17 20:04:29 +01:00
|
|
|
echo ""
|
2026-01-18 18:17:15 +01:00
|
|
|
echo "You MUST regenerate:"
|
|
|
|
|
echo " - All database passwords"
|
|
|
|
|
echo " - authentik_secret_key"
|
|
|
|
|
echo " - authentik_bootstrap_password"
|
|
|
|
|
echo " - authentik_bootstrap_token"
|
|
|
|
|
echo " - All other passwords"
|
2026-01-17 20:04:29 +01:00
|
|
|
echo ""
|
|
|
|
|
echo "Press Enter to open editor..."
|
|
|
|
|
read -r
|
|
|
|
|
|
|
|
|
|
# Open in SOPS
|
|
|
|
|
sops "$SECRETS_FILE"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Secrets file configured${NC}"
|
2026-01-08 16:56:19 +01:00
|
|
|
echo ""
|
|
|
|
|
fi
|
|
|
|
|
|
2026-01-18 18:17:15 +01:00
|
|
|
# Load Hetzner API token from SOPS if not already set
|
2026-01-08 16:56:19 +01:00
|
|
|
if [ -z "${HCLOUD_TOKEN:-}" ]; then
|
2026-01-18 18:17:15 +01:00
|
|
|
echo -e "${BLUE}Loading Hetzner API token from SOPS...${NC}"
|
|
|
|
|
# shellcheck source=scripts/load-secrets-env.sh
|
|
|
|
|
source "$SCRIPT_DIR/load-secrets-env.sh"
|
|
|
|
|
echo ""
|
2026-01-08 16:56:19 +01:00
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if [ -z "${SOPS_AGE_KEY_FILE:-}" ]; then
|
|
|
|
|
echo -e "${YELLOW}Warning: SOPS_AGE_KEY_FILE not set, using default${NC}"
|
|
|
|
|
export SOPS_AGE_KEY_FILE="$PROJECT_ROOT/keys/age-key.txt"
|
|
|
|
|
fi
|
|
|
|
|
|
2026-01-17 21:34:05 +01:00
|
|
|
# Check if client exists in terraform.tfvars
|
|
|
|
|
TFVARS_FILE="$PROJECT_ROOT/tofu/terraform.tfvars"
|
|
|
|
|
if ! grep -q "^[[:space:]]*${CLIENT_NAME}[[:space:]]*=" "$TFVARS_FILE"; then
|
|
|
|
|
echo -e "${RED}Error: Client '${CLIENT_NAME}' not found in terraform.tfvars${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Cannot rebuild a client that doesn't exist in OpenTofu configuration."
|
|
|
|
|
echo ""
|
|
|
|
|
echo "To deploy a new client, use:"
|
|
|
|
|
echo " ./scripts/deploy-client.sh $CLIENT_NAME"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
2026-01-08 16:56:19 +01:00
|
|
|
# Start timer
|
|
|
|
|
START_TIME=$(date +%s)
|
|
|
|
|
|
|
|
|
|
echo -e "${BLUE}========================================${NC}"
|
|
|
|
|
echo -e "${BLUE}Rebuilding client: $CLIENT_NAME${NC}"
|
|
|
|
|
echo -e "${BLUE}========================================${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Step 1: Check if infrastructure exists and destroy it
|
2026-01-17 20:24:53 +01:00
|
|
|
echo -e "${YELLOW}[1/5] Checking existing infrastructure...${NC}"
|
2026-01-08 16:56:19 +01:00
|
|
|
|
|
|
|
|
cd "$PROJECT_ROOT/tofu"
|
|
|
|
|
|
|
|
|
|
if tofu state list 2>/dev/null | grep -q "hcloud_server.client\[\"$CLIENT_NAME\"\]"; then
|
|
|
|
|
echo -e "${YELLOW}⚠ Existing infrastructure found${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
read -p "Destroy existing infrastructure? (yes/no): " destroy_confirm
|
|
|
|
|
|
|
|
|
|
if [ "$destroy_confirm" = "yes" ]; then
|
|
|
|
|
echo "Destroying existing infrastructure..."
|
|
|
|
|
"$SCRIPT_DIR/destroy-client.sh" "$CLIENT_NAME"
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Existing infrastructure destroyed${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Waiting 10 seconds for cleanup to complete..."
|
|
|
|
|
sleep 10
|
|
|
|
|
else
|
|
|
|
|
echo -e "${RED}Cannot proceed without destroying existing infrastructure${NC}"
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
|
|
|
|
else
|
|
|
|
|
echo -e "${GREEN}✓ No existing infrastructure found${NC}"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Step 2: Provision infrastructure
|
2026-01-17 20:24:53 +01:00
|
|
|
echo -e "${YELLOW}[2/5] Provisioning infrastructure with OpenTofu...${NC}"
|
2026-01-08 16:56:19 +01:00
|
|
|
|
|
|
|
|
cd "$PROJECT_ROOT/tofu"
|
|
|
|
|
|
|
|
|
|
# Apply full configuration to create server AND DNS records
|
|
|
|
|
tofu apply -auto-approve -var-file="terraform.tfvars"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Infrastructure provisioned (server + DNS)${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Wait for server to be ready
|
|
|
|
|
echo -e "${YELLOW}Waiting 60 seconds for server to initialize...${NC}"
|
|
|
|
|
sleep 60
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Step 3: Setup base system
|
2026-01-17 20:24:53 +01:00
|
|
|
echo -e "${YELLOW}[3/5] Setting up base system (Docker, Traefik)...${NC}"
|
2026-01-08 16:56:19 +01:00
|
|
|
|
|
|
|
|
cd "$PROJECT_ROOT/ansible"
|
|
|
|
|
|
|
|
|
|
~/.local/bin/ansible-playbook -i hcloud.yml playbooks/setup.yml --limit "$CLIENT_NAME"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Base system configured${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
|
|
|
|
|
# Step 4: Deploy applications
|
2026-01-17 20:24:53 +01:00
|
|
|
echo -e "${YELLOW}[4/5] Deploying applications (Authentik, Nextcloud, SSO)...${NC}"
|
2026-01-08 16:56:19 +01:00
|
|
|
|
|
|
|
|
~/.local/bin/ansible-playbook -i hcloud.yml playbooks/deploy.yml --limit "$CLIENT_NAME"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Applications deployed${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
|
2026-01-17 20:24:53 +01:00
|
|
|
# Step 5: Update client registry
|
|
|
|
|
echo -e "${YELLOW}[5/5] Updating client registry...${NC}"
|
|
|
|
|
|
|
|
|
|
cd "$PROJECT_ROOT/tofu"
|
|
|
|
|
|
|
|
|
|
# Get server information from Terraform state
|
|
|
|
|
SERVER_IP=$(tofu output -json client_ips 2>/dev/null | jq -r ".\"$CLIENT_NAME\"" || echo "")
|
|
|
|
|
SERVER_ID=$(tofu state show "hcloud_server.client[\"$CLIENT_NAME\"]" 2>/dev/null | grep "^[[:space:]]*id[[:space:]]*=" | awk '{print $3}' | tr -d '"' || echo "")
|
|
|
|
|
SERVER_TYPE=$(tofu state show "hcloud_server.client[\"$CLIENT_NAME\"]" 2>/dev/null | grep "^[[:space:]]*server_type[[:space:]]*=" | awk '{print $3}' | tr -d '"' || echo "")
|
|
|
|
|
SERVER_LOCATION=$(tofu state show "hcloud_server.client[\"$CLIENT_NAME\"]" 2>/dev/null | grep "^[[:space:]]*location[[:space:]]*=" | awk '{print $3}' | tr -d '"' || echo "")
|
|
|
|
|
|
|
|
|
|
# Determine role (dev is canary, everything else is production by default)
|
|
|
|
|
ROLE="production"
|
|
|
|
|
if [ "$CLIENT_NAME" = "dev" ]; then
|
|
|
|
|
ROLE="canary"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# Update registry
|
|
|
|
|
"$SCRIPT_DIR/update-registry.sh" "$CLIENT_NAME" deploy \
|
|
|
|
|
--role="$ROLE" \
|
|
|
|
|
--server-ip="$SERVER_IP" \
|
|
|
|
|
--server-id="$SERVER_ID" \
|
|
|
|
|
--server-type="$SERVER_TYPE" \
|
|
|
|
|
--server-location="$SERVER_LOCATION"
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}✓ Registry updated${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
|
2026-01-17 20:53:15 +01:00
|
|
|
# Collect deployed versions
|
|
|
|
|
echo -e "${YELLOW}Collecting deployed versions...${NC}"
|
|
|
|
|
|
|
|
|
|
"$SCRIPT_DIR/collect-client-versions.sh" "$CLIENT_NAME" 2>/dev/null || {
|
|
|
|
|
echo -e "${YELLOW}⚠ Could not collect versions automatically${NC}"
|
|
|
|
|
echo "Run manually later: ./scripts/collect-client-versions.sh $CLIENT_NAME"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
echo ""
|
|
|
|
|
|
2026-01-08 16:56:19 +01:00
|
|
|
# Calculate duration
|
|
|
|
|
END_TIME=$(date +%s)
|
|
|
|
|
DURATION=$((END_TIME - START_TIME))
|
|
|
|
|
MINUTES=$((DURATION / 60))
|
|
|
|
|
SECONDS=$((DURATION % 60))
|
|
|
|
|
|
|
|
|
|
# Success summary
|
|
|
|
|
echo -e "${GREEN}========================================${NC}"
|
|
|
|
|
echo -e "${GREEN}✓ Rebuild complete!${NC}"
|
|
|
|
|
echo -e "${GREEN}========================================${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${BLUE}Time taken: ${MINUTES}m ${SECONDS}s${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Services deployed:"
|
|
|
|
|
|
|
|
|
|
# Load client domain from secrets
|
|
|
|
|
CLIENT_DOMAIN=$(sops -d "$SECRETS_FILE" | grep "^client_domain:" | awk '{print $2}')
|
|
|
|
|
AUTHENTIK_DOMAIN=$(sops -d "$SECRETS_FILE" | grep "^authentik_domain:" | awk '{print $2}')
|
|
|
|
|
NEXTCLOUD_DOMAIN=$(sops -d "$SECRETS_FILE" | grep "^nextcloud_domain:" | awk '{print $2}')
|
|
|
|
|
|
|
|
|
|
echo " ✓ Authentik SSO: https://$AUTHENTIK_DOMAIN"
|
|
|
|
|
echo " ✓ Nextcloud: https://$NEXTCLOUD_DOMAIN"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Admin credentials:"
|
|
|
|
|
echo " Authentik: akadmin / (see secrets file)"
|
|
|
|
|
echo " Nextcloud: admin / (see secrets file)"
|
|
|
|
|
echo ""
|
|
|
|
|
echo -e "${GREEN}Ready to use! No manual configuration required.${NC}"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "To view secrets:"
|
|
|
|
|
echo " sops $SECRETS_FILE"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "To destroy this client:"
|
|
|
|
|
echo " ./scripts/destroy-client.sh $CLIENT_NAME"
|
|
|
|
|
echo ""
|