Web App Development with Docker Compose

Docker Compose is a powerful tool that simplifies managing multi-container Docker applications, making it especially useful for web app development. With Docker Compose, you can easily define and manage services like databases, web servers, and other dependencies in a single configuration.
In this guide, we’ll walk you through how to leverage Docker Compose and Nginx to efficiently set up a local development and production-ready environment for a Django application with PostgreSQL as the database. This guide will provide a real-world example of how Docker Compose, Gunicorn, and Nginx can work together to streamline web application deployment.
While we focus on setting up the environment with Docker, we won’t go into deep details on Django’s internal structure. However, this guide will help you understand how to efficiently containerize and deploy a Django app using Nginx as a reverse proxy.
The project files for this section are available in our GitHub repository. If you'd prefer to skip the detailed steps of Django web app development, simply clone the project files and proceed to Step 12.
If you'd like to learn more about Django, check out our Django Introduction Guide.
Step 1: Set Up Project Environment
Before configuring Docker and Docker Compose, we need to set up the project directory structure to support Django, PostgreSQL, and Nginx.
By running the commands below, you'll have the following project structure in place, ready for Docker Compose to orchestrate the services:
mkdir ch6-docker-compose-demo && cd ch6-docker-compose-demo
mkdir -p django postgres/data nginx
touch docker-compose.yml django/Dockerfile django/requirements.txt nginx/nginx.conf
Project Structure:
ch6-docker-compose-demo/
├── django/
│ ├── Dockerfile
│ ├── requirements.txt
├── nginx/
│ ├── nginx.conf
├── postgres/
│ ├── data/
├── docker-compose.yml
Step 2: Write the Dockerfile for Django
To containerize our Django application, we'll create a Dockerfile. This file defines the steps to build a Docker image for our Django app, ensuring that it has all the necessary dependencies and is set up to run smoothly inside a container.
In this step, we'll walk you through writing the Dockerfile
for Django. This file will specify the base image, install dependencies, copy project files, and configure the app to run inside the container. Using Docker Compose, we can then manage this container alongside other services like the database.
Add the following content to the django/
Dockerfile
:
FROM python:3.9-slim
WORKDIR /app
RUN apt-get update && apt-get install -y \
libpq-dev gcc && \
rm -rf /var/lib/apt/lists/*
COPY requirements.txt .
RUN pip install --no-cache-dir -r requirements.txt
COPY . .
CMD ["gunicorn", "--bind", "0.0.0.0:8000", "config.wsgi:application"]
Explanation of the Dockerfile:
- FROM python:3.9-slim: Python 3.9 slim image is a lightweight version of Python that contains the essentials needed to run a Python app, reducing the overall image size.
- WORKDIR /app: This sets
/app
as the working directory inside the container. All subsequent commands will run in this directory. - RUN apt-get update && apt-get install -y libpq-dev gcc && rm -rf /var/lib/apt/lists/: Install necessary system packages like
libpq-dev
(needed for PostgreSQL database interaction) andgcc
(a C compiler required for some Python dependencies). After the installation, we clean up the package list to minimize the image size. - COPY requirements.txt .: copies the
requirements.txt
file from your local project into the container. This file contains the list of Python dependencies required by the project. - RUN pip install --no-cache-dir -r requirements.txt: This command installs the dependencies listed in
requirements.txt
using pip. The--no-cache-dir
option ensures that pip doesn’t store cached files, further reducing the image size. - COPY . .: Copies all the files from the current directory (i.e., your Django app) into the container's
/app
directory. - RUN python manage.py collectstatic --noinput: This command collects static files before running the app
- CMD ["gunicorn", "--bind", "0.0.0.0:8000", "config.wsgi:application"]: Finally, we specify the command to start the application using Gunicorn, a production-grade web server for Python applications. This command binds the app to
0.0.0.0:8000
(making it accessible externally on port 8000) and tells it to use the WSGI entry point inconfig.wsgi:application
to run the app.
Step 3: Write the docker-compose.yml File
In this step, we'll define the Docker Compose configuration that will set up the necessary containers for our Django application, PostgreSQL database, and Nginx server. Docker Compose allows us to define multi-container applications in a single YAML file, which simplifies the process of starting and managing our environment with just one command. This configuration will not only build the web app container but also link it to a database container and an Nginx container, allowing the three services to communicate seamlessly.
services:
web:
build: ./django
container_name: django_app
expose:
- "8000"
depends_on:
- db
environment:
- DEBUG=True
- DB_NAME=postgres
- DB_USER=postgres
- DB_PASSWORD=postgres
- DB_HOST=db
- DB_PORT=5432
volumes:
- ./django:/app
- ./staticfiles:/app/staticfiles
db:
image: postgres:13
container_name: postgres_db
restart: always
environment:
- POSTGRES_USER=postgres
- POSTGRES_PASSWORD=postgres
- POSTGRES_DB=postgres
volumes:
- postgres-data:/var/lib/postgresql/data
ports:
- "5432:5432"
nginx:
image: nginx:latest
container_name: nginx
restart: always
ports:
- "80:80"
depends_on:
- web
volumes:
- ./nginx/nginx.conf:/etc/nginx/conf.d/default.conf:ro # Load custom Nginx config
- ./staticfiles:/app/staticfiles # Serve static files
volumes:
postgres-data:
Explanation of docker-compose.yml
File:
services:
This section defines the three main services needed for deployment.
web
(Django application using Gunicorn)
This service runs the Django app inside Gunicorn.
build: ./django
→ Tells Docker Compose to build the image from the./django
directory, where theDockerfile
is located.container_name: django_app
→ Names the containerdjango_app
for easier management.expose: - "8000"
→ Makes port 8000 accessible internally to Nginx (but does not expose it to the host machine).depends_on: db
→ Ensures the PostgreSQL container (db
) starts beforeweb
.environment
→- Defines database connection variables so Django can connect to PostgreSQL.
volumes
→./django:/app
→ Mounts the local Django project directory into the container for development../staticfiles:/app/staticfiles
→ Ensures static files are accessible to Nginx.
db
(PostgreSQL database)
This service runs a PostgreSQL database to store application data.
image: postgres:13
→ Uses the official PostgreSQL image (version 13).container_name: postgres_db
→ Names the database container aspostgres_db
.restart: always
→ Ensures the database restarts automatically if stopped.environment
→- Defines database credentials (username, password, and database name).
volumes
→postgres-data:/var/lib/postgresql/data
→ Stores PostgreSQL data outside the container so it persists even if the container is removed.
ports: - "5432:5432"
→ Exposes PostgreSQL port 5432 to allow external database connections (useful for local development).
nginx
(Reverse proxy for handling requests and serving static files)
This service forwards client requests to Django and serves static files.
image: nginx:latest
→ Uses the latest official Nginx image.container_name: nginx
→ Names the containernginx
.restart: always
→ Ensures Nginx restarts automatically if stopped.ports: - "80:80"
→ Maps port 80 of the container to port 80 on the host machine (making the app accessible viahttp://localhost
).depends_on: web
→ Ensures the Django app (web
) starts before Nginx.
Subscribe now for
uninterrupted access.