# Gateway - Nginx reverse proxy for unified deployment # Routes traffic between frontend and backend under single vhost FROM nginx:alpine # Install envsubst for runtime configuration RUN apk add --no-cache gettext # Copy nginx configuration template COPY <<'EOF' /etc/nginx/templates/default.conf.template # Upstream servers upstream backend { server ${BACKEND_HOST}:${BACKEND_PORT}; keepalive 32; } upstream frontend { server ${FRONTEND_HOST}:${FRONTEND_PORT}; keepalive 32; } # Rate limiting zones limit_req_zone $binary_remote_addr zone=api_limit:10m rate=10r/s; limit_req_zone $binary_remote_addr zone=download_limit:10m rate=5r/s; # Cache configuration proxy_cache_path /var/cache/nginx/static levels=1:2 keys_zone=static_cache:10m max_size=100m inactive=60m use_temp_path=off; server { listen 80; server_name ${SERVER_NAME}; # Security headers add_header X-Frame-Options "SAMEORIGIN" always; add_header X-Content-Type-Options "nosniff" always; add_header X-XSS-Protection "1; mode=block" always; add_header Referrer-Policy "strict-origin-when-cross-origin" always; # Client body size for package uploads client_max_body_size 500M; client_body_timeout 300s; # Logging access_log /var/log/nginx/access.log combined; error_log /var/log/nginx/error.log warn; # API endpoints - proxy to backend location /api/ { # Rate limiting limit_req zone=api_limit burst=20 nodelay; # Proxy settings proxy_pass http://backend/; proxy_http_version 1.1; # Headers proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-Proto $scheme; proxy_set_header X-Forwarded-Host $host; proxy_set_header X-Forwarded-Port $server_port; # Connection reuse proxy_set_header Connection ""; # Timeouts for long-running operations proxy_connect_timeout 60s; proxy_send_timeout 300s; proxy_read_timeout 300s; # Buffer settings proxy_buffering on; proxy_buffer_size 4k; proxy_buffers 8 4k; proxy_busy_buffers_size 8k; } # Health check endpoint location /health { proxy_pass http://backend/health; proxy_http_version 1.1; proxy_set_header Connection ""; access_log off; } # Metrics endpoint (optional - may want to restrict access) location /metrics { # Uncomment to restrict to internal networks # allow 10.0.0.0/8; # allow 172.16.0.0/12; # allow 192.168.0.0/16; # deny all; proxy_pass http://backend/metrics; proxy_http_version 1.1; proxy_set_header Connection ""; } # Package download endpoints with rate limiting location ~ ^/(npm|pypi|go)/ { limit_req zone=download_limit burst=10 nodelay; proxy_pass http://backend; proxy_http_version 1.1; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-Proto $scheme; proxy_set_header Connection ""; # Extended timeouts for package downloads proxy_connect_timeout 60s; proxy_send_timeout 600s; proxy_read_timeout 600s; # Large buffer for package downloads proxy_buffering on; proxy_buffer_size 128k; proxy_buffers 4 256k; proxy_busy_buffers_size 256k; } # Frontend - serve SPA location / { proxy_pass http://frontend; proxy_http_version 1.1; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-Proto $scheme; proxy_set_header Connection ""; # Cache static assets proxy_cache static_cache; proxy_cache_valid 200 1h; proxy_cache_bypass $http_cache_control; add_header X-Cache-Status $upstream_cache_status; } # WebSocket support (if needed for future features) location /ws/ { proxy_pass http://backend/ws/; proxy_http_version 1.1; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection "upgrade"; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-Proto $scheme; # WebSocket timeouts proxy_connect_timeout 7d; proxy_send_timeout 7d; proxy_read_timeout 7d; } } # HTTPS server (uncomment and configure SSL certificates) # server { # listen 443 ssl http2; # server_name ${SERVER_NAME}; # # ssl_certificate /etc/nginx/ssl/cert.pem; # ssl_certificate_key /etc/nginx/ssl/key.pem; # # # SSL configuration # ssl_protocols TLSv1.2 TLSv1.3; # ssl_ciphers HIGH:!aNULL:!MD5; # ssl_prefer_server_ciphers on; # ssl_session_cache shared:SSL:10m; # ssl_session_timeout 10m; # # # Include all location blocks from above # # ... (copy from HTTP server block) # } EOF # Create cache directory RUN mkdir -p /var/cache/nginx/static && \ chown -R nginx:nginx /var/cache/nginx # Expose port EXPOSE 80 443 # Health check HEALTHCHECK --interval=30s --timeout=10s --start-period=5s --retries=3 \ CMD wget --quiet --tries=1 --spider http://localhost/health || exit 1 # Environment variables with defaults ENV BACKEND_HOST=gohoarder-server \ BACKEND_PORT=8080 \ FRONTEND_HOST=gohoarder-frontend \ FRONTEND_PORT=80 \ SERVER_NAME=_ # Use nginx with template substitution CMD ["/bin/sh", "-c", "envsubst '$$BACKEND_HOST $$BACKEND_PORT $$FRONTEND_HOST $$FRONTEND_PORT $$SERVER_NAME' < /etc/nginx/templates/default.conf.template > /etc/nginx/conf.d/default.conf && nginx -g 'daemon off;'"]