Building and deploying Docker images with GitHub Actions
Slowly moving my projects from a selfhosted GitLab instance to GitHub, while doing so I'm learning how to use GitHub Actions to build and delpoy Docker images to the GitHub Container Registry (ghcr.io).
After having most of my personal projects hosted on a Docker server at home, I decided to move everything out again after a power outage and multiple fiber outages.
One of the steps involved is moving projects from a self hosted GitLab instance to GitHub and using GitHub Actions to build and deploy Docker images to the GitHub Container Registry (ghcr.io).
In this blog post I will show you how to build and deploy a Docker image to the GitHub Container Registry using GitHub Actions. I’ll use the workflow used to build and deploy this website as an example to get you started with GitHub Actions and Docker.
Website repository: github.com/jeroenvanwissen/jeroenvanwissen.nl
Create a Dockerfile
First, create a Dockerfile in the root of your repository. This file will be used to build the Docker image. Here is the Dockerfile used to build the Docker image of this website, which is an Astro.build application deployed using Nginx:
FROM node:lts-alpine as build-stage
WORKDIR /app
COPY package*.json ./
RUN npm install
COPY . .
RUN npm run build
FROM nginx:alpine
COPY --from=build-stage /app/dist /usr/share/nginx/html
COPY ./conf/default.conf /etc/nginx/conf.d/default.conf
EXPOSE 80
HEALTHCHECK --start-period=30s --interval=5m --timeout=1s --retries=5 CMD curl -sf http://localhost/health || exit 1
CMD ["nginx", "-g", "daemon off;"]
This Dockerfile uses a multi-stage build to first build the Astro.build application using Node.js, and then copies the built files to an Nginx container. The Nginx container serves the static HTML files generated by the Astro.build application.
Create a GitHub Actions workflow
Next, create a GitHub Actions workflow file in the .github/workflows
directory of your repository. This file will define the steps to build and deploy the Docker image. Here is the GitHub Actions workflow file used to build and push the Docker image of this website to the GitHub Container Registry:
name: Build and push Docker image to ghcr.io
on:
push:
tags: [ 'v*.*.*' ]
env:
REGISTRY: ghcr.io
# github.repository as <account>/<repo>
IMAGE_NAME: ${{ github.repository }}
jobs:
build:
runs-on: ubuntu-latest
permissions:
contents: read
packages: write
steps:
- name: Checkout repository
uses: actions/checkout@v4.1.4
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3.3.0
- name: Log into registry ${{ env.REGISTRY }}
if: github.event_name != 'pull_request'
uses: docker/login-action@v3.1.0
with:
registry: ${{ env.REGISTRY }}
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Extract Docker metadata
id: meta
uses: docker/metadata-action@v5.5.1
with:
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
- name: Build and push Docker image
id: build-and-push
uses: docker/build-push-action@v5.3.0
with:
context: .
push: ${{ github.event_name != 'pull_request' }}
tags: |
${{ steps.meta.outputs.tags }}
${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
labels: ${{ steps.meta.outputs.labels }}
This GitHub Actions workflow file defines a workflow that is triggered when a new tag is pushed to the repository. It uses the docker/build-push-action
action to build and push the Docker image to the GitHub Container Registry. It tags the image in the registry with the given version tag AND ‘latest’ tag, so the latest versionned image will always have the ‘latest’ tag. The workflow logs into the registry using the GitHub token, which is automatically provided by GitHub Actions.
Trigger the workflow
Now that you have created the Dockerfile, GitHub Actions workflow, you can trigger the workflow by adding a tag to your repository. Once the workflow is triggered, it will build the Docker image and push it to the GitHub Container Registry.
To use this Docker image in my Docker Compose file, I simply referenced the image:
...
services:
website:
image: ghcr.io/jeroenvanwissen/jeroenvanwissen.nl:latest
...
That’s it! You have successfully built and deployed a Docker image to the GitHub Container Registry using GitHub Actions. You can now use this workflow as a template for building and deploying other Docker images in your projects.
I hope this blog post was helpful to you. If you have any questions or feedback, feel free to contact me on Twitter/X or Mastodon. I’d love to hear from you!