Guide: dockerfile

Maintain Dockerfile project update

Dockerfile

Goal

This guide analyzes how the Jenkins project automates its Dockerfile file updates.

It’s one thing to publish a Docker image, but it’s another one to maintain it over time.

To scope this guide, we focus on monitoring Java version, and more precisely Java 17.

This guide is heavily inspired by this Jenkins Dockerfile: github.com/jenkinssci/docker

The ultimate goal is to automatically open a pull request such as PR#1944 containing all Java 17 updates.

Requirement

This guide requires:

  1. updatecli

  2. IDE

  3. GitHub Personal Access Token

updatecli

Updatecli is a declarative dependency management tool. The application is available as a command line compatible with Linux/macOS/Windows. Updatecli is designed to work the same way, both from a local machine and a CI environment.

It can easily be installed using Homebrew by running the two following commands:

-> brew tap updatecli/updatecli
-> brew install updatecli

Additional installation instructions are available at www.updatecli.io/docs/prologue/installation.

IDE

The best way to write Updatecli manifest(s) is by using an IDE compatible with jsonschema hosted on schemastore.org, such as Vscode, Intellij, or Neovim. The full list of compatible IDE is available on www.schemastore.org.

For the IDE to load the correct Updatecli jsonschema, Updatecli manifest must have both a parent directory named "updatecli.d" and one of the file extension ".yaml", or ".yml". This provides auto-completion and validation out of the box.

For example, from VScode, typing [ctrl][space] should display a box with various suggestions

GitHub PAT

The GitHub Personal Access Token is the token used to interact with the GitHub API.

In this guide, we use it to clone the repository, create a branch, and open a pull request.

More information on docs.github.com

Pipeline

Description

The pipeline we are analyzing could be schematized as follows:

guide dockerfile

The schema was generated using the following command:

  • updatecli manifest show --config updatecli.d/manifest.yaml --values updatecli/values.yaml --experimental --graph --graph-flavor=mermaid

Manifest

The Updatecli manifest used to generate this pipeline is:

updatecli.d/manifest.yaml
---
name: Bump JDK17 version

scms:
  default:
    kind: github
    spec:
      user: "{{ .github.user }}"
      email: "{{ .github.email }}"
      owner: "{{ .github.owner }}"
      repository: "{{ .github.repository }}"
      token: "{{ requiredEnv .github.token }}"
      username: "{{ .github.username }}"
      branch: "{{ .github.branch }}"

sources:
  lastVersion:
    kind: temurin
    name: Get the latest Adoptium JDK17 version
    spec:
      featureversion: 17
    transformers:
      - trimprefix: "jdk-"

conditions:
  checkTemurinAllReleases:
    name: Check if the "<lastVersion>" is available for all platforms
    kind: temurin
    sourceid: lastVersion
    spec:
      featureversion: 17
      platforms:
        - alpine-linux/x64
        - linux/x64
        - linux/aarch64
        - linux/ppc64le
        - linux/s390x
        - windows/x64

targets:
  ## Global config files
  setJDK17VersionDockerBake:
    name: "Bump JDK17 version for Linux images in the docker-bake.hcl file"
    kind: hcl
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: docker-bake.hcl
      path: variable.JAVA17_VERSION.default
    scmid: default
  setJDK17VersionWindowsDockerCompose:
    name: "Bump JDK17 version in build-windows.yaml"
    kind: yaml
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      files:
        - build-windows.yaml
      key: $.services.jdk17.build.args.JAVA_VERSION
    scmid: default
  ## Dockerfiles
  # Setting default JAVA_VERSION ARG to current Jenkins default JDK17
  setJDK17VersionAlpine:
    name: "Bump JDK17 version for Linux images in the Alpine Linux Dockerfile"
    kind: dockerfile
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: alpine/hotspot/Dockerfile
      instruction:
        keyword: ARG
        matcher: JAVA_VERSION
    scmid: default
  setJDK17VersionDebian:
    name: "Bump JDK17 version for Linux images in the Debian Dockerfiles"
    kind: dockerfile
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      files:
        - debian/bookworm/hotspot/Dockerfile
        - debian/bookworm-slim/hotspot/Dockerfile
      instruction:
        keyword: ARG
        matcher: JAVA_VERSION
    scmid: default
  setJDK17VersionRhel:
    name: "Bump JDK17 version for Linux images in the Rhel Dockerfile"
    kind: dockerfile
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: rhel/ubi9/hotspot/Dockerfile
      instruction:
        keyword: ARG
        matcher: JAVA_VERSION
    scmid: default
  setJDK17VersionWindowsDockerImage:
    name: "Bump default JDK17 version for Linux images in the Windows Dockerfile"
    kind: dockerfile
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: windows/windowsservercore/hotspot/Dockerfile
      instruction:
        keyword: ARG
        matcher: JAVA_VERSION
    scmid: default

actions:
  default:
    kind: github/pullrequest
    scmid: default
    title: Bump JDK17 version to {{ source "lastVersion" }}
    spec:
      labels:
        - dependencies
        - jdk17
values.yaml
github:
  user: "GitHub Actions"
  email: "41898282+github-actions[bot]@users.noreply.github.com"
  username: "github-actions"
  token: "UPDATECLI_GITHUB_TOKEN"
  branch: "master"
  owner: "jenkinsci"
  repository: "docker"# Values file

Then you can run:

export UPDATECLI_GITHUB_TOKEN=<your_PAT>
export UPDATECLI_GITHUB_ACTOR=<your GH username>
updatecli diff --config updatecli.d/manifest.yaml --values values.yaml`

This manifest named Bump JDK17 version can be split into five sections:

  1. scms

  2. sources

  3. conditions

  4. targets

  5. actions

SCMS
updatecli.d/manifest.yaml
  It’s one thing to publish a Docker image, but it’s another one to maintain it over time, like the Jenkins project does.

    spec:
      user: "{{ .github.user }}"
      email: "{{ .github.email }}"
      owner: "{{ .github.owner }}"
      repository: "{{ .github.repository }}"
      token: "{{ requiredEnv "UPDATECLI_GITHUB_TOKEN" }}"
      username: "{{ .github.username }}"
      branch: "{{ .github.branch }}"

The scm section defines the git repositories to interact with, either to retrieve information or to update them.

The scm section accepts different kind of configuration such as git, gitea, gitlab, etc.

In the current scenario, we need to update files on a GitHub repository so we use the scm of kind github. We could also have used the kind git, but as you’ll discover later in this guide, we need to leverage specific GitHub behavior such as GitHub pull request.

Worth to mention, most of the time we keep the scm settings parametrized so we can easily override all pipelines values. For example, to test the workflow toward a different Git repository.

For that we use a file named values.yaml with different settings.

updatecli/values.yaml
#updatecli.d/manifest.yaml
github:
  user: "GitHub Actions"
  email: "41898282+github-actions[bot]@users.noreply.github.com"
  username: "github-actions"
  token: "UPDATECLI_GITHUB_TOKEN"
  branch: "master"
  owner: "jenkinsci"
  repository: "docker"# Values file

You could easily replace in the file values.yaml the owner and repository values to point to your fork.

Then you can run:

  • Dry run mode: updatecli diff --config updatecli/updatecli.d/jdk17.yaml --values updatecli/values.yaml

  • Apply mode: updatecli apply --config updatecli/updatecli.d/jdk17.yaml --values updatecli/values.yaml

Sources
updatecli/manifest.yaml
sources:
  lastVersion:
    kind: temurin
    name: Get the latest Adoptium JDK17 version
    spec:
      featureversion: 17
    transformers:
      - trimprefix: "jdk-"

The sources section defines how to retrieve source information. There are many different kind of source like dockerimage, dockerdigest, etc, where each plugin relies on different parameters specified within the spec section. In this case we decide to use a source of kind temurin which allow use to retrieve specific Temurin version.

The first step here is to identify the "latest" version for Java 1.17.

Then to get the information we are looking for, we have to "transform" the retrieved version, using a set of transformers rules, in this example we don’t need the prefix "jdk-" as all we need is the version number.

Conditions
Details
conditions:
  checkTemurinAllReleases:
    name: Check if the "<lastVersion>" is available for all platforms
    kind: temurin
    sourceid: lastVersion
    spec:
      platforms:
        - alpine-linux/x64
        - linux/x64
        - linux/aarch64
        - linux/ppc64le
        - linux/s390x
        - windows/x64

Quite often, before updating a file, we want to run early checks to ensure that the updates are relevant. Dockerfile is no exception. We want to be sure that we’ll still be able to build our Docker image after the updates, otherwise it makes no sense to update our Dockerfile…​

Once again, there are many things we would like to test, in this case we want to be sure that the Temurin version is available for all the architecture we need to build a Docker image for.

This example relies on the parameter sourceid to define what source output will be used to fetch the default Temurin version we need to check.

Targets

The targets section defines the files to monitor for update.

We can have as many target as we want. Our manifest contains five targets, but only three of them are shown here

Each target with the same scmid will create one Git commit, and then targeting the same pull request as defined later by our action.

Dockerfile
Details
targets:
  setJDK17VersionWindowsDockerImage:
    name: "Bump default JDK17 version for Linux images in the Windows Dockerfile"
    kind: dockerfile
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: windows/windowsservercore/hotspot/Dockerfile
      instruction:
        keyword: ARG
        matcher: JAVA_VERSION
    scmid: default

The first target is to update the Dockerfile ARG JAVA_VERSION for the Windows Server Core image. Since we only defined one source in the manifest, Updatecli uses that source output as the default entry for this target.

The only subtlety here is that our source output contains "+" in the version number, and we need to replace it by "_", so we use a transformer of kind replacer to do so. Please note that a transformer defined in a target is only applied to this target, while a transformer defined in a source is applied to all targets using this source.

The scmid is set to default so we know that this target will monitor the file "windows/windowsservercore/hotspot/Dockerfile" in the GitHub repository defined by the scm configuration.

YAML
Details
targets:
  setJDK17VersionWindowsDockerCompose:
    name: "Bump JDK17 version in build-windows.yaml"
    kind: yaml
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      files:
        - build-windows.yaml
      key: $.services.jdk17.build.args.JAVA_VERSION
    scmid: default

The second target is to update the JDK17 version in the build-windows.yaml file. Instead of using the plugin dockerfile, we use the plugin yaml to update the version number in the build-windows.yaml file. Once again, we need to replace the "+" character by "_" in the version number.

HCL
Details
targets:
  ## Global config files
  setJDK17VersionDockerBake:
    name: "Bump JDK17 version for Linux images in the docker-bake.hcl file"
    kind: hcl
    transformers:
      - replacer:
          from: "+"
          to: "_"
    spec:
      file: docker-bake.hcl
      path: variable.JAVA17_VERSION.default
    scmid: default

The third target is to update the JDK17 version in the docker-bake.hcl file used by Packer. Instead of using the plugin dockerfile, we use the plugin hcl to update the version number in the docker-bake.hcl file.

Actions
Details
actions:
  default:
    kind: github/pullrequest
    scmid: default
    title: Bump JDK17 version to {{ source "lastVersion" }}
    spec:
      labels:
        - dependencies
        - jdk17

An action is executed when at least one target is modified. Once again there are many different kind of actions but in this case we are leveraging the action of kind github/pullrequest that must be bound to a scm using the parameter scmid. So we are telling Updatecli to open a pullrequest on the GitHub repository https://https://github.com/jenkinsci/docker and to assign labels dependencies and jdk17

It worth mentioning that by default, the GitHub scm will do all its operation from a temporary Git branch named updatecli_main_xxx so the pullrequest will be opened from this branch.

Going further

Once you’re happy with the Update pipeline, you can use your CI environment to run it regularly. That’s what we do on the Jenkins project where we run this pipeline every Monday github.com/jenkinsci/docker

You’ll need updatecli installed in your CI environment, and the right credentials. More information:

This guide demonstrated a more advanced update pipeline. You can find similar Updatecli pipelines on the Jenkins project github.com/jenkinsci/docker

We are curious about your advanced pipelines, so feel free to share them with us.

Feel free to chat with us on the Updatecli Matrix channel.

Top