feat: combine Admin UI into single container with nginx proxy
Some checks failed
CI / Scanner Lint (push) Has been cancelled
CI / Scanner Tests (push) Has been cancelled
CI / Banner Lint & Typecheck (push) Has been cancelled
CI / Banner Tests (push) Has been cancelled
CI / Banner Build (push) Has been cancelled
CI / Admin UI Typecheck (push) Has been cancelled
CI / Admin UI Tests (push) Has been cancelled
CI / Detect changes (push) Has been cancelled
CI / API Lint (push) Has been cancelled
CI / API Tests (push) Has been cancelled
CI / Admin UI Build (push) Has been cancelled

This commit is contained in:
Ami
2026-04-21 11:30:56 +07:00
parent f195a44707
commit f8cdbf8d74
3 changed files with 74 additions and 39 deletions

View File

@@ -1,4 +1,4 @@
# ── Build stage ────────────────────────────────────────────────────── # ── Build stage: Python deps ────────────────────────────────────────────
FROM python:3.12-slim AS builder FROM python:3.12-slim AS builder
WORKDIR /build WORKDIR /build
@@ -7,48 +7,61 @@ RUN apt-get update && apt-get install -y --no-install-recommends \
gcc libpq-dev curl \ gcc libpq-dev curl \
&& rm -rf /var/lib/apt/lists/* && rm -rf /var/lib/apt/lists/*
# Copy pyproject.toml for both api and scanner
COPY apps/api/pyproject.toml ./api/pyproject.toml COPY apps/api/pyproject.toml ./api/pyproject.toml
COPY apps/scanner/pyproject.toml ./scanner/pyproject.toml COPY apps/scanner/pyproject.toml ./scanner/pyproject.toml
# Install API dependencies
RUN pip install --no-cache-dir --prefix=/install api/. RUN pip install --no-cache-dir --prefix=/install api/.
# Install Scanner dependencies (Playwright + Chromium)
# PYTHONPATH needed because --prefix=/install doesn't auto-set site-packages path
RUN pip install --no-cache-dir --prefix=/install scanner/. \ RUN pip install --no-cache-dir --prefix=/install scanner/. \
&& PYTHONPATH=/install/lib/python3.12/site-packages \ && PYTHONPATH=/install/lib/python3.12/site-packages \
/install/bin/playwright install chromium --with-deps /install/bin/playwright install chromium --with-deps
# ── Runtime stage ──────────────────────────────────────────────────── # ── Build stage: banner bundle ─────────────────────────────────────────
FROM node:20-slim AS banner-builder
WORKDIR /build/banner
COPY apps/banner/package.json apps/banner/package-lock.json ./
RUN npm ci
COPY apps/banner/ .
RUN npm run build
# ── Build stage: admin UI ──────────────────────────────────────────────
FROM node:20-slim AS admin-builder
WORKDIR /build/admin
COPY apps/admin-ui/package.json apps/admin-ui/package-lock.json ./
RUN npm ci
COPY apps/admin-ui/ .
COPY --from=banner-builder /build/banner/dist/ ./public/
RUN npx vite build
# ── Runtime stage ──────────────────────────────────────────────────────
FROM python:3.12-slim FROM python:3.12-slim
WORKDIR /app WORKDIR /app
RUN apt-get update && apt-get install -y --no-install-recommends \ RUN apt-get update && apt-get install -y --no-install-recommends \
libpq5 curl tini supervisor \ libpq5 curl tini supervisor nginx \
&& rm -rf /var/lib/apt/lists/* \ && rm -rf /var/lib/apt/lists/* \
&& apt-get clean && apt-get clean
# Copy installed dependencies from builder # Copy Python deps from builder
COPY --from=builder /install /usr/local COPY --from=builder /install /usr/local
# Copy application code # Copy application code
COPY apps/api/src ./src COPY apps/api/src ./src
COPY apps/scanner/src ./src_scanner COPY apps/scanner/src ./src_scanner
COPY supervisord.conf /etc/supervisord.conf
# Move scanner source into api structure
RUN if [ -d src_scanner ]; then \ RUN if [ -d src_scanner ]; then \
cp -r src_scanner/* src/ 2>/dev/null || true; \ cp -r src_scanner/* src/ 2>/dev/null || true; \
fi fi
# Healthcheck for API # Copy built Admin UI static files
HEALTHCHECK --interval=30s --timeout=5s --start-period=30s --retries=3 \ COPY --from=admin-builder /build/admin/dist /var/www/html
CMD curl -f http://localhost:8000/health || exit 1
# Copy configs
COPY apps/admin-ui/nginx.conf /etc/nginx/conf.d/default.conf
COPY supervisord.conf /etc/supervisord.conf
HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \
CMD curl -f http://localhost/health || exit 1
# Use tini as init system for proper signal handling
ENTRYPOINT ["/usr/bin/tini", "--"] ENTRYPOINT ["/usr/bin/tini", "--"]
# supervisord manages multiple processes
CMD ["supervisord", "-c", "/etc/supervisord.conf"] CMD ["supervisord", "-c", "/etc/supervisord.conf"]

View File

@@ -1,14 +1,17 @@
server { server {
listen 80; listen 80;
root /usr/share/nginx/html; root /var/www/html;
index index.html; index index.html;
# Health check endpoint for nginx itself
location = /health {
access_log off;
return 200 "nginx ok\n";
add_header Content-Type text/plain;
}
# Banner entry points — cross-origin script loads from customer # Banner entry points — cross-origin script loads from customer
# sites, so they need permissive CORS. Served from the web root # sites, so they need permissive CORS.
# because the loader derives the bundle URL from its own origin
# (see apps/banner/src/loader.ts). Declared before the SPA
# fallback so nginx doesn't rewrite them to index.html when the
# files aren't yet built in dev.
location = /consent-loader.js { location = /consent-loader.js {
add_header Access-Control-Allow-Origin "*" always; add_header Access-Control-Allow-Origin "*" always;
add_header Access-Control-Allow-Methods "GET, OPTIONS" always; add_header Access-Control-Allow-Methods "GET, OPTIONS" always;
@@ -23,24 +26,31 @@ server {
try_files $uri =404; try_files $uri =404;
} }
# SPA fallback — serve index.html for all other routes # Proxy API requests to FastAPI backend
location / {
try_files $uri $uri/ /index.html;
}
# Proxy API requests to the backend
# Uses Docker's embedded DNS with a variable so nginx resolves at request
# time rather than at startup — prevents crash if api is temporarily down.
location /api/ { location /api/ {
resolver 127.0.0.11 valid=10s; proxy_pass http://127.0.0.1:8000;
set $upstream http://api:8000;
proxy_pass $upstream;
proxy_set_header Host $host; proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme; proxy_set_header X-Forwarded-Proto $scheme;
} }
# Proxy /docs, /openapi.json to FastAPI (Swagger UI)
location /docs {
proxy_pass http://127.0.0.1:8000;
proxy_set_header Host $host;
}
location /openapi.json {
proxy_pass http://127.0.0.1:8000;
proxy_set_header Host $host;
}
# SPA fallback — serve index.html for all other routes
location / {
try_files $uri $uri/ /index.html;
}
# Cache static assets # Cache static assets
location /assets/ { location /assets/ {
expires 1y; expires 1y;

View File

@@ -4,8 +4,20 @@ logfile=/var/log/supervisor/supervisord.log
pidfile=/var/run/supervisord.pid pidfile=/var/run/supervisord.pid
user=root user=root
[program:nginx]
command=nginx -g "daemon off;" -c /etc/nginx/conf.d/default.conf
autostart=true
autorestart=true
stdout_logfile=/dev/stdout
stdout_logfile_maxbytes=0
stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0
stopwaitsecs=5
killasgroup=true
priority=100
[program:api] [program:api]
command=sh -c "uvicorn src.main:app --host 0.0.0.0 --port 8000 --workers ${API_WORKERS:-4} --access-log --proxy-headers --forwarded-allow-ips '*'" command=sh -c "uvicorn src.main:app --host 127.0.0.1 --port 8000 --workers ${API_WORKERS:-4} --access-log --proxy-headers --forwarded-allow-ips '*'"
directory=/app directory=/app
autostart=true autostart=true
autorestart=true autorestart=true
@@ -15,7 +27,7 @@ stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0 stderr_logfile_maxbytes=0
stopwaitsecs=10 stopwaitsecs=10
killasgroup=true killasgroup=true
priority=100 priority=200
[program:worker] [program:worker]
command=celery -A src.celery_app worker --loglevel=info --concurrency=2 command=celery -A src.celery_app worker --loglevel=info --concurrency=2
@@ -28,7 +40,7 @@ stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0 stderr_logfile_maxbytes=0
stopwaitsecs=30 stopwaitsecs=30
killasgroup=true killasgroup=true
priority=200 priority=300
[program:beat] [program:beat]
command=celery -A src.celery_app beat --loglevel=info command=celery -A src.celery_app beat --loglevel=info
@@ -41,7 +53,7 @@ stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0 stderr_logfile_maxbytes=0
stopwaitsecs=10 stopwaitsecs=10
killasgroup=true killasgroup=true
priority=300 priority=400
[program:scanner] [program:scanner]
command=python -m src.worker command=python -m src.worker
@@ -54,5 +66,5 @@ stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0 stderr_logfile_maxbytes=0
stopwaitsecs=10 stopwaitsecs=10
killasgroup=true killasgroup=true
priority=400 priority=500
environment=PYTHONUNBUFFERED="1" environment=PYTHONUNBUFFERED="1"