Merge pull request #1146 from kristof-mattei/renovate/returntocorp-se… #2656
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# yaml-language-server: $schema=https://json.schemastore.org/github-workflow.json | |
name: Build | |
on: | |
push: | |
branches: | |
- main | |
pull_request: | |
branches: | |
- main | |
permissions: | |
contents: read | |
checks: write | |
pull-requests: write | |
issues: write | |
packages: write | |
env: | |
CARGO_TERM_COLOR: always | |
# set this to true in GitHub variables to enable building the container | |
# HAS_CONTAINER: true | |
# Use docker.io for Docker Hub if empty | |
REGISTRY: ghcr.io | |
# github.repository as <account>/<repo> | |
IMAGE_NAME: ${{ github.repository }} | |
# just a name, but storing it separately as we're nice people | |
RUSTFLAGS: --deny=warnings | |
concurrency: | |
# each new commit to a PR runs this workflow | |
# so we need to avoid a long running older one from overwriting the "pr-<number>-latest" | |
group: "${{ github.workflow }} @ ${{ github.ref_name }}" | |
cancel-in-progress: true | |
jobs: | |
repo-has-container: | |
name: Repo has container? | |
runs-on: ubuntu-latest | |
outputs: | |
has_container: ${{ steps.determine.outputs.has_container }} | |
steps: | |
- name: Repo has docker container? | |
shell: bash | |
id: determine | |
run: | | |
HAS_CONTAINER="${{ vars.HAS_CONTAINER }}" | |
echo "has_container=${HAS_CONTAINER:-false}" >> ${GITHUB_OUTPUT} | |
changes: | |
name: Detect changes | |
runs-on: ubuntu-latest | |
outputs: | |
code: ${{ steps.filter.outputs.code }} | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Check if we actually made changes | |
uses: dorny/paths-filter@de90cc6fb38fc0963ad72b210f1f284cd68cea36 # v3.0.2 | |
id: filter | |
with: | |
token: ${{ secrets.GITHUB_TOKEN }} | |
filters: .github/file-filters.yml | |
calculate-version: | |
name: Calculate version | |
runs-on: ubuntu-latest | |
needs: | |
- changes | |
- repo-has-container | |
outputs: | |
version: ${{ steps.version.outputs.next_version }} | |
if: | | |
github.event_name == 'pull_request' && | |
fromJSON(needs.repo-has-container.outputs.has_container) == true && | |
fromJSON(needs.changes.outputs.code) == true | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
fetch-depth: 0 | |
- name: Cache dependencies | |
uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | |
env: | |
CACHE_NAME: cargo-cache-dependencies | |
with: | |
path: | | |
~/.cargo | |
./target | |
key: ${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}-cocogitto | |
restore-keys: | | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}- | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}- | |
- name: Set up mold | |
uses: rui314/setup-mold@f80524ca6eeaa76759b57fb78ddce5d87a20c720 # v1 | |
- name: Set up toolchain | |
shell: bash | |
run: | | |
rm ${HOME}/.cargo/bin/cargo-fmt | |
rm ${HOME}/.cargo/bin/rust-analyzer | |
rm ${HOME}/.cargo/bin/rustfmt | |
rustup update | |
cargo --version | |
- name: Get binstall | |
shell: bash | |
run: | | |
cd /tmp | |
archive="cargo-binstall-x86_64-unknown-linux-musl.tgz" | |
wget "https://github.com/cargo-bins/cargo-binstall/releases/latest/download/${archive}" | |
tar -xvf "./${archive}" | |
rm "./${archive}" | |
mv ./cargo-binstall ~/.cargo/bin/ | |
- name: Install cocogitto to get the next version number | |
shell: bash | |
run: | | |
cargo binstall --no-confirm cocogitto --target x86_64-unknown-linux-musl --pkg-url "{ repo }/releases/download/{ version }/{ name }-{ version }-{ target }.tar.gz" --bin-dir "{ bin }" --pkg-fmt tgz | |
- name: Calculate next version | |
shell: bash | |
id: version | |
run: | | |
VERSION="$(cog bump --auto --dry-run || true)" | |
if [[ "$VERSION" =~ ^v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then | |
echo "New version: ${VERSION}" | |
else | |
VERSION="$(git tag --points-at "$(git rev-list --tags --max-count=1)" | sort --reverse | head --lines 1)" | |
echo "No version generated, defaulting to latest git tag: ${VERSION}" | |
fi | |
# remove v | |
VERSION="${VERSION//v/}" | |
# store | |
echo "next_version=${VERSION}" >> ${GITHUB_OUTPUT} | |
cargo-build: | |
name: Cargo build | |
runs-on: ubuntu-latest | |
needs: | |
- changes | |
if: | | |
github.event_name == 'pull_request' && | |
fromJSON(needs.changes.outputs.code) == true | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Cache dependencies | |
uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | |
env: | |
CACHE_NAME: cargo-cache-dependencies | |
with: | |
path: | | |
~/.cargo | |
./target | |
key: ${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}-build | |
restore-keys: | | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}- | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}- | |
- name: Set up mold | |
uses: rui314/setup-mold@f80524ca6eeaa76759b57fb78ddce5d87a20c720 # v1 | |
- name: Set up toolchain | |
shell: bash | |
run: | | |
rm ${HOME}/.cargo/bin/cargo-fmt | |
rm ${HOME}/.cargo/bin/rust-analyzer | |
rm ${HOME}/.cargo/bin/rustfmt | |
rustup update | |
cargo --version | |
- name: Build | |
shell: bash | |
run: | | |
cargo build --all-targets --workspace --verbose | |
cargo-fmt: | |
name: Cargo fmt | |
runs-on: ubuntu-latest | |
needs: | |
- changes | |
if: | | |
github.event_name == 'pull_request' && | |
fromJSON(needs.changes.outputs.code) == true | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Cache dependencies | |
uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | |
env: | |
CACHE_NAME: cargo-cache-dependencies | |
with: | |
path: | | |
~/.cargo | |
./target | |
key: ${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}-fmt | |
restore-keys: | | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}- | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}- | |
- name: Set up mold | |
uses: rui314/setup-mold@f80524ca6eeaa76759b57fb78ddce5d87a20c720 # v1 | |
- name: Set up toolchain | |
shell: bash | |
run: | | |
rm ${HOME}/.cargo/bin/cargo-fmt | |
rm ${HOME}/.cargo/bin/rust-analyzer | |
rm ${HOME}/.cargo/bin/rustfmt | |
rustup update | |
cargo --version | |
- name: Check formatting | |
shell: bash | |
run: | | |
cargo fmt --all -- --check --verbose | |
cargo-test-and-report: | |
name: Cargo test (and report) | |
runs-on: ubuntu-latest | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Cache dependencies | |
uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | |
env: | |
CACHE_NAME: cargo-cache-dependencies | |
with: | |
path: | | |
~/.cargo | |
./target | |
key: ${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}-test | |
restore-keys: | | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}- | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}- | |
- name: Set up mold | |
uses: rui314/setup-mold@f80524ca6eeaa76759b57fb78ddce5d87a20c720 # v1 | |
- name: Set up toolchain | |
shell: bash | |
run: | | |
rm ${HOME}/.cargo/bin/cargo-fmt | |
rm ${HOME}/.cargo/bin/rust-analyzer | |
rm ${HOME}/.cargo/bin/rustfmt | |
rustup update | |
cargo --version | |
- name: Install llvm-tools-preview | |
shell: bash | |
run: | | |
rustup component add llvm-tools-preview | |
- name: Get binstall | |
shell: bash | |
run: | | |
archive="cargo-binstall-x86_64-unknown-linux-musl.tgz" | |
wget "https://github.com/ryankurte/cargo-binstall/releases/latest/download/${archive}" | |
tar -xvf "./${archive}" | |
rm "./${archive}" | |
mv ./cargo-binstall ~/.cargo/bin/ | |
- name: Install nextest, custom test runner, with native support for junit | |
shell: bash | |
run: | | |
cargo binstall --no-confirm cargo-nextest; | |
- name: Install grcov | |
shell: bash | |
run: | | |
cargo binstall --no-confirm grcov --pkg-url "{ repo }/releases/download/v{ version }/{ name }-{ target }.tar.bz2" --pkg-fmt tbz2 --bin-dir "{ bin }"; | |
- name: Build with instrumentation support | |
shell: bash | |
env: | |
RUSTFLAGS: "${{ env.RUSTFLAGS }} --allow=warnings -C instrument-coverage" | |
run: | | |
cargo build --all-targets --all-features --workspace --verbose | |
- name: Run nextest | |
shell: bash | |
id: tests | |
env: | |
RUSTFLAGS: "${{ env.RUSTFLAGS }} --allow=warnings -C instrument-coverage" | |
LLVM_PROFILE_FILE: "profiling/profile-%p-%m.profraw" | |
run: | | |
cargo nextest run --profile ci --no-fail-fast --all-targets --all-features --workspace | |
continue-on-error: true | |
- name: Upload test results | |
uses: EnricoMi/publish-unit-test-result-action@170bf24d20d201b842d7a52403b73ed297e6645b # v2.18.0 | |
with: | |
check_name: Test results | |
github_token: ${{ secrets.GITHUB_TOKEN }} | |
junit_files: reports/results.xml | |
- name: Run grcov | |
shell: bash | |
run: | | |
grcov $(find profiling -name "profile-*.profraw" -print) --source-dir . --binary-path ./target/debug/ --output-type lcov --branch --ignore-not-existing --llvm --keep-only "src/**" --keep-only "tests/**" --output-path ./reports/lcov.info | |
- name: Upload coverage results (to Codecov.io) | |
uses: codecov/codecov-action@5a605bd92782ce0810fa3b8acc235c921b497052 # v5.2.0 | |
with: | |
disable_search: true | |
fail_ci_if_error: true | |
files: reports/lcov.info | |
plugins: "" | |
token: ${{ secrets.CODECOV_TOKEN }} | |
- name: Upload test results to Codecov | |
uses: codecov/test-results-action@4e79e65778be1cecd5df25e14af1eafb6df80ea9 # v1.0.2 | |
with: | |
disable_search: true | |
fail_ci_if_error: true | |
files: reports/results.xml | |
token: ${{ secrets.CODECOV_TOKEN }} | |
- name: Fail if tests failed | |
shell: bash | |
if: | | |
steps.tests.outcome != 'success' | |
run: | | |
# the test reporter we use (or any for that matter) | |
# all show a report. But we cannot depend on that report because | |
# we don't know which subsection it belongs in GitHub | |
# so we explicitly fail this one | |
# which will fail All Done | |
exit 1; | |
cargo-clippy-and-report: | |
name: Cargo clippy (and report) | |
runs-on: ubuntu-latest | |
needs: | |
- changes | |
if: | | |
github.event_name == 'pull_request' && | |
fromJSON(needs.changes.outputs.code) == true | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Cache dependencies | |
uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0 | |
env: | |
CACHE_NAME: cargo-cache-dependencies | |
with: | |
path: | | |
~/.cargo | |
./target | |
key: ${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}-clippy | |
restore-keys: | | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}-${{ hashFiles('Cargo.lock') }}- | |
${{ runner.os }}-build-${{ env.CACHE_NAME }}- | |
- name: Set up mold | |
uses: rui314/setup-mold@f80524ca6eeaa76759b57fb78ddce5d87a20c720 # v1 | |
- name: Set up toolchain | |
shell: bash | |
run: | | |
rm ${HOME}/.cargo/bin/cargo-fmt | |
rm ${HOME}/.cargo/bin/rust-analyzer | |
rm ${HOME}/.cargo/bin/rustfmt | |
rustup update | |
cargo --version | |
- name: Run Clippy for GitHub Actions report | |
uses: actions-rs-plus/clippy-check@30fef0f891edb491831cd248156cfb18d7d12fda # v2.2.0 | |
with: | |
args: --workspace --all-targets --all-features --no-deps | |
docker-build: | |
name: Build Docker container for ${{ matrix.platform.docker }}-${{ matrix.platform.rust }} | |
runs-on: ubuntu-latest | |
needs: | |
- calculate-version | |
# if: | |
# ... is not needed because calculate-version will not run if we disable building the docker container | |
env: | |
APPLICATION_NAME: PLACEHOLDER # overridden in step 'Set application name', this is merely to satisfy the linter | |
PATH_TO_TAR: PLACEHOLDER # same ^ | |
PLATFORM_PAIR: PLACEHOLDER # same ^ | |
PLATFORM_UNIQUE_TAG: PLACEHOLDER # same ^ | |
strategy: | |
fail-fast: false | |
matrix: | |
platform: | |
- docker: linux/amd64 | |
rust: x86_64-unknown-linux-musl | |
- docker: linux/arm64 | |
rust: aarch64-unknown-linux-musl | |
steps: | |
- name: Checkout | |
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | |
with: | |
show-progress: false | |
- name: Prepare name | |
run: | | |
platform=${{ matrix.platform.docker }}-${{ matrix.platform.rust }} | |
echo "PLATFORM_PAIR=${platform//\//-}" >> ${GITHUB_ENV} | |
- name: Set the Cargo.toml version before we copy in the data into the Docker container | |
shell: bash | |
run: | | |
./.github/scripts/update-version.sh ${{ needs.calculate-version.outputs.version }} | |
- name: Set up Docker Buildx | |
uses: docker/setup-buildx-action@6524bf65af31da8d45b59e8c27de4bd072b392f5 # v3.8.0 | |
# TODO validate no changes between github.event.pull_request.head.sha and the actual current sha (representing the hypothetical merge) | |
- name: Lowercase the image name | |
shell: bash | |
run: | | |
echo "IMAGE_NAME=${IMAGE_NAME,,}" >> ${GITHUB_ENV} | |
- name: Set Docker tag | |
shell: bash | |
run: | | |
PLATFORM_UNIQUE_TAG=pr-${{ github.event.pull_request.base.sha }}-${{ github.event.pull_request.head.sha }}-${{ env.PLATFORM_PAIR }} | |
echo "PLATFORM_UNIQUE_TAG=${PLATFORM_UNIQUE_TAG##*/}" >> ${GITHUB_ENV} | |
# Extract metadata (tags, labels) for Docker | |
# https://github.com/docker/metadata-action | |
- name: Extract Docker metadata | |
uses: docker/metadata-action@369eb591f429131d6889c46b94e711f089e6ca96 # v5.6.1 | |
id: meta | |
with: | |
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} | |
tags: | | |
type=raw,value=${{ env.PLATFORM_UNIQUE_TAG }} | |
labels: | | |
org.opencontainers.image.version=pr-${{ github.event.number }} | |
org.opencontainers.image.source=${{ github.event.pull_request.html_url }} | |
- name: Log into registry ${{ env.REGISTRY }} | |
uses: docker/login-action@9780b0c442fbb1117ed29e0efdff1e18412f7567 # v3.3.0 | |
with: | |
registry: ${{ env.REGISTRY }} | |
username: ${{ github.actor }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
- name: Set application name | |
shell: bash | |
run: | | |
APPLICATION_NAME=${{ github.repository }} | |
echo "APPLICATION_NAME=${APPLICATION_NAME##*/}" >> ${GITHUB_ENV} | |
- name: Build Docker image | |
uses: docker/build-push-action@67a2d409c0a876cbe6b11854e3e25193efe4e62d # v6.12.0 | |
with: | |
build-args: | | |
APPLICATION_NAME=${{ env.APPLICATION_NAME }} | |
TARGET=${{ matrix.platform.rust }} | |
context: . | |
# this container is THE PR's artifact, and we will re-tag it | |
# once the PR has been accepted | |
tags: ${{ steps.meta.outputs.tags }} | |
labels: ${{ steps.meta.outputs.labels }} | |
cache-from: type=registry,ref=${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:buildcache-${{ env.PLATFORM_PAIR }} | |
cache-to: type=registry,ref=${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:buildcache-${{ env.PLATFORM_PAIR }},mode=max | |
platforms: ${{ matrix.platform.docker }} | |
outputs: type=docker,dest=/tmp/${{ env.PLATFORM_UNIQUE_TAG }}.tar | |
- name: Upload artifact | |
uses: actions/upload-artifact@65c4c4a1ddee5b72f698fdd19549f0f0fb45cf08 # v4.6.0 | |
with: | |
name: containers-${{ env.PLATFORM_PAIR }} | |
path: /tmp/${{ env.PLATFORM_UNIQUE_TAG }}.tar | |
if-no-files-found: error | |
retention-days: 1 | |
docker-publish: | |
name: Publish Docker container | |
runs-on: ubuntu-latest | |
needs: | |
- cargo-build | |
- cargo-fmt | |
- cargo-test-and-report | |
- cargo-clippy-and-report | |
- docker-build | |
# Check if the event is not triggered by a fork | |
if: | | |
github.event.pull_request.head.repo.full_name == github.repository && | |
github.event_name == 'pull_request' | |
steps: | |
- name: Set up Docker Buildx | |
uses: docker/setup-buildx-action@6524bf65af31da8d45b59e8c27de4bd072b392f5 # v3.8.0 | |
- name: Download artifact | |
uses: actions/download-artifact@fa0a91b85d4f404e444e00e005971372dc801d16 # v4.1.8 | |
with: | |
path: /tmp/containers | |
pattern: containers-* | |
merge-multiple: true | |
- name: Log into registry ${{ env.REGISTRY }} | |
uses: docker/login-action@9780b0c442fbb1117ed29e0efdff1e18412f7567 # v3.3.0 | |
with: | |
registry: ${{ env.REGISTRY }} | |
username: ${{ github.actor }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
- name: Lowercase the image name | |
shell: bash | |
run: | | |
echo "IMAGE_NAME=${IMAGE_NAME,,}" >> ${GITHUB_ENV} | |
- name: Load images from artifacts | |
shell: bash | |
run: | | |
ls -l /tmp/containers/ | |
for container in /tmp/containers/* | |
do | |
echo $container | |
docker load --input $container | |
tag=$(basename -- $container .tar) | |
echo ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}@sha256:${tag} | |
docker push ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:${tag} | |
done | |
- name: Extract Docker metadata | |
uses: docker/metadata-action@369eb591f429131d6889c46b94e711f089e6ca96 # v5.6.1 | |
id: meta | |
with: | |
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} | |
tags: | | |
type=ref,event=pr,suffix=-latest | |
type=raw,value=pr-${{ github.event.pull_request.base.sha }}-${{ github.event.pull_request.head.sha }} | |
- name: Merge images | |
shell: bash | |
working-directory: /tmp/containers | |
run: | | |
# all files in dir | |
platform_tags=(*) | |
# yeet extension | |
platform_tags=${platform_tags[@]%.tar} | |
new_tags="${{ join(steps.meta.outputs.tags, ' ') }}" | |
new_tags=$(printf -- '--tag %s ' $new_tags) | |
expanded_platform_tags=$(printf '${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:%s ' $platform_tags) | |
docker buildx imagetools create $new_tags $expanded_platform_tags | |
for new_tag in $(echo "${{ join(steps.meta.outputs.tags, ' ') }}"); do | |
docker buildx imagetools inspect --raw $new_tag | |
done | |
all-done: | |
name: All done | |
# this is the job that should be marked as required on GitHub. It's the only one that'll reliably trigger | |
# when any upstream fails: success | |
# when all upstream skips: pass | |
# when all upstream success: success | |
# combination of upstream skip and success: success | |
runs-on: ubuntu-latest | |
needs: | |
- calculate-version | |
- cargo-build | |
- cargo-fmt | |
- cargo-clippy-and-report | |
- cargo-test-and-report | |
- docker-build | |
- docker-publish | |
if: | | |
always() | |
steps: | |
- name: Fail! | |
shell: bash | |
if: | | |
contains(needs.*.result, 'failure') || | |
contains(needs.*.result, 'cancelled') | |
run: | | |
echo "One / more upstream failed or was cancelled. Failing job..." | |
exit 1 | |
- name: Success! | |
shell: bash | |
run: | | |
echo "Great success!" |