GitHub Action
Tagging Strategy
A github action for easily creating a strategy for a whole bunch of tags, and generating all those tags given a single raw semver compatible tag.
Best used for creating docker tags, but creates a comma separated list of strings for whatever uses you may have.
This action supports dynamically resolving whether or not to render each pattern provided using github expressions attached to the tags.
Name | Type | Required | Description |
---|---|---|---|
image_name |
String |
no | an image to pass to tags for docker, omit if not using for docker images |
tags |
csv/list of strategies, optionally add ::<Boolean> |
yes | The strategies to parse the tag paylod with. See conditionally including a strategy to learn how to use conditional expressions for each strategy |
tag_name |
String |
yes (default is X.Y.Z ) |
A semver parseable string |
extra_tags |
csv/list |
no | Optional extra tags appended to output list of tags. These are not parsed by semver, but support conditional expressions |
Name | Type | Description |
---|---|---|
tags |
csv string | The transformed tags |
This project was inspired by the need to create multiple tags for a single docker image. While it can work with any tagging needs for any software.
It is especially well geared towards being used with the Docker Build and Push github action.
The format of the tags
output is natively compatible with the tags
input of the build-and-push action.
steps:
-
id: tagging-strategy
uses: HackerHappyHour/tagging-strategy@v2
with:
tags: '%X%, %X%-debian, %X.Y%, %X.Y%-debian'
tag_name: '1.0.0'
image_name: hello/world
extra_tags: 'latest,debian,edge'
-
uses: docker/setup-buildx-action@v1
-
uses: docker/build-push-action@v2
with:
tags: ${{ steps.tagging-strategy.outputs.tags }}
platforms: linux/amd64,linux/arm64,linux/arm/v7
The example above would result in a multi-platform image where the digests for each platform match across all the tags.
Example output.
Notice how the digests for `linux/amd64` match for each tag (likewise for each arch
Tag
===
latest
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
debian
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
edge
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
1
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
1-debian
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
1.0
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
Tag
===
1.0-debian
Digest OS/ARCH
====== =======
57f8a1d499bb linux/amd64
7fbe9ad1fbc6 linux/arm/v7
ff87a758e329 linux/arm64
The forumula for each individual entry of the tags
input: pattern
+<prerelease>
+<variant>
You can also add a ::<Boolean>
on the end of each tag provided, which allows you to dynamically specify
conditions for which each strategy provided is included. This allows you to use github action expressions
that resolve to true
or false
to conditionally specify whether or not to include a given strategy in the output.
For example, let's say you have a workflows that triggers on both prereleased
and released
events. For a normal
release you want to produce X
, X.Y
, and X.Y.Z
for the given tag_name
, but if it's a prerelease, you don't want
to produce the X
or X.Y
tags.
on:
release:
types: [prereleased, released]
steps:
# checkout steps etc...
# Image Tagger
- name: Image Tag Strategy
id: tagging
uses: HackerHappyHour/tagging-strategy@v2
if: ${{ github.event_name == 'release' }}
with:
tags: |
%X%-foobar::${{ github.event.action != 'prerelease' }}
%X.Y%-foobar::${{ github.event.action != 'prerelease' }}
%X.Y.Z%-foobar
extra_tags: 'latest'
A strategy is comprised of a valid or coercable semver pattern,
using X
, Y
, and Z
, as well as the word latest
.
Valid pattern examples include:
# all strategies below support <strategy>::<Boolean>
%X%
%Z%
%X.Y.Z%
%X.Y%
%X%-foobar
%X.Y.Z%-foobar-baz
Sections of the pattern are denoted using %
. Currently only X
, Y
, and Z
will be translated.
X
- returns Major version
Y
- returns Minor version
Z
- returns Patch version
A prerelease
is parsed from the tag_name
from your release event. This string
will match anything that follows the identified version number from the tag.
For example when creating a release tag in github using the examples below,
the highlighted sections indicate what would be returned as the prerelease
value.
1.0.0-beta1
1.0.0-rc.1
1.0.0-build-3467821
The variant
is any modifier you want to add to the tag. This can be used
to produce a matrix of tags that have the same version, but multiple variants.
Common uses are to use variant
to define features or operating systems for each tag.
Examples of using a variant:
%X%-ubuntu => given 1.0.0 returns 1-ubuntu
%X.Y.Z%-ubuntu => given 1.0.0-rc.1 returns 1.0.0-rc1-ubuntu
All of the features supported by this plugin are used in the OctoPrint/octoprint-docker release workflow. Check it out for the best example of the flexibility and power this action provides.
jobs:
myReleaseExample:
runs-on: ubuntu-latest
steps:
# checkout steps etc...
# Image Tagger
- name: Image Tag Strategy
id: tagging
uses: HackerHappyHour/tagging-strategy@v2
if: ${{ github.event_name == 'release' }}
with:
tags: |
%X%-foobar
%X.Y%-foobar
%X.Y.Z%-foobar
tag_name: ${{ github.ref }}
image_name: foo/bar
extra_tags: |
latest
- name: Setup Buildx
id: setup
uses: crazy-max/ghaction-docker-buildx@v3
- name: Build
run: |
docker buildx build -t ${{ github.repo }}:${{ steps.tagging.outputs.tag }} .
Setup a repository_dispatch
job on the external repository:
name: Dispatcher
on:
release:
types: [prereleased, released]
jobs:
dispatch:
runs-on: ubuntu-latest
steps:
- name: dispatch
id: dispatch
uses: peter-evans/repository-dispatch@v1
with:
token: ${{ secrets.OCTOPRINT_DOCKER_DISPATCH_TOKEN }}
repository: HackerHappyHour/tagging-strategy
event-type: 'release'
client-payload: '{"tag_name": "${{ github.event.release.tag_name }}"}'
Then set up your job in your repo performing the tagging:
name: Release Dispatch
on:
repository_dispatch:
types: [release]
jobs:
debug:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Produce Docker Tags
id: tagging
uses: HackerHappyHour/tagging-strategy@v2
with:
tags: |
%X%-foobar
%X.Y%-foobar
%X.Y.Z%-foobar
tag_name: ${{ github.event.client_payload.tag_name }}
image_name: foo/bar
extra_tags: |
latest
- name: Use Tag
run: echo ${{ steps.tagging.outputs.tags }}
You can conditionally produce a latest
tag by using an expression that evaluates to a boolean
.
Let's say you have a job that can run on push
and pull_request
events on the master
branch, but you only want to output latest
when
the event is a push:
steps:
id: tags
uses: HackerHappyHour/tagging-strategy@v2
with:
tags: |
%X%
extra_tags: |
latest::${{ github.event_name == 'push'}}