Shared GitHub Actions CI for the metio organization: one reusable workflow per
language or build tool, so every project gets the same gates from one place.
| Workflow | For |
|---|---|
golang.yml |
Go projects — build, race tests, vet, staticcheck, gosec, gofumpt, govulncheck, and (auto-detected) arch-go and envtest |
More languages/tools follow the same shape.
The Go pipeline is zero-config: a detect job inspects the repo and the
optional gates skip themselves, so a controller with envtest-backed tests and a
plain library use the identical call.
# .github/workflows/verify.yml in a Go project
# SPDX-License-Identifier: 0BSD
name: Verify
on:
pull_request:
branches: [main]
permissions:
contents: read
jobs:
go:
name: Golang # checks then read "Golang / test", "Golang / lint", …
uses: metio/ci/.github/workflows/golang.yml@main
# … the project's own non-Go jobs (reuse, yaml, docs, container image, dco) …
verify:
name: Verify
needs: [go] # plus the project's other jobs
if: always()
runs-on: ubuntu-latest
steps:
- env:
NEEDS: ${{ toJSON(needs) }}
run: |
bad=$(echo "$NEEDS" | jq -r 'to_entries[] | select(.value.result != "success" and .value.result != "skipped") | "\(.key)=\(.value.result)"')
[ -z "$bad" ] || { echo "::error::$bad"; exit 1; }What detect decides, with no inputs:
- envtest — on when any
*_test.goimportssigs.k8s.io/controller-runtime/pkg/envtest. Thetestjob then runs over the newest Kubernetes minors envtest supports (discovered at runtime) withKUBEBUILDER_ASSETSset; off elsewhere, tests run once. - arch-go — the
architecturejob runs only when anarch-go.ymlis present; otherwise it skips cleanly (no runner spun up).
Inputs (go-version, runs-on) exist for the rare override; most callers need
none.
Composite actions for release pipelines — call them directly from a repo's release workflow.
The org's single version calculator (see Versioning). Named for the scheme so another scheme could be added alongside it later:
- id: version
uses: metio/ci/calver@main
# with: { prefix: v } # "v" prefix for the default shape (e.g. Terraform-provider tags)
# with: { library: true } # imported Go module → v1.YYYYMMDD.SSSSS
- run: echo "${{ steps.version.outputs.version }}" # 2026.6.20143022, or v1.20260620.52222 in library modeClassifies the repo and emits library (true for an imported Go library — a Go
module with no package main), which you pass straight to calver. So a release
workflow is one shape for every repo:
- id: kind
uses: metio/ci/detect-repo-type@main
- id: version
uses: metio/ci/calver@main
with:
library: ${{ steps.kind.outputs.library }}Decides whether a release is warranted — true on a first release (no prior
tag) or when commits since the last release touched the given paths, so a
scheduled run no-ops on a quiet period. Needs a full-history checkout:
- uses: actions/checkout@v5
with:
fetch-depth: 0
- id: gate
uses: metio/ci/needs-release@main
with:
paths: go.mod main.go internal api Dockerfile
- if: steps.gate.outputs.needed == 'true'
run: echo "releasing ${{ steps.gate.outputs.last }} → next"Installs git-cliff and renders release notes for a version using the org-wide
git-cliff config, writing them to a file for gh release create --notes-file:
- id: notes
uses: metio/ci/release-notes@main
with:
version: ${{ steps.version.outputs.version }}
previous: ${{ steps.gate.outputs.last }} # empty on the first release
- run: gh release create "$VERSION" --notes-file "${{ steps.notes.outputs.file }}"The git-cliff version and the config pin live in the action (Renovate bumps
both), so consumers just bump the action ref — no per-repo git-cliff wiring. This
repo's own release.yml uses it.
A release's notes cover the range since the previous tag, and that lower bound is
read (via needs-release) when the run starts — before the run
creates its own tag. Two merges in quick succession would each read the bound
before either has tagged, so both releases get nearly identical, overlapping
notes. concurrency is a per-workflow key, so it can't live in these actions —
every release workflow that uses release-notes must add it itself:
# top level in release.yml, alongside `on:` / `permissions:`
concurrency:
group: ${{ github.workflow }}
cancel-in-progress: falseQueueing (never cancelling the in-flight run) lets the earlier release finish and
tag first, so the next run reads that tag and its notes start there. Moving the
notes step later does not fix this — if two runs' release jobs interleave, the
later-triggered one can still tag first and overlap; serializing the runs is what
guarantees ordering. Trade-off: with cancel-in-progress: false, three merges
inside one run's duration keep only the newest queued run (GitHub cancels the
middle one), so that one calendar version is skipped — no commits are lost from
the changelog, there's just no separate release for it.
Installs cosign and keyless-signs a file, writing <file>.bundle (e.g. the
checksums). Needs permissions: { id-token: write }:
- uses: metio/ci/cosign-sign-blob@main
with:
file: dist/SHA256SUMSBuilds and pushes a multi-arch image (SBOM + provenance + OCI labels) and signs
it with cosign keyless. Needs permissions: { id-token: write, packages: write, contents: read }:
- id: image
uses: metio/ci/container-release@main
with:
image: ghcr.io/metio/jaas
version: ${{ steps.version.outputs.version }}Each workflow ends in one aggregate job — Verify for the PR gate (build,
unit + integration tests, lint, security, docs, container scan) and, where a repo
has it, System Tests for the real-cluster end-to-end layer. Mark only those
aggregates required in branch protection, then turn on auto-merge. Each runs
if: always(), needs every other job in its workflow, and fails unless each is
success or skipped. Because each is a single, fixed-name check:
- a skipped optional gate doesn't block merge;
- the envtest matrix's per-version job names can change freely;
- adding or removing jobs never touches branch protection.
That's the zero-maintenance auto-merge gate.
The calver action is the org's single version calculator:
YYYY.M.D with the UTC time-of-day appended to the day (e.g. 2026.6.20143022).
Second precision lifts the one-release-per-day ceiling of a date-only scheme, and
the three leading-zero-free components keep it valid semver, so container images,
Helm charts, and Terraform providers can all consume it.
Imported Go libraries can't use that shape: a CalVer major (2026) is ≥ 2, so
Go's module rules would force a /v2026 suffix in the import path that changes
every year. CalVer's library mode keeps them under one scheme anyway —
v1.YYYYMMDD.SSSSS (e.g. v1.20260620.52222): the major is pinned at 1 (a
valid v1.x.y module, no path suffix), the padded date sits in the minor so
versions stay ordered, and seconds-since-midnight fills the patch for second
precision with no leading zero. Every release is therefore a minor bump within
v1, which Renovate offers (and can auto-merge) downstream without it ever being
gated as a major update. The trade-off: pinning the major at 1 gives up
semver's breaking-change signal, so consumers should pin versions rather than
assume v1.x is non-breaking.
Until the ci repo cuts its own releases, consume its workflows and actions at
@main; pin to a tag (or SHA) once releases land, with Renovate keeping the pin
current.
0BSD, REUSE-compliant.