Bundle FastAPI backend into Next.js Docker container

The Next.js app was proxying /api/* to the old Flask/FastAPI server
at legal-ai.nautilus.marcusgroup.org. When that server went down,
the Next.js app's API calls failed with 503.

Now both services run in the same container:
- FastAPI (uvicorn) on :8000 — the API backend
- Next.js (node) on :3000 — proxies /api/* to localhost:8000

Changes:
- Dockerfile: multi-stage build with Python 3.12 + Node.js
- next.config.ts: default proxy target is now 127.0.0.1:8000
- start.sh: launches uvicorn in background + node in foreground
- pyproject.toml: add fastapi + uvicorn as explicit deps

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-13 14:33:52 +00:00
parent cc50f0ffde
commit 94bc66d7c1
5 changed files with 65 additions and 19 deletions

View File

@@ -4,3 +4,12 @@ mcp-server/.venv/
**/__pycache__/
*.pyc
.git/
.taskmaster/
web/static/
web/__pycache__/
scripts/
skills/
docs/
legacy/
node_modules/
.next/

View File

@@ -1,21 +1,20 @@
# ══════════════════════════════════════════════════════════════
# Dockerfile — Next.js 16 web-ui (ui-rewrite branch only)
# Dockerfile — Next.js frontend + FastAPI backend (single container)
#
# This file REPLACES the FastAPI Dockerfile on this branch so that
# Coolify's default /Dockerfile lookup builds the new Next.js staging
# UI. The FastAPI Dockerfile lives on `main` and is unaffected.
# The container runs both:
# - FastAPI (uvicorn) on :8000 — the API backend
# - Next.js (node) on :3000 — the frontend (proxies /api/* to :8000)
#
# When the rewrite is merged to main, decide between:
# (a) keeping both via separate Dockerfiles + dockerfile_location config, or
# (b) a multi-stage Dockerfile that serves both, or
# (c) fully replacing FastAPI's StaticFiles with this Next.js front end.
# start.sh launches both processes.
# ══════════════════════════════════════════════════════════════
# ── Stage 1: Node deps ────────────────────────────────────────
FROM node:20-alpine AS deps
WORKDIR /app
COPY web-ui/package.json web-ui/package-lock.json ./
RUN npm ci --no-audit --no-fund
# ── Stage 2: Build Next.js ────────────────────────────────────
FROM node:20-alpine AS builder
WORKDIR /app
COPY --from=deps /app/node_modules ./node_modules
@@ -23,18 +22,44 @@ COPY web-ui/ ./
ENV NEXT_TELEMETRY_DISABLED=1
RUN npm run build
FROM node:20-alpine AS runner
# ── Stage 3: Install Python deps ─────────────────────────────
FROM python:3.12-alpine AS pydeps
WORKDIR /opt/api
COPY mcp-server/ ./mcp-server/
RUN pip install --no-cache-dir ./mcp-server
# ── Stage 4: Runner ───────────────────────────────────────────
FROM python:3.12-alpine AS runner
WORKDIR /app
# Install Node.js (needed for Next.js standalone server)
RUN apk add --no-cache nodejs
ENV NODE_ENV=production
ENV NEXT_TELEMETRY_DISABLED=1
ENV PORT=3000
ENV HOSTNAME=0.0.0.0
# next.config.ts uses output: 'standalone', so we copy only the minimal runtime
# Copy Python packages from pydeps stage
COPY --from=pydeps /usr/local/lib/python3.12/site-packages /usr/local/lib/python3.12/site-packages
COPY --from=pydeps /usr/local/bin/uvicorn /usr/local/bin/uvicorn
# Copy Next.js standalone build
COPY --from=builder /app/public ./public
COPY --from=builder /app/.next/standalone ./
COPY --from=builder /app/.next/static ./.next/static
# Copy FastAPI backend code
COPY web/ ./web/
COPY mcp-server/src/ ./mcp-server/src/
# Make mcp-server source available to web/app.py (it does sys.path.insert for legal_mcp)
ENV PYTHONPATH=/app/mcp-server/src
# Copy startup script
COPY start.sh ./start.sh
RUN chmod +x ./start.sh
EXPOSE 3000
CMD ["node", "server.js"]
CMD ["./start.sh"]

View File

@@ -17,6 +17,8 @@ dependencies = [
"rq>=1.16.0",
"pillow>=10.0.0",
"google-cloud-vision>=3.7.0",
"fastapi>=0.115.0",
"uvicorn[standard]>=0.30.0",
]
[build-system]

13
start.sh Executable file
View File

@@ -0,0 +1,13 @@
#!/bin/sh
# Start FastAPI backend + Next.js frontend in the same container.
# Uvicorn runs in the background; Node.js runs in the foreground
# so that Docker gets its exit signal from the primary process.
set -e
echo "Starting FastAPI backend on :8000 ..."
uvicorn web.app:app --host 127.0.0.1 --port 8000 --workers 1 &
UVICORN_PID=$!
echo "Starting Next.js frontend on :3000 ..."
exec node server.js

View File

@@ -1,17 +1,14 @@
import type { NextConfig } from "next";
/**
* Staging config — proxies /api/* and /openapi.json to the production FastAPI
* at legal-ai.nautilus.marcusgroup.org. This lets the new Next.js UI call the
* existing backend without CORS and without running a second FastAPI instance.
*
* When the rewrite branch is cut over to production, set NEXT_PUBLIC_API_BASE_URL
* and/or move the FastAPI in front of this app via traefik routing.
* Proxies /api/* and /openapi.json to the FastAPI backend.
* In Docker both processes run in the same container, so the default
* target is http://127.0.0.1:8000. Override with NEXT_PUBLIC_API_ORIGIN
* if the backend lives elsewhere (e.g. during local dev).
*/
const API_ORIGIN =
process.env.NEXT_PUBLIC_API_ORIGIN ??
"https://legal-ai.nautilus.marcusgroup.org";
process.env.NEXT_PUBLIC_API_ORIGIN ?? "http://127.0.0.1:8000";
const nextConfig: NextConfig = {
output: "standalone",