# syntax=docker/dockerfile:1.7 # === Stage 1: install deps with uv === FROM docker.io/library/python:3.12-slim AS deps ENV PYTHONDONTWRITEBYTECODE=1 \ PYTHONUNBUFFERED=1 \ PIP_DISABLE_PIP_VERSION_CHECK=1 # Install uv (fast, reproducible Python package manager) RUN apt-get update && apt-get install -y --no-install-recommends curl ca-certificates \ && rm -rf /var/lib/apt/lists/* \ && curl -LsSf https://astral.sh/uv/install.sh | sh \ && ln -s /root/.local/bin/uv /usr/local/bin/uv WORKDIR /app COPY pyproject.toml ./ # Resolve & install deps into a dedicated venv. After running `uv lock` locally, # switch this to `uv sync --frozen --no-dev` for fully reproducible builds. RUN uv venv /opt/venv \ && uv pip install --python /opt/venv/bin/python --no-cache . # === Stage 2: runtime === FROM docker.io/library/python:3.12-slim AS runtime ENV PYTHONDONTWRITEBYTECODE=1 \ PYTHONUNBUFFERED=1 \ PATH="/opt/venv/bin:$PATH" # Non-root user RUN groupadd --gid 10001 metamorph \ && useradd --uid 10001 --gid metamorph --shell /usr/sbin/nologin --create-home metamorph \ && mkdir -p /data/evidence \ && chown -R metamorph:metamorph /data COPY --from=deps /opt/venv /opt/venv WORKDIR /app COPY --chown=metamorph:metamorph app ./app COPY --chown=metamorph:metamorph alembic ./alembic COPY --chown=metamorph:metamorph alembic.ini pyproject.toml ./ USER metamorph EXPOSE 8000 # Healthcheck hits the local API. HEALTHCHECK --interval=30s --timeout=3s --start-period=10s --retries=3 \ CMD python -c "import urllib.request,sys; \ sys.exit(0 if urllib.request.urlopen('http://127.0.0.1:8000/api/v1/health',timeout=2).status==200 else 1)" CMD ["gunicorn", "app.main:app", \ "--bind", "0.0.0.0:8000", \ "--workers", "2", \ "--threads", "4", \ "--access-logfile", "-", \ "--error-logfile", "-", \ "--log-level", "info"] # === Stage 3: test image — runtime deps + dev extras + tests dir === # Built only when explicitly targeted (`build --target test`). Not used in prod. FROM docker.io/library/python:3.12-slim AS test ENV PYTHONDONTWRITEBYTECODE=1 \ PYTHONUNBUFFERED=1 \ PATH="/opt/venv/bin:$PATH" RUN apt-get update && apt-get install -y --no-install-recommends curl ca-certificates \ && rm -rf /var/lib/apt/lists/* \ && curl -LsSf https://astral.sh/uv/install.sh | sh \ && ln -s /root/.local/bin/uv /usr/local/bin/uv COPY --from=deps /opt/venv /opt/venv WORKDIR /app COPY pyproject.toml ./ # Install the dev extras (pytest, ruff, httpx) on top of the runtime venv. RUN uv pip install --python /opt/venv/bin/python --no-cache ".[dev]" COPY app ./app COPY alembic ./alembic COPY alembic.ini ./ COPY tests ./tests CMD ["python", "-m", "pytest", "tests", "-v"]