• Modern UX

    Edit and navigate faster in the terminal with Warp's IDE-like input editor.

  • Warp AI

    AI suggests what commands to run and learns from your documentation.

  • Agent Mode

    Delegate tasks to AI and use natural language on the command line.

  • Warp Drive

    Save and share interactive notebooks, workflows, and environment variables.

  • All Features

Launch PostgreSQL Using Docker Compose

Razvan Ludosanu

Razvan Ludosanu

Founder, learnbackend.dev

Published: 4/5/2024

About Terminus

The short answer

To set up a PostgreSQL database using Docker Compose, you can start by creating a compose.yaml file with the following content:

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - ./data:/var/lib/postgresql/data
    ports:
      - 5432:5432

Where:

  • POSTGRES_USER is used to create the specified user with superuser power.
  • POSTGRES_PASSWORD is used to define the password of the user identified by the POSTGRES_USER variable.
  • POSTGRES_DB is used to define a different name for the default database that is created when the image is first started. If it is not specified, then the value of POSTGRES_USER will be used.

Then run the following docker-compose command to start the PostgreSQL container:

$ docker compose -f compose.yaml up -d

Where:

  • The -f flag is used to specify the file path of the Compose file.
  • The -d flag is used to launch the services in the background (i.e. detached mode).

Note that when running this command, the database data will be stored in the data directory relative to the compose.yaml file.

You can learn more about the docker compose up by visiting the official Docker documentation page.

Easily retrieve this syntax using Warp AI

If you’re using Warp as your terminal, you can easily retrieve this syntax using the Warp AI feature:

Entering docker compose postgres in the AI question input will prompt a human-readable step by step guide including code snippets.

Persisting database data with named volumes

To persist the database data in a volume instead of a bind mount, you can create a new named volume using the top-level volumes field:

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - pgdata:/var/lib/postgresql/data
    ports:
      - 5432:5432

volumes:
  pgdata:

In this example, the database data will be stored in a volume named pgdata, which can be managed using the available docker volume commands.

You can learn more about using volumes in Compose by checking out the official documentation page.

You can also read our other article on how to mount files in Compose.

Executing scripts and SQL files on startup

To initialize the database with a shell script or a file containing SQL instructions, for example to insert dummy data or migrate an existing database, you can place your *.sh and *.sql files within a directory on your local machine and mount it to the /docker-entrypoint-initdb.d directory using the volumes field as follows:

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - ./data:/var/lib/postgresql/data
      - ./init-scripts:/docker-entrypoint-initdb.d
    ports:
      - 5432:5432

Once the user and the database are created, postgres will run any of the files found in the /docker-entrypoint-initdb.d directory before starting the service.

Connecting Postgres to an application

To connect a PostgreSQL database to an application you can use the depends_on field, which is used to express dependency between services.

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - ./data:/var/lib/postgresql/data
    ports:
      - 5432:5432

  app:
    image: node
    depends_on:
      - db

The use of this field causes Compose to start services in dependency order, which means that, in this example, the db service will be started first, then the app service.

Controlling the application's startup with a health check

By default, Compose doesn't wait for a service to be ready to start the next one, only until it's running. This can cause issues if, for example, you have an application that needs to wait for the database to be up and running before being able to handle incoming connections.

To solve this issue you can use the condition field to define a startup condition based on the status of the service it depends on.

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - ./data:/var/lib/postgresql/data
    healthcheck:
      test: ["CMD-SHELL", "pg_isready"]
      interval: 10s
      timeout: 5s
      retries: 5
    ports:
      - 5432:5432

  app:
    image: node
    depends_on:
      db:
        condition: service_healthy

In the above example, we are telling Compose to wait for the db service to be up and running before launching the app service by defining a startup condition whose value is set to service_healthy.

The service_healthy represents a boolean value returned by the execution of the pg_isready command declared in the healthcheck field of the db service, which is a utility for checking the connection status of a PostgreSQL database server. The exit status specifies the result of the connection check.

This way, Compose will wait for the pg_isready utility to return true before launching the app service.

Connecting pgAdmin to Postgres

To manage your database using the pgAdmin interface, you can use the following compose.yaml file:

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: <username>
      POSTGRES_PASSWORD: <password>
      POSTGRES_DB: <database>
    volumes:
      - ./data:/var/lib/postgresql/data
    ports:
      - 5432:5432

  pgadmin:
    image: dpage/pgadmin4
    environment:
      PGADMIN_DEFAULT_EMAIL: <email>
      PGADMIN_DEFAULT_PASSWORD: <password>
    ports:
      - 8080:80

Where:

  • PGADMIN_DEFAULT_EMAIL is used to create a new user account on pgAdmin.
  • PGADMIN_DEFAULT_PASSWORD is used as a password for the user account identified by PGADMIN_DEFAULT_EMAIL.

You can then start both PostgreSQL and pgAdmin using the following command:

$ docker compose -f compose.yaml up -d

And connect to the pgAdmin interface by navigating to the following address 127.0.0.1:8080 in your web browser.

Using .env files for storing sensitive data

In general, sensitive data such as usernames and passwords should live in a different place than the configuration or the code.

In the case of Compose, this data can be stored in a file named .env located at the root of your project, next to the compose.yaml file.

For example, to launch a PostgreSQL container with Compose, you can declare the following environment variables into an .env file:

USERNAME=john
PASSWORD=hello
DATABASE=staging

And use them in the compose.yaml file using the following syntax:

version: '3'

services:
  db:
    image: postgres
    environment:
      POSTGRES_USER: ${USERNAME}
      POSTGRES_PASSWORD: ${PASSWORD}
      POSTGRES_DB: ${DATABASE}
    volumes:
      - ./data:/var/lib/postgresql/data
    ports:
      - 5432:5432

Note that when running the docker compose up command, Compose will automatically detect the .env file and perform the required variable substitutions before launching your service.

You can learn more about environment variables and secrets by visiting the official Docker documentation pages.

Written by

Razvan Ludosanu

Razvan Ludosanu

Founder, learnbackend.dev

Filed Under

Related Articles

Override the Container Entrypoint With docker run

Learn how to override and customize the entrypoint of a Docker container using the docker run command.

Docker

The Dockerfile ARG Instruction

Learn how to define and set build-time variables for Docker images using the ARG instruction and the --build-arg flag.

Docker
Razvan Ludosanu

Start a Docker Container

Learn how to start a new Docker container from an image in both the foreground and the background using the docker-run command.

Docker
Razvan Ludosanu

Stop All Docker Containers

How to gracefully shutdown running containers and forcefully kill unresponsive containers with signals in Docker using the docker-stop and docker-kill commands.

Docker
Razvan Ludosanu

Use An .env File In Docker

Learn how to write and use .env files in Docker to populate the environment of containers on startup.

Docker

Run SSH In Docker

Learn how to launch and connect to a containerized SSH server in Docker using password-based authentication and SSH keys.

Docker
Gabriel Manricks

Launch MySQL Using Docker Compose

Learn how to launch a MySQL container in Docker Compose.

DockerSQL

Execute in a Docker Container

Learn how to execute one or multiple commands in a Docker container using the docker exec command.

Docker
Razvan Ludosanu

Expose Docker Container Ports

Learn how to publish and expose Docker container ports using the docker run command and Dockerfiles.

Docker

Restart Containers In Docker Compose

Learn how to restart and rebuild one or more containers in Docker Compose.

Docker
Razvan Ludosanu

Output Logs in Docker Compose

Learn how to output, monitor, customize and filter the logs of the containers related to one or more services in Docker Compose

Docker
Razvan Ludosanu

Rename A Docker Image

Learn how to rename Docker images locally and remotely using the docker tag command.

Docker

Trusted by hundreds of thousands of professional developers

Download Warp to get started

Download for Mac