| | |
| | |
| |
|
| | user nginx; |
| | worker_processes auto; |
| | error_log /var/log/nginx/error.log warn; |
| | pid /var/run/nginx.pid; |
| |
|
| | |
| | load_module modules/ngx_http_image_filter_module.so; |
| |
|
| | events { |
| | worker_connections 4096; |
| | use epoll; |
| | multi_accept on; |
| | } |
| |
|
| | http { |
| | include /etc/nginx/mime.types; |
| | default_type application/octet-stream; |
| |
|
| | |
| | log_format main '$remote_addr - $remote_user [$time_local] "$request" ' |
| | '$status $body_bytes_sent "$http_referer" ' |
| | '"$http_user_agent" "$http_x_forwarded_for" ' |
| | 'rt=$request_time uct="$upstream_connect_time" ' |
| | 'uht="$upstream_header_time" urt="$upstream_response_time"'; |
| |
|
| | access_log /var/log/nginx/access.log main buffer=32k flush=5s; |
| |
|
| | |
| | sendfile on; |
| | tcp_nopush on; |
| | tcp_nodelay on; |
| | keepalive_timeout 65; |
| | types_hash_max_size 2048; |
| | server_tokens off; |
| | client_max_body_size 100M; |
| |
|
| | |
| | ssl_protocols TLSv1.2 TLSv1.3; |
| | ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-AES256-GCM-SHA384:DHE-RSA-AES128-GCM-SHA256:DHE-DSS-AES128-GCM-SHA256:kEDH+AESGCM:ECDHE-RSA-AES128-SHA256:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA:ECDHE-ECDSA-AES128-SHA:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA:ECDHE-ECDSA-AES256-SHA:DHE-RSA-AES128-SHA256:DHE-RSA-AES128-SHA:DHE-DSS-AES128-SHA256:DHE-RSA-AES256-SHA256:DHE-DSS-AES256-SHA:DHE-RSA-AES256-SHA:!aNULL:!eNULL:!EXPORT:!DES:!RC4:!3DES:!MD5:!PSK; |
| | ssl_prefer_server_ciphers on; |
| | ssl_session_cache shared:SSL:50m; |
| | ssl_session_timeout 1d; |
| | ssl_stapling on; |
| | ssl_stapling_verify on; |
| |
|
| | |
| | gzip on; |
| | gzip_vary on; |
| | gzip_proxied any; |
| | gzip_comp_level 6; |
| | gzip_types text/plain text/css text/xml text/javascript |
| | application/json application/javascript application/xml+rss |
| | application/rss+xml application/atom+xml image/svg+xml |
| | text/x-js text/x-cross-domain-policy application/x-font-ttf |
| | application/x-font-opentype application/vnd.ms-fontobject |
| | image/x-icon; |
| |
|
| | |
| | proxy_cache_path /var/cache/nginx/proxy levels=1:2 keys_zone=proxy_cache:100m |
| | max_size=10g inactive=60m use_temp_path=off; |
| | |
| | proxy_cache_path /var/cache/nginx/images levels=1:2 keys_zone=images:100m |
| | max_size=50g inactive=30d use_temp_path=off; |
| |
|
| | |
| | limit_req_zone $binary_remote_addr zone=api:10m rate=10r/s; |
| | limit_req_zone $binary_remote_addr zone=upload:10m rate=1r/s; |
| | limit_conn_zone $binary_remote_addr zone=addr:10m; |
| |
|
| | |
| | upstream api_backend { |
| | least_conn; |
| | server api1:8000 max_fails=3 fail_timeout=30s; |
| | server api2:8000 max_fails=3 fail_timeout=30s backup; |
| | keepalive 32; |
| | } |
| |
|
| | upstream web_backend { |
| | least_conn; |
| | server web1:3000 max_fails=3 fail_timeout=30s; |
| | server web2:3000 max_fails=3 fail_timeout=30s backup; |
| | keepalive 32; |
| | } |
| |
|
| | upstream websocket_backend { |
| | ip_hash; |
| | server ws1:8001; |
| | server ws2:8001; |
| | } |
| |
|
| | |
| | server { |
| | listen 80; |
| | listen [::]:80; |
| | server_name backgroundfx.pro www.backgroundfx.pro; |
| | return 301 https://$server_name$request_uri; |
| | } |
| | |
| | # Main HTTPS Server |
| | server { |
| | listen 443 ssl http2; |
| | listen [::]:443 ssl http2; |
| | server_name backgroundfx.pro www.backgroundfx.pro; |
| | |
| | # SSL Certificate |
| | ssl_certificate /etc/nginx/ssl/fullchain.pem; |
| | ssl_certificate_key /etc/nginx/ssl/privkey.pem; |
| | ssl_trusted_certificate /etc/nginx/ssl/chain.pem; |
| | |
| | # 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; |
| | add_header Content-Security-Policy "default-src 'self' https:; script-src 'self' 'unsafe-inline' 'unsafe-eval' https://cdn.jsdelivr.net https://unpkg.com; style-src 'self' 'unsafe-inline' https://fonts.googleapis.com; font-src 'self' https://fonts.gstatic.com; img-src 'self' data: https:; connect-src 'self' https://api.backgroundfx.pro wss://ws.backgroundfx.pro" always; |
| | add_header Strict-Transport-Security "max-age=31536000; includeSubDomains; preload" always; |
| | |
| | # Root and Index |
| | root /var/www/html; |
| | index index.html; |
| |
|
| | |
| | location /api/ { |
| | limit_req zone=api burst=20 nodelay; |
| | limit_conn addr 10; |
| |
|
| | proxy_pass http://api_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 ""; |
| | |
| | # Timeouts |
| | proxy_connect_timeout 60s; |
| | proxy_send_timeout 60s; |
| | proxy_read_timeout 60s; |
| | |
| | # Buffering |
| | proxy_buffering on; |
| | proxy_buffer_size 4k; |
| | proxy_buffers 8 4k; |
| | proxy_busy_buffers_size 8k; |
| | |
| | # Cache for GET requests |
| | proxy_cache proxy_cache; |
| | proxy_cache_valid 200 1m; |
| | proxy_cache_valid 404 1m; |
| | proxy_cache_use_stale error timeout http_500 http_502 http_503 http_504; |
| | proxy_cache_bypass $http_cache_control; |
| | add_header X-Cache-Status $upstream_cache_status; |
| | } |
| | |
| | # Upload Endpoint |
| | location /api/upload { |
| | limit_req zone=upload burst=5 nodelay; |
| | client_max_body_size 100M; |
| | client_body_buffer_size 1M; |
| | |
| | proxy_pass http://api_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; |
| | |
| | # Longer timeout for uploads |
| | proxy_connect_timeout 120s; |
| | proxy_send_timeout 300s; |
| | proxy_read_timeout 300s; |
| | } |
| | |
| | # WebSocket Endpoint |
| | location /ws/ { |
| | proxy_pass http://websocket_backend; |
| | 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; |
| | } |
| | |
| | # Static Files |
| | location /static/ { |
| | alias /var/www/static/; |
| | expires 30d; |
| | add_header Cache-Control "public, immutable"; |
| | |
| | # Enable gzip for static files |
| | gzip_static on; |
| | |
| | # CORS for fonts |
| | location ~* \.(eot|ttf|woff|woff2)$ { |
| | add_header Access-Control-Allow-Origin *; |
| | expires 30d; |
| | } |
| | } |
| | |
| | # Processed Images with Caching |
| | location /images/ { |
| | alias /var/www/images/; |
| | expires 7d; |
| | add_header Cache-Control "public, max-age=604800"; |
| | |
| | # Image processing on the fly |
| | location ~ ^/images/resize/(\d+)x(\d+)/(.+)$ { |
| | alias /var/www/images/$3; |
| | image_filter resize $1 $2; |
| | image_filter_jpeg_quality 85; |
| | image_filter_buffer 20M; |
| | error_page 415 = /empty; |
| | } |
| | |
| | # Cache processed images |
| | proxy_cache images; |
| | proxy_cache_valid 200 30d; |
| | proxy_cache_valid 404 1h; |
| | } |
| | |
| | # Health Check |
| | location /health { |
| | access_log off; |
| | return 200 "healthy\n"; |
| | add_header Content-Type text/plain; |
| | } |
| |
|
| | |
| | location /metrics { |
| | allow 10.0.0.0/8; |
| | deny all; |
| | proxy_pass http://api_backend/metrics; |
| | } |
| |
|
| | |
| | location / { |
| | proxy_pass http://web_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 ""; |
| | |
| | # Next.js specific |
| | proxy_set_header X-Forwarded-Host $host; |
| | proxy_set_header X-Forwarded-Port $server_port; |
| | } |
| | |
| | # Error Pages |
| | error_page 404 /404.html; |
| | error_page 500 502 503 504 /50x.html; |
| | |
| | location = /404.html { |
| | root /var/www/error; |
| | internal; |
| | } |
| | |
| | location = /50x.html { |
| | root /var/www/error; |
| | internal; |
| | } |
| | |
| | location = /empty { |
| | empty_gif; |
| | } |
| | } |
| | |
| | # CDN Server |
| | server { |
| | listen 443 ssl http2; |
| | listen [::]:443 ssl http2; |
| | server_name cdn.backgroundfx.pro; |
| | |
| | ssl_certificate /etc/nginx/ssl/cdn-fullchain.pem; |
| | ssl_certificate_key /etc/nginx/ssl/cdn-privkey.pem; |
| | |
| | root /var/www/cdn; |
| |
|
| | location / { |
| | expires 1y; |
| | add_header Cache-Control "public, immutable"; |
| | add_header Access-Control-Allow-Origin "*"; |
| | |
| | |
| | gzip_static on; |
| | |
| | |
| | gzip_types *; |
| | } |
| | } |
| | } |