aa92bcd96d
Redirect NGINX_ENVSUBST_OUTPUT_DIR to /tmp and update nginx.conf include path so envsubst works with read_only: true in docker-compose. Add tmpfs mount for /etc/nginx/conf.d for additional write layer.
55 lines
2.1 KiB
Docker
55 lines
2.1 KiB
Docker
# =============================================================================
|
|
# FRONTEND DOCKERFILE - Security Hardened
|
|
# =============================================================================
|
|
# Security measures applied:
|
|
# - Non-root user execution (nginx user)
|
|
# - Multi-stage build (minimal runtime)
|
|
# - Read-only filesystem compatible
|
|
# - No unnecessary packages
|
|
# - Specific image versions pinned
|
|
# - Unprivileged nginx configuration
|
|
# =============================================================================
|
|
|
|
# -----------------------------------------------------------------------------
|
|
# Stage 1: Builder
|
|
# -----------------------------------------------------------------------------
|
|
FROM node:22-slim AS builder
|
|
|
|
WORKDIR /app
|
|
|
|
# Install dependencies first (better layer caching)
|
|
COPY package.json package-lock.json* ./
|
|
RUN npm ci --ignore-scripts
|
|
|
|
# Copy source and build
|
|
COPY tsconfig.json tsconfig.node.json vite.config.ts index.html ./
|
|
COPY src ./src
|
|
COPY public ./public
|
|
RUN npm run build
|
|
|
|
# -----------------------------------------------------------------------------
|
|
# Stage 2: Production Runtime (nginx unprivileged)
|
|
# -----------------------------------------------------------------------------
|
|
FROM nginxinc/nginx-unprivileged:1.27-alpine AS runner
|
|
|
|
# Redirect envsubst output to /tmp (writable under read_only: true)
|
|
# and update nginx main config to include from there instead of /etc/nginx/conf.d/
|
|
ENV NGINX_ENVSUBST_OUTPUT_DIR=/tmp
|
|
RUN sed -i 's|include /etc/nginx/conf.d/\*.conf;|include /tmp/default.conf;|' /etc/nginx/nginx.conf
|
|
|
|
# Copy custom nginx config as template for envsubst processing
|
|
# nginx-unprivileged automatically substitutes env vars in .template files
|
|
COPY nginx.conf /etc/nginx/templates/default.conf.template
|
|
|
|
# Copy built static files with correct ownership (nginx user = uid 101)
|
|
COPY --from=builder --chown=101:101 /app/dist /usr/share/nginx/html
|
|
|
|
# nginx-unprivileged listens on 8080 by default
|
|
EXPOSE 8080
|
|
|
|
# Already runs as non-root (nginx user, uid 101)
|
|
USER nginx
|
|
|
|
# Start nginx (entrypoint processes templates automatically)
|
|
CMD ["nginx", "-g", "daemon off;"]
|