Docker Dev vs Prod — Volume Mounts, Multi-Container Setup
How to structure Docker for local development vs production with Django + Next.js. The key insight: dev mounts your code as a volume; prod bakes code into the image.
Why / When to Use
When you have a Django backend + Next.js frontend and want hot-reload during development, while keeping prod containers clean and self-contained.
Core Concept / Commands
Dev docker-compose (laptop)
# docker-compose.dev.yml
services:
db:
image: postgres:15
environment:
POSTGRES_DB: myapp
POSTGRES_USER: postgres
POSTGRES_PASSWORD: devpassword
ports:
- "5432:5432"
volumes:
- pgdata_dev:/var/lib/postgresql/data
backend:
build:
context: ./backend
dockerfile: Dockerfile.dev
volumes:
- ./backend:/app # ← your local code mounted into container
environment:
DATABASE_URL: postgres://postgres:devpassword@db/myapp
SECRET_KEY: dev-secret-key
DEBUG: "True"
ports:
- "8000:8000"
depends_on:
db:
condition: service_healthy
command: python manage.py runserver 0.0.0.0:8000
frontend:
build:
context: ./frontend
dockerfile: Dockerfile.dev
volumes:
- ./frontend:/app # ← your local code mounted into container
- /app/node_modules # keep node_modules inside container
environment:
NEXT_PUBLIC_API_URL: http://localhost:8000
WATCHPACK_POLLING: "true" # needed for hot reload inside docker
ports:
- "3000:3000"
command: npm run dev
volumes:
pgdata_dev:Dev Dockerfiles (no COPY . . — code comes via volume)
# backend/Dockerfile.dev
FROM python:3.12-slim
WORKDIR /app
COPY requirements.txt .
RUN pip install -r requirements.txt
# No COPY . . — code is mounted
EXPOSE 8000
CMD ["python", "manage.py", "runserver", "0.0.0.0:8000"]# frontend/Dockerfile.dev
FROM node:20-alpine
WORKDIR /app
COPY package*.json .
RUN npm install
# No COPY . . — code is mounted
EXPOSE 3000
CMD ["npm", "run", "dev"]Prod Dockerfiles (code baked in)
# backend/Dockerfile
FROM python:3.12-slim
WORKDIR /app
COPY requirements.txt .
RUN pip install -r requirements.txt
COPY . . # ← code baked into image
RUN python manage.py collectstatic --noinput
EXPOSE 8000
CMD ["gunicorn", "myapp.wsgi:application", "--bind", "0.0.0.0:8000"]# frontend/Dockerfile (multi-stage build)
FROM node:20-alpine AS builder
WORKDIR /app
COPY package*.json .
RUN npm ci
COPY . .
RUN npm run build # ← build baked into image
FROM node:20-alpine
WORKDIR /app
COPY --from=builder /app/.next ./.next
COPY --from=builder /app/public ./public
COPY --from=builder /app/package*.json ./
RUN npm ci --production
EXPOSE 3000
CMD ["npm", "start"]Start dev environment
docker compose -f docker-compose.dev.yml upRun Django commands in dev
docker compose -f docker-compose.dev.yml exec backend python manage.py makemigrations
docker compose -f docker-compose.dev.yml exec backend python manage.py migrate
docker compose -f docker-compose.dev.yml exec backend python manage.py createsuperuser
docker compose -f docker-compose.dev.yml exec backend python manage.py shellMakefile shortcuts
dev:
docker compose -f docker-compose.dev.yml up
down:
docker compose -f docker-compose.dev.yml down
migrate:
docker compose -f docker-compose.dev.yml exec backend python manage.py migrate
makemigrations:
docker compose -f docker-compose.dev.yml exec backend python manage.py makemigrations
shell-backend:
docker compose -f docker-compose.dev.yml exec backend python manage.py shell
logs:
docker compose -f docker-compose.dev.yml logs -fKey Options / Variants
Zero-downtime prod deploy (blue-green)
Run two named backend instances (backend_blue, backend_green). Deploy new code to the inactive one, wait for healthcheck, reload nginx to switch traffic, then stop the old one.
# nginx.conf switch (no reload downtime)
sed -i "s/backend_$OLD/backend_$NEW/g" nginx.conf
docker compose exec nginx nginx -s reloadRolling update (Docker Swarm / Compose v2)
# docker-compose.yml — prod
backend:
deploy:
replicas: 2
update_config:
parallelism: 1
delay: 10s
order: start-first # start new before stopping oldGotchas
WATCHPACK_POLLING: "true"is required for Next.js hot reload inside Docker on some systems- Never use
docker compose down -vin prod — it destroys thepgdatavolume and all data --no-depsindocker compose uprestarts only that container, leaving db untouchednode_modulesmust be excluded from the volume mount with/app/node_modulesto avoid host/container conflicts- Claude Code on your laptop can manage all Docker commands (build, exec, logs) from the host — it doesn’t need to be inside the container
Source
Conversation: “CI/CD setup for Django and Next.js” — 2026-05-13