Skip to content

GitHub action to calculate major and minor semver tags, e.g. for tagging containers

License

Notifications You must be signed in to change notification settings

jupyterhub/action-major-minor-tag-calculator

Use this GitHub action with your project
Add this Action to an existing workflow or create a new one
View on Marketplace

Repository files navigation

GitHub Action: Calculates all version tags to update

build-test

Intelligently tag a build artifact starting from a semantic version tag (major.minor.patch), and breaking it into partial components (i.e. major.minor, and major), taking into account existing partial tags.

This pattern is often used to tag Container images. For example, the most recent image may be tagged 2.3.0 and latest, but if you also tag the image as 2.3 and 2 users can automatically pull the most recent image in the current major release series e.g when you release 2.3.1, but they won't automatically update when you release 3.0.0.

This GitHub action takes existing tags into account. Using the previous example, if you backported a fix to an older minor version and released 2.2.1 it should be tagged 2.2. It should not be tagged 2 because that tags already corresponds to the newer 2.3.0.

Example output

The GitHub action's only output is named tags and is a JSON formatted list. See the example workflow below for details on how to convert the JSON formatted list to something else. Tags will always be in decreasing order of specificity.

Pushed reference GitHub repo tags tags output Comment
... "[]" No git reference associated with the triggering workflow
main ... "[main]" A branch reference get no associated version tags
1.2.3 1.2.0 "[1.2.3, 1.2, 1, latest]"
1.2.3 1.2.0, 2.0.0 "[1.2.3, 1.2, 1]"
1.2.3 1.2.0, 1.3.0 "[1.2.3, 1.2]"
1.2.3 1.2.0, 1.2.5 "[1.2.3]"
1.2.3-alpha.1 1.2.0 "[1.2.3-alpha.1]" A pre-release suffix on a version is treated like a branch
1.2.3a1 1.2.0 "[1.2.3a1]" A pre-release suffix (loose parsing) on a version is treated like a branch
1.2.3-4 1.2.0, 1.2.3 "[1.2.3-4, 1.2.3, 1.2, 1, latest]" A build number suffix on a version is treated like the version but even newer

Recommended input parameters

By default this action works with Git tags. It will fetch tags from the current repository, and if this action is triggered by a tag push it will autodetect the current tag.

  • githubToken: The GitHub token, required so this action can fetch tags using the GitHub API. If this parameter is not set then defaultTag (if set) or an empty list will be returned.

You can use this action with non-Git tags by explicitly providing the list of existing tags, and the new tag. In this case you should omit githubToken as no GitHub API calls are required.

  • existingTags: List of existing tags as a JSON array, provide this if you aren't processing Git tags, e.g. ["1.2.3-0", "1.2.3-1"]
  • newTag: The new tag, provide this if you aren't processing Git tags, e.g. "1.2.4-0"

Optional input parameters

  • prefix: One or more whitespace or comma delimited prefixes for returned tags.

    As an example, prefix could be set to ghcr.io/a/b: quay.io/a/b: to provide image names with tags for two separate repositories.

    The string "" can be used to represent an empty string, and all combinations of prefixes and suffixes are used.

  • suffix: One or more whitespace or comma delimited suffixes for returned tags.

    As an example, suffix could be set to "" -debian to provide a default image variant without suffix next to an image variant referred to with a -debian suffix.

    The string "" can be used to represent an empty string, and all combinations of prefixes and suffixes are used.

  • defaultTag: If the tag output would be empty return this tag instead. This can be useful for running a workflow in pull requests where no suitable git references are present. prefix or suffix are not automatically added.

  • branchRegex: If a branch name does not match this regex return defaultTag or empty instead.

Output parameters

  • tags: A JSON formatted list of calculated tags.

Example workflow

name: Build and publish image to DockerHub

on:
  push:
    # Uncomment this if you also want a tag created for all pushed branches
    # branches:
    #   - "*"
    tags:
      - "*"

jobs:
  push-images-to-dockerhub:
    runs-on: ubuntu-22.04
    steps:
      - name: Checkout code
        uses: actions/checkout@v4

      # https://github.com/jupyterhub/action-major-minor-tag-calculator
      - name: Get all relevant tags
        id: gettags
        uses: jupyterhub/action-major-minor-tag-calculator@v3
        with:
          githubToken: ${{ secrets.GITHUB_TOKEN }}
          prefix: "my-username/my-image-name:"
          branchRegex: "^[^/]+$"

      # https://github.com/docker/login-action
      - name: Login to DockerHub
        uses: docker/login-action@v3
        with:
          username: ${{ secrets.DOCKERHUB_USERNAME }}
          password: ${{ secrets.DOCKERHUB_ACCESS_TOKEN }}

      # https://github.com/docker/build-push-action
      - name: Build, tag, and push images
        uses: docker/build-push-action@v5
        with:
          push: true
          # The tags expression below could with a git reference triggering
          # this GitHub workflow evaluate to a string like:
          #
          # "my-username/my-image:1.2.3,my-username/my-image:1.2,my-username/my-image:1,my-username/my-image:latest"
          #
          tags: ${{ join(fromJson(steps.gettags.outputs.tags)) }}

Developer notes

This project runs under multiple NodeJS versions, but if you are updating code you must use version 14.x. This is because the contents of the dist/ folder are autogenerated in the packaging step but this depends on the NodeJS version, causing a failure of the CI job that verifies the generated code matches the source.

Install the dependencies:

$ npm install

Build the typescript, run the formatter and linter:

$ npm run build && npm run format && npm run lint

Package the code for distribution (uses ncc):

$ npm run package

Run the tests ✔️

$ npm test

The tests use nock to mock GitHub API responses, no real requests are made so manual testing is still required.

Shortcut:

$ npm run all

Actions are run from GitHub repos so you must checkin the packed dist folder:

$ npm run all
$ git add dist
$ git commit
$ git push origin main

About

GitHub action to calculate major and minor semver tags, e.g. for tagging containers

Topics

Resources

License

Code of conduct

Stars

Watchers

Forks

Packages

No packages published