#!/bin/bash # Comprehensive setup script for Hetzner server with Docker, user setup, Node Exporter, and OpenTelemetry # Exit on error set -e echo "=====================================================" echo "🚀 STARTING SERVER SETUP" echo "=====================================================" # Load environment variables from .env.setup if present ENV_FILE="$(dirname "$0")/.env.setup" if [ -f "$ENV_FILE" ]; then echo "📂 Loading environment from $ENV_FILE" set -a # shellcheck source=/dev/null source "$ENV_FILE" set +a else echo "â„šī¸ No .env.setup file found, using environment variables" fi # Verify required environment variables if [ -z "$OTEL_EXPORTER_OTLP_ENDPOINT" ] || [ -z "$OTEL_AUTH_HEADER" ]; then echo "❌ ERROR: Required environment variables are not set!" echo "Please set OTEL_EXPORTER_OTLP_ENDPOINT and OTEL_AUTH_HEADER in .env.setup or the environment" exit 1 fi # CF_ORIGIN_CERT and CF_ORIGIN_KEY: Cloudflare Origin Certificate and private key. # Generate at: Cloudflare dashboard → SSL/TLS → Origin Server → Create Certificate if [ -z "$CF_ORIGIN_CERT" ] || [ -z "$CF_ORIGIN_KEY" ]; then echo "❌ ERROR: CF_ORIGIN_CERT and CF_ORIGIN_KEY are not set!" echo "Generate an origin certificate at: Cloudflare → SSL/TLS → Origin Server → Create Certificate" echo "Then add CF_ORIGIN_CERT and CF_ORIGIN_KEY to .env.setup" exit 1 fi echo "🔄 Updating system..." apt update && apt upgrade -y # Check if Docker is already installed if command -v docker &> /dev/null; then echo "Docker is already installed" else echo "đŸŗ Installing Docker..." # Install Docker using official script curl -fsSL https://get.docker.com -o get-docker.sh sh get-docker.sh systemctl enable --now docker echo "Docker installed successfully" fi echo "👤 Setting up openfront user..." # Create openfront user if it doesn't exist if id "openfront" &> /dev/null; then echo "User openfront already exists" else useradd -m -s /bin/bash openfront echo "User openfront created" fi # Check if openfront is already in docker group if groups openfront | grep -q '\bdocker\b'; then echo "User openfront is already in the docker group" else # Add openfront to docker group usermod -aG docker openfront echo "Added openfront to docker group" fi # Create .ssh directory for openfront if it doesn't exist if [ ! -d "/home/openfront/.ssh" ]; then mkdir -p /home/openfront/.ssh chmod 700 /home/openfront/.ssh echo "Created .ssh directory for openfront" fi # Copy SSH keys from root if they exist and haven't been copied yet if [ -f /root/.ssh/authorized_keys ] && [ ! -f /home/openfront/.ssh/authorized_keys ]; then cp /root/.ssh/authorized_keys /home/openfront/.ssh/ chmod 600 /home/openfront/.ssh/authorized_keys echo "SSH keys copied from root to openfront" fi # Configure UDP buffer sizes for Cloudflare Tunnel # https://github.com/quic-go/quic-go/wiki/UDP-Buffer-Sizes echo "🔧 Configuring UDP buffer sizes..." # Check if settings already exist in sysctl.conf if grep -q "net.core.rmem_max" /etc/sysctl.conf && grep -q "net.core.wmem_max" /etc/sysctl.conf; then echo "UDP buffer size settings already configured" else # Add UDP buffer size settings to sysctl.conf echo "# UDP buffer size settings for improved QUIC performance" >> /etc/sysctl.conf echo "net.core.rmem_max=7500000" >> /etc/sysctl.conf echo "net.core.wmem_max=7500000" >> /etc/sysctl.conf # Apply the settings immediately sysctl -p echo "UDP buffer sizes configured and applied" fi # Set proper ownership for openfront's home directory chown -R openfront:openfront /home/openfront echo "Set proper ownership for openfront's home directory" # Set up Traefik reverse proxy echo "🔀 Setting up Traefik..." # Create the shared Docker network used by Traefik and app containers if docker network ls --format '{{.Name}}' | grep -q '^web$'; then echo "Docker network 'web' already exists" else docker network create web echo "Created Docker network 'web'" fi TRAEFIK_CONFIG_DIR="/home/openfront/traefik" TRAEFIK_CERTS_DIR="$TRAEFIK_CONFIG_DIR/certs" mkdir -p "$TRAEFIK_CERTS_DIR" # Write Cloudflare origin certificate and key (passed as env vars) echo "$CF_ORIGIN_CERT" > "$TRAEFIK_CERTS_DIR/origin.crt" echo "$CF_ORIGIN_KEY" > "$TRAEFIK_CERTS_DIR/origin.key" chmod 600 "$TRAEFIK_CERTS_DIR/origin.crt" "$TRAEFIK_CERTS_DIR/origin.key" # No [api] block — dashboard is disabled for production. # To access it for debugging, SSH tunnel: ssh -L 8080:localhost:8080 user@server cat > "$TRAEFIK_CONFIG_DIR/traefik.toml" << 'EOF' [log] level = "INFO" [entryPoints] [entryPoints.web] address = ":80" [entryPoints.websecure] address = ":443" [providers] [providers.docker] endpoint = "unix:///var/run/docker.sock" exposedByDefault = false # Only route containers with traefik.enable=true network = "web" watch = true [providers.file] filename = "/etc/traefik/tls.toml" watch = true EOF # Static TLS configuration referencing the Cloudflare origin cert cat > "$TRAEFIK_CONFIG_DIR/tls.toml" << 'EOF' [[tls.certificates]] certFile = "/certs/origin.crt" keyFile = "/certs/origin.key" [tls.options] [tls.options.default] minVersion = "VersionTLS12" EOF cat > "$TRAEFIK_CONFIG_DIR/compose.yaml" << 'EOF' networks: web: # External so blue/green containers can join independently. external: true services: traefik: image: traefik:v3.6 container_name: traefik restart: unless-stopped ports: - "80:80" - "443:443" volumes: - /var/run/docker.sock:/var/run/docker.sock:ro - /home/openfront/traefik/traefik.toml:/etc/traefik/traefik.toml:ro - /home/openfront/traefik/tls.toml:/etc/traefik/tls.toml:ro - /home/openfront/traefik/certs:/certs:ro networks: - web EOF chown -R openfront:openfront "$TRAEFIK_CONFIG_DIR" docker compose -f "$TRAEFIK_CONFIG_DIR/compose.yaml" pull docker compose -f "$TRAEFIK_CONFIG_DIR/compose.yaml" up -d if docker ps | grep -q traefik; then echo "✅ Traefik started successfully!" else echo "❌ Failed to start Traefik. Check logs with: docker logs traefik" exit 1 fi # Create directory for OpenTelemetry configuration echo "📊 Setting up Node Exporter and OpenTelemetry Collector..." OTEL_CONFIG_DIR="/home/openfront/otel" if [ ! -d "$OTEL_CONFIG_DIR" ]; then mkdir -p "$OTEL_CONFIG_DIR" echo "Created OpenTelemetry configuration directory" fi # Create OpenTelemetry Collector configuration cat > "$OTEL_CONFIG_DIR/otel-collector-config.yaml" << EOF receivers: prometheus: config: scrape_configs: - job_name: 'node' scrape_interval: 10s static_configs: - targets: ['localhost:9100'] # Node Exporter endpoint relabel_configs: - source_labels: [__address__] regex: '.*' target_label: openfront.host replacement: "\${HOSTNAME}" processors: batch: # Batch metrics before sending timeout: 10s send_batch_size: 1000 exporters: otlphttp: endpoint: "${OTEL_EXPORTER_OTLP_ENDPOINT}" headers: Authorization: "${OTEL_AUTH_HEADER}" tls: insecure: true # Set to false in production with proper certs service: pipelines: metrics: receivers: [prometheus] processors: [batch] exporters: [otlphttp] EOF # Set ownership of all files chmod 600 "$OTEL_CONFIG_DIR/otel-collector-config.yaml" chown -R openfront:openfront "$OTEL_CONFIG_DIR" # Run Node Exporter echo "🚀 Starting Node Exporter..." docker pull prom/node-exporter:latest docker rm -f node-exporter 2> /dev/null || true docker run -d \ --name=node-exporter \ --restart=unless-stopped \ --net="host" \ --pid="host" \ -v "/:/host:ro,rslave" \ prom/node-exporter:latest \ --path.rootfs=/host # Run OpenTelemetry Collector echo "🚀 Starting OpenTelemetry Collector..." docker pull otel/opentelemetry-collector-contrib:latest docker rm -f otel-collector 2> /dev/null || true # Run OpenTelemetry Collector with appropriate permissions echo "🚀 Starting OpenTelemetry Collector..." docker pull otel/opentelemetry-collector-contrib:latest docker rm -f otel-collector 2> /dev/null || true docker run -d \ --name=otel-collector \ --restart=unless-stopped \ --network=host \ --user=0 \ -v "$OTEL_CONFIG_DIR/otel-collector-config.yaml:/etc/otelcol-contrib/config.yaml:ro" \ otel/opentelemetry-collector-contrib:latest # Check if containers are running if docker ps | grep -q node-exporter && docker ps | grep -q otel-collector; then echo "✅ Node Exporter and OpenTelemetry Collector started successfully!" else echo "❌ Failed to start containers. Check logs with: docker logs node-exporter or docker logs otel-collector" exit 1 fi echo "=====================================================" echo "🎉 SETUP COMPLETE!" echo "=====================================================" echo "The openfront user has been set up and has Docker permissions." echo "UDP buffer sizes have been configured for optimal QUIC/WebSocket performance." echo "Traefik reverse proxy is running (HTTP :80, HTTPS :443 with Cloudflare origin cert)." echo "Node Exporter is collecting system metrics." echo "OpenTelemetry Collector is forwarding metrics to your endpoint." echo "" echo "📝 Configuration:" echo " - Traefik Config: $TRAEFIK_CONFIG_DIR" echo " - OTEL Config Directory: $OTEL_CONFIG_DIR" echo " - OpenTelemetry Endpoint: $OTEL_EXPORTER_OTLP_ENDPOINT" echo "====================================================="