Cees-Jan

- Post source at 🐙

Building Secure Images with GitHub Actions

When I initially started building Docker images only amd64 was relevant, and having a build ⇒ Scan ⇒ Test ⇒ Push cycle was as easy as using docker save and docker load. But with arm64 becoming more and more popular, including with my own home cluster, I needed to add images for arm64. For a while this meant I was pushing images with -amd64 and -arm64 suffixes before combining them into one “image”. All because I want to ensure no images with CVE’s or bugs are pushed. Skipping those is not acceptable for me as a Software Engineer.

Photo of an astronaut pained on the back of shipping containers

Photo by Pixabay

The initial problem is that most tooling for building you will find with a quick search will build and push multi arch images in one move. While this is cool if you want to get a multi arch image out there, it does not meet my requirements. So instead I kept building mainly the same way, but now I had another dimension on my CI matrix that is the arch to build for. To be specific linux/amd64 and linux/arm64. (The OS + arch combination is referred to as the image platform.) This will let me test each variant of an image across all supported platforms using QEMU to simulate the arch, short for architecture, they are build for and will run on. At the end of the pipeline where each image is pushed, I had two steps:

  • One to push all different image variants
  • One to create a manifest combining each platform for each variant and push that

Today, as I’m writing this which is two days ago when you first might be reading this, I finally figured out how to do this without pushing to an external registry first. Not going to bore and confuse you with the changes between my previous iteration, instead I’ll walk you through the, be it a somewhat stripped, workflow that makes this possible.

This set up solves a few issues:

  • BuildX is cool, but it doesn’t let you save and load multiplatform images as docker build does
  • No images with suffixes on the registry anymore that soil it with images anyone will unlikely use directly
  • Only one step to push instead of two

Note: This post is based on this PR on wyrihaximusnet/docker-redirect if you just want to skip to the workflow. It’s a project I started to learn a ton of languages just a bit, but it got knocked into the background due to the impact the covid pandemic had on me.

Set up

The workflow needs some basic set up which includes the image name, which registries to push to, a job that does some JSON magic, and a job that will make the supported platforms available. (The jobs could use some polishing preferably into a way that doesn’t require any jobs. But that is an improvement for another time.)

name: Continuous Integration
env:
  DOCKER_IMAGE: wyrihaximusnet/redirect
  DOCKER_IMAGE_REGISTRIES_SECRET_MAPPING: '{"ghcr.io":"GHCR_TOKEN","docker.io":"HUB_PASSCODE"}'
on:
  push:
  schedule:
    - cron:  '0 0 * * 0'
jobs:
  registry-matrix:
    name: Extract registries from registry secret mapping
    if: (github.event_name == 'push' || github.event_name == 'schedule') && github.ref == 'refs/heads/master'
    runs-on: ubuntu-latest
    needs:
      - tests
    outputs:
      registry: ${{ steps.registry-matrix.outputs.registry }}
    steps:
      - uses: actions/checkout@v4
      - id: registry-matrix
        name: Extract registries from registry secret mapping
        run: |
          echo "registry=$(printenv DOCKER_IMAGE_REGISTRIES_SECRET_MAPPING | jq -c 'keys')" >> $GITHUB_OUTPUT
  supported-arch-matrix:
    name: Supported processor architectures
    runs-on: ubuntu-latest
    outputs:
      arch: ${{ steps.supported-arch-matrix.outputs.arch }}
    steps:
      - uses: actions/checkout@v4
      - id: supported-arch-matrix
        name: Generate Arch
        run: |
          echo "arch=[\"linux/amd64\",\"linux/arm64\"]" >> $GITHUB_OUTPUT

Building

To build the image for multiple platforms we need one thing: QEMU to emulate the arch we’re building for if it’s not the runners native arch. So we have to make sure it’s installed before we can build:

- name: Set up QEMU
  uses: docker/setup-qemu-action@v3

Once it’s set up we can build the image using the normal docker build command. We pass in the platform using the --platform flag and use the environment variable PLATFORM_PAIR we created at the start of the job as the suffix on the image tag:

docker image build --platform=${{ matrix.platform }} -t "${DOCKER_IMAGE}:reactphp-${{ env.PLATFORM_PAIR }}" --no-cache .

Once the image has been built we use good old docker save to save the image to a tarball, for later use we make sure we include the platform in the file name:

docker save "${DOCKER_IMAGE}:reactphp-${{ env.PLATFORM_PAIR }}" -o ./docker-image/docker_image-${{ env.PLATFORM_PAIR }}.tar

Then, we upload the directory the tarball is in as an artifact, and make sure we use the platform in the name, this will come in handy later:

- uses: actions/upload-artifact@v4
  with:
    name: docker-image-reactphp-${{ env.PLATFORM_PAIR }}
    path: ./docker-image

The full job:

  build-docker-image:
    name: Build reactphp Docker (${{ matrix.platform }})
    strategy:
      fail-fast: false
      matrix:
        platform: ${{ fromJson(needs.supported-arch-matrix.outputs.arch) }}
    needs:
      - supported-arch-matrix
    runs-on: ubuntu-latest
    steps:
      - name: Prepare
        run: |
          platform=${{ matrix.platform }}
          echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
      - name: Set up QEMU
        uses: docker/setup-qemu-action@v3
      - uses: actions/checkout@v4
      - run: mkdir ./docker-image
      - run: docker image build --platform=${{ matrix.platform }} --build-arg BUILD_DATE=`date -u +"%Y-%m-%dT%H:%M:%SZ"` --build-arg VCS_REF=`git rev-parse --short HEAD` -t "${DOCKER_IMAGE}:reactphp-${{ env.PLATFORM_PAIR }}" --no-cache .
      - run: docker save "${DOCKER_IMAGE}:reactphp-${{ env.PLATFORM_PAIR }}" -o ./docker-image/docker_image-${{ env.PLATFORM_PAIR }}.tar
      - uses: actions/upload-artifact@v4
        with:
          name: docker-image-reactphp-${{ env.PLATFORM_PAIR }}
          path: ./docker-image

CVE Scanning/Testing

For this post’s sake it doesn’t matter if I cover testing or scanning, so only the set up is covered and the rest is left for your imagination. (You can always check the PR mentioned earlier in this post of course.)

Depending on what you are going to do, you will need to install QEMU again if you are going to run the image. In the project this is taken from the image will be started and k6 is used to test it functionally.

Next we’ll get the image artifact:

- uses: actions/download-artifact@v4
  with:
    name: docker-image-reactphp-${{ env.PLATFORM_PAIR }}
    path: /tmp/docker-image

Next we load the image into Docker, this works fine because it’s only built for a single platform and no multi platform manifest is at play:

docker load --input /tmp/docker-image/docker_image-${{ env.PLATFORM_PAIR }}.tar

After that, go wild and do what you have to do to make sure the image is up to spec, in my case I scan for CVE’s to make sure they don’t make it onto the registry:

echo -e "${{ env.DOCKER_IMAGE }}:reactphp-${{ env.PLATFORM_PAIR }}" | xargs -I % sh -c 'docker run -v /tmp/trivy:/var/lib/trivy -v /var/run/docker.sock:/var/run/docker.sock -t aquasec/trivy:latest --cache-dir /var/lib/trivy image --exit-code 1 --no-progress --format table %'

The full job:

  go-wild:
    name: Scan reactphp for vulnerabilities (${{ matrix.platform }})
    strategy:
      fail-fast: false
      matrix:
        platform: ${{ fromJson(needs.supported-arch-matrix.outputs.arch) }}
    needs:
      - supported-arch-matrix
      - build-docker-image
    runs-on: ubuntu-latest
    steps:
      - name: Prepare
        run: |
          platform=${{ matrix.platform }}
          echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
      - name: Set up QEMU
        uses: docker/setup-qemu-action@v3
      - uses: actions/checkout@v4
      - uses: actions/download-artifact@v4
        with:
          name: docker-image-reactphp-${{ env.PLATFORM_PAIR }}
          path: /tmp/docker-image
      - run: docker load --input /tmp/docker-image/docker_image-${{ env.PLATFORM_PAIR }}.tar
      - run: rm -Rf /tmp/docker-image/
      - run: # Go wild

Pushing

The reason we don’t need a public registry is because for the pushing we’ll run one locally as a service on the job. We’ll use it in pretty much the same way as the public registry, but this way we don’t clutter it with temporary tags:

services:
  registry:
    image: registry:2
    ports:
      - 5000:5000

Before we can push we need QEMU again, and Buildx running on the host network:

- name: Set up QEMU
  uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx
  uses: docker/setup-buildx-action@v3
  with:
    driver-opts: network=host

Next, we download all artifacts for this specific image using the pattern option on the download-artifact action:

- uses: actions/download-artifact@v4
  with:
    pattern: docker-image-reactphp-*
    path: /tmp/docker-image
    merge-multiple: true

Once they are all downloaded, and in the same directory we can load them into docker one by one:

- run: |
    for f in /tmp/docker-image/docker_image-*.tar; do
      docker load --input $f
    done

Before we can use the images to combine them into one, we have to retag and push them to the local registry. Which comes down to prefixing the existing tag with localhost:5000/.

- run: |
    archs=${{ join(fromJson(needs.supported-arch-matrix.outputs.arch), ',') }}
    for arch in ${archs//,/ }
    do
      docker tag "${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}" "localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}"
      docker push "localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}"
    done

The easiest way possible to combine multiple images for different platforms into one is by using some Docker FROM magic. There are few build in ARGs you can use in the FROM instruction of a Dockerfile. In this case we use TARGETOS and TARGETARCH because those match with linux and arm64 in linux/arm64.

echo "FROM localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-\${TARGETOS}-\${TARGETARCH}" >> docker-file-${{ matrix.registry }}-wyrihaximusnet-redirect-reactphp

This way we only have to tell Buildx which Dockerfile to use, which platforms to build, what the image tag will be, and to push it when done:

docker buildx build -f docker-file-${{ matrix.registry }}-wyrihaximusnet-redirect-reactphp --platform=${{ join(fromJson(needs.supported-arch-matrix.outputs.arch), ',') }} -t ${{ matrix.registry }}/${{ env.DOCKER_IMAGE }}:reactphp . --push

The full job:

  push-image:
    if: (github.event_name == 'push' || github.event_name == 'schedule') && github.ref == 'refs/heads/master'
    name: Push reactphp to ${{ matrix.registry }}
    strategy:
      fail-fast: false
      matrix:
        registry: ${{ fromJson(needs.registry-matrix.outputs.registry) }}
    needs:
      - supported-arch-matrix
      - go-wild
      - registry-matrix
    runs-on: ubuntu-latest
    services:
      registry:
        image: registry:2
        ports:
          - 5000:5000
    steps:
      - name: Set up QEMU
        uses: docker/setup-qemu-action@v3
      - name: Set up Docker Buildx
        uses: docker/setup-buildx-action@v3
        with:
          driver-opts: network=host
      - uses: actions/download-artifact@v4
        with:
          pattern: docker-image-reactphp-*
          path: /tmp/docker-image
          merge-multiple: true
      - run: |
          for f in /tmp/docker-image/docker_image-*.tar; do
            docker load --input $f
          done
      - run: rm -Rf /tmp/docker-image/
      - run: |
          archs=${{ join(fromJson(needs.supported-arch-matrix.outputs.arch), ',') }}
          for arch in ${archs//,/ }
          do
            docker tag "${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}" "localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}"
            docker push "localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-${arch//\//-}"
          done
      - name: Login to ${{ matrix.registry }}
        run: |
          echo "${{ env.DOCKER_PASSWORD }}" | \
          docker login ${{ matrix.registry }} \
            --username "${{ env.DOCKER_USER }}" \
            --password-stdin
        env:
          DOCKER_USER: ${{ secrets.HUB_USERNAME }}
          DOCKER_PASSWORD: ${{ secrets[fromJson(env.DOCKER_IMAGE_REGISTRIES_SECRET_MAPPING)[matrix.registry]] }}
      - name: Create merge Dockerfile
        run: echo "FROM localhost:5000/${{ env.DOCKER_IMAGE }}:reactphp-\${TARGETOS}-\${TARGETARCH}" >> docker-file-${{ matrix.registry }}-wyrihaximusnet-redirect-reactphp
      - run: cat docker-file-${{ matrix.registry }}-wyrihaximusnet-redirect-reactphp
      - name: Merged different arch imags into one
        run: docker buildx build -f docker-file-${{ matrix.registry }}-wyrihaximusnet-redirect-reactphp --platform=${{ join(fromJson(needs.supported-arch-matrix.outputs.arch), ',') }} -t ${{ matrix.registry }}/${{ env.DOCKER_IMAGE }}:reactphp . --push

Conclusion

This has been something I’ve been wanting to do for a few years, ever since I started building multi platform images. Will polish it before putting it with the rest of my centralized GitHub Action Workflows. But for now, I’m happy 😎.