server { listen 80; server_name _; root /usr/share/nginx/html; index index.html; # Disable sendfile for Docker sendfile off; tcp_nopush off; tcp_nodelay on; # Increase buffer sizes to prevent content-length mismatch client_body_buffer_size 128k; client_max_body_size 10m; client_header_buffer_size 1k; large_client_header_buffers 4 8k; output_buffers 32 32k; postpone_output 1460; # Gzip compression - disabled for js/css to prevent content-length issues gzip on; gzip_vary on; gzip_min_length 1024; gzip_proxied any; gzip_comp_level 3; gzip_types text/plain text/xml application/xml+rss application/json; gzip_disable "msie6"; # 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; # Cache static assets (images, fonts) location ~* \.(jpg|jpeg|png|gif|ico|svg|woff|woff2|ttf|eot)$ { expires 1y; add_header Cache-Control "public, immutable"; } # JS and CSS - no compression, short cache location ~* \.(js|css)$ { expires 1h; add_header Cache-Control "public, no-transform"; gzip off; } # Service Worker - never cache location = /sw.js { add_header Cache-Control "no-cache, no-store, must-revalidate"; expires 0; } # API proxy - forward backend API requests to the backend service location /api { proxy_pass http://backend:5000; 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; proxy_cache_bypass $http_upgrade; } # API proxy for subpath deployment (/drohnenfuehrer/api) location /drohnenfuehrer/api/ { proxy_pass http://backend:5000/api/; 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; proxy_cache_bypass $http_upgrade; } # Service Worker for subpath deployment location = /drohnenfuehrer/sw.js { add_header Cache-Control "no-cache, no-store, must-revalidate"; expires 0; alias /usr/share/nginx/html/sw.js; } # index.html - never cache so new builds are picked up immediately location = /index.html { add_header Cache-Control "no-cache, no-store, must-revalidate"; add_header Pragma "no-cache"; expires 0; } # SPA fallback for subpath deployment (/drohnenfuehrer) location ^~ /drohnenfuehrer/ { rewrite ^/drohnenfuehrer(/.*)$ $1 break; try_files $uri $uri/ /index.html; add_header Cache-Control "no-cache, no-store, must-revalidate"; add_header Pragma "no-cache"; expires 0; } # SPA fallback - serve index.html for all routes location / { try_files $uri $uri/ /index.html; add_header Cache-Control "no-cache, no-store, must-revalidate"; add_header Pragma "no-cache"; expires 0; } # Health check endpoint location /health { access_log off; return 200 "OK\n"; add_header Content-Type text/plain; } }