;workflows: release: create release, upload latest binaries #460
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
# The main hledger continuous integration tests. | |
# Builds all packages expecting no warnings, runs lots of tests, | |
# and on success, saves the binaries as an artifact. | |
# Code must pass this successfully before it can be merged or pushed to master | |
# (https://github.com/simonmichael/hledger/settings/branch_protection_rules/17386787). | |
name: ci | |
on: | |
# When manually triggered in github ui, it runs in master. | |
workflow_dispatch: | |
# When there's a push to the ci branch, it runs in that branch. | |
# After it passes, those commits can be merged/pushed to master. | |
# (Don't use these branches for pull requests, or it will run twice, | |
# https://github.community/t/how-to-trigger-an-action-on-push-or-pull-request-but-not-both/16662/2) | |
push: | |
branches: [ ci ] | |
# When there's a pull request against master, it runs in the pull request's branch. | |
# After it passes, that branch can be merged/pushed to master. | |
pull_request: | |
branches: [ master ] | |
# Uncomment to run it only for changes to these paths: (but that could prevent merging) | |
# paths: | |
# - '.github/workflows/pushpull.yml' | |
# - 'stack*.yaml' | |
# - 'hledger-lib/**' | |
# - 'hledger/**' | |
# - 'hledger-ui/**' | |
# - 'hledger-web/**' | |
# - 'bin/*.hs' | |
# - 'examples/**' | |
# Or to ignore certain paths: | |
# # examples | |
# - '!**.journal' | |
# - '!**.j' | |
# - '!**.ledger' | |
# - '!**.csv' | |
# # docs | |
# - '!**.m4' | |
# - '!**.md' | |
# - '!**.1' | |
# - '!**.5' | |
# - '!**.info' | |
# - '!**.txt' | |
jobs: | |
ci: | |
runs-on: ubuntu-24.04 | |
env: | |
# This workflow uses github's preinstalled ghc & stack on ubuntu. | |
# Keep these synced with the latest ghc version at https://github.com/actions/runner-images/blob/ubuntu22/20240514.2/images/ubuntu/Ubuntu2404-Readme.md#haskell-tools | |
# | |
# caching id for this ghc's build artifacts: | |
ghc: 982 | |
# stack config for this ghc: | |
stack: stack --system-ghc | |
# flag for skipping later steps, declared here to prevent "Context access might be invalid" warnings | |
do-all: | |
steps: | |
- name: Check out | |
uses: actions/checkout@v4 | |
# have to fetch everything for git describe for hledger's --version | |
with: | |
fetch-depth: 0 | |
# - name: Print some context for troubleshooting | |
# env: | |
# GITHUB_CONTEXT: ${{ toJson(github) }} | |
# run: | | |
# echo $GITHUB_CONTEXT | |
# # echo "$GITHUB_SHA" | |
# # echo "$GITHUB_REF" | |
# # echo "$GITHUB_HEAD_REF" | |
# # echo "$GITHUB_BASE_REF" | |
# # git log "$GITHUB_BASE_REF".. | |
# # tools/commitlint "$GITHUB_BASE_REF".. | |
# EARLY ACTIONS | |
- name: Check commit messages | |
# keep this step synced in all workflows which do it | |
# For a PR, the range will be: master..origin/$GITHUB_HEAD_REF | |
# For a push it will be: $BEFORE.. | |
# For a force push, BEFORE is the previous HEAD, and on github (though not locally) this is an "invalid revision range". | |
# 202310: we skip this check when we can't detect the commits, which happens in certain cases | |
# related: https://stackoverflow.com/questions/64708371/how-to-run-github-workflow-on-every-commit-of-a-push | |
# 202312: ignore this if it fails, it may be not worth the hassle | |
env: | |
BEFORE: ${{ github.event.before }} | |
# NUM: 5 | |
shell: bash | |
run: | | |
RANGE=${BEFORE:-origin/master}..${GITHUB_HEAD_REF:-} | |
echo "debug: last 10 commits:" | |
echo "$(git log --format='%h -%d %s (%an, %ci)' -10)" | |
echo "debug: origin/master:" | |
echo "$(git log --format='%h -%d %s (%an, %ci)' -1 origin/master)" | |
echo "debug: BEFORE=$BEFORE" | |
echo "$(git log --format='%h -%d %s (%an, %ci)' -1 $BEFORE)" | |
echo "debug: GITHUB_HEAD_REF=$GITHUB_HEAD_REF" | |
echo "$(git log --format='%h -%d %s (%an, %ci)' -1 $GITHUB_HEAD_REF)" | |
echo "debug: RANGE=$RANGE" | |
echo "debug: commits to check:" | |
echo "$(git log --format='%h -%d %s (%an, %ci)' --abbrev-commit --date=relative --date-order $RANGE)" | |
if git rev-list --quiet $RANGE | |
then | |
tools/commitlint $RANGE || echo "commit lint failed, ignoring" | |
else | |
# echo "could not identify commits, checking last $NUM instead:"; tools/commitlint -$NUM | |
echo "could not identify commits, not checking them" # XXX | |
fi | |
- name: Skip remaining steps if the last commit message begins with ; | |
shell: bash | |
run: | | |
echo "git log -1 --pretty='%s' ${GITHUB_HEAD_REF:+origin/$GITHUB_HEAD_REF} >> $$.gitlog" | |
(git log -1 --pretty='%s' ${GITHUB_HEAD_REF:+origin/$GITHUB_HEAD_REF} >> $$.gitlog \ | |
&& (grep -qE '^ *;' $$.gitlog || echo "do-all=true" >> $GITHUB_ENV)) \ | |
|| ( echo "could not identify commit range, continuing CI steps"; echo "do-all=true" >> $GITHUB_ENV ) | |
# Can't uncache to /usr/bin: | |
# /usr/bin/tar -xf /home/runner/work/_temp/5cef703c-9831-41db-adb3-470b839f8a0e/cache.tzst -P -C /home/runner/work/hledger/hledger --use-compress-program unzstd | |
# /usr/bin/tar: ../../../../../usr/bin/rg: Cannot open: Permission denied | |
# - name: Cache - extra tools (ripgrep) in /usr/bin | |
# id: extratools | |
# uses: actions/cache@v4 | |
# with: | |
# path: /usr/bin/rg | |
# key: ${{ runner.os }}-extratools # should have image version in there too | |
# if: env.do-all | |
- name: Check embedded files | |
run: | | |
if [[ ! -x /usr/bin/rg ]]; then sudo apt install -y ripgrep; fi | |
tools/checkembeddedfiles | |
if: env.do-all | |
# CACHES | |
- name: Cache - stack global package db | |
id: stack-global | |
uses: actions/cache@v4 | |
with: | |
path: ~/.stack | |
# XXX if stack.yaml is a symlink, this fails with | |
# Error: The template is not valid. .github/workflows/push.yml (Line: 103, Col: 14): hashFiles('**.yaml') failed. | |
# Fail to hash files under directory '/home/runner/work/hledger/hledger' | |
key: ${{ runner.os }}-stack-global-$ghc-${{ hashFiles('**.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-stack-global-$ghc | |
if: env.do-all | |
- name: Cache - stack-installed programs in ~/.local/bin | |
id: stack-programs | |
uses: actions/cache@v4 | |
with: | |
path: ~/.local/bin | |
key: ${{ runner.os }}-stack-programs-$ghc-${{ hashFiles('**.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-stack-programs-$ghc | |
if: env.do-all | |
- name: Cache - .stack-work | |
uses: actions/cache@v4 | |
with: | |
path: .stack-work | |
key: ${{ runner.os }}-stack-work-$ghc-${{ hashFiles('**.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-stack-work-$ghc | |
if: env.do-all | |
- name: Cache - hledger-lib/.stack-work | |
uses: actions/cache@v4 | |
with: | |
path: hledger-lib/.stack-work | |
key: ${{ runner.os }}-hledger-lib-stack-work-$ghc-${{ hashFiles('hledger-lib/package.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-hledger-lib-stack-work-$ghc | |
if: env.do-all | |
- name: Cache - hledger/.stack-work | |
uses: actions/cache@v4 | |
with: | |
path: hledger/.stack-work | |
key: ${{ runner.os }}-hledger-stack-work-$ghc-${{ hashFiles('hledger/package.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-hledger-stack-work-$ghc | |
if: env.do-all | |
- name: Cache - hledger-ui/.stack-work | |
uses: actions/cache@v4 | |
with: | |
path: hledger-ui/.stack-work | |
key: ${{ runner.os }}-hledger-ui-stack-work-$ghc-${{ hashFiles('hledger-ui/package.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-hledger-ui-stack-work-$ghc | |
if: env.do-all | |
- name: Cache - hledger-web/.stack-work | |
uses: actions/cache@v4 | |
with: | |
path: hledger-web/.stack-work | |
key: ${{ runner.os }}-hledger-web-stack-work-$ghc-${{ hashFiles('hledger-web/package.yaml') }} | |
restore-keys: | | |
${{ runner.os }}-hledger-web-stack-work-$ghc | |
if: env.do-all | |
# ACTIONS | |
# in modular steps for faster & more focussed failures | |
- name: Install GHC | |
run: | | |
$stack setup --install-ghc | |
if: env.do-all | |
- name: Build deps of hledger-lib | |
run: | | |
$stack build --test --bench hledger-lib --only-dependencies | |
if: env.do-all | |
- name: Build/test hledger-lib | |
run: | | |
$stack install --test --bench hledger-lib --fast --ghc-options=-Werror | |
if: env.do-all | |
- name: Build deps of hledger | |
run: | | |
$stack build --test --bench hledger --only-dependencies | |
if: env.do-all | |
- name: Build/test hledger | |
run: | | |
$stack install --test --bench hledger --fast --ghc-options=-Werror | |
if: env.do-all | |
- name: Build deps of hledger-ui | |
run: | | |
$stack build --test --bench hledger-ui --only-dependencies | |
if: env.do-all | |
- name: Build/test hledger-ui | |
run: | | |
$stack install --test --bench hledger-ui --fast --ghc-options=-Werror | |
if: env.do-all | |
- name: Build deps of hledger-web | |
run: | | |
$stack build --test --bench hledger-web --only-dependencies | |
if: env.do-all | |
- name: Build/test hledger-web | |
run: | | |
$stack install --test --bench hledger-web --fast --ghc-options=-Werror | |
if: env.do-all | |
- name: Install shelltestrunner | |
run: | | |
export PATH=~/.local/bin:$PATH | |
if [[ ! -x ~/.local/bin/shelltest ]]; then $stack install shelltestrunner-1.10; fi | |
shelltest --version | |
if: env.do-all | |
# Takes ~30s on a 2023 github worker. | |
- name: Test functional tests (excluding addons) | |
run: | | |
export PATH=~/.local/bin:$PATH | |
COLUMNS=80 $stack exec -- shelltest --execdir -j16 hledger/test -x /_ -x /addons -x ledger-compat/ledger-baseline -x ledger-compat/ledger-regress -x ledger-compat/ledger-collected | |
# XXX run the bin/ func tests corresponding to the GHC version enabled above, only | |
if: env.do-all | |
# Takes 1m+ on a 2023 github worker. | |
# Moved to binaries-mac-arm64 workflow instead; | |
# haddock breakage might not be found until release time but it's easy to fix. | |
# - name: Test haddock generation | |
# env: | |
# stack: ${{ matrix.plan.stack }} | |
# run: | | |
# printf "haddock version: "; haddock --version | |
# time $stack build --fast --haddock --no-haddock-deps --no-haddock-hyperlink-source --haddock-arguments="--no-print-missing-docs" || echo "HADDOCK FAILED, IGNORING" | |
# # --no-haddock-hyperlink-source is 25% faster | |
# # --no-print-missing-docs is 600% quieter | |
# if: env.do-all | |
# ARTIFACTS | |
- name: Gather binaries | |
id: exes | |
run: | | |
mkdir tmp | |
cd tmp | |
cp -P ~/.local/bin/hledger . | |
cp -P ~/.local/bin/hledger-ui . | |
cp -P ~/.local/bin/hledger-web . | |
strip hledger | |
strip hledger-ui | |
strip hledger-web | |
tar cvf hledger-linux-x64.tar hledger hledger-ui hledger-web | |
if: env.do-all | |
# upload-artifact loses execute permissions, so we tar the binaries to preserve them. | |
# github UI always zips artifacts when they are downloaded, so we don't bother compressing the tar. | |
# Unfortunately it means users must both unzip and untar. | |
- name: Upload binaries | |
uses: actions/upload-artifact@v4 | |
with: | |
name: hledger-linux-x64 | |
path: tmp/hledger-linux-x64.tar | |
if: env.do-all | |
# SNIPPETS | |
# how to set a context variable, and an attempt to make a nice artifact version suffix: | |
# echo "::set-output name=version::$(git branch --show-current | sed 's/-.*//')-$(git rev-parse --short HEAD)" | |
# - name: show stuff | |
# run: | | |
# if [[ -e ~/.local/bin ]]; then ls -lFRa ~/.local/bin; fi | |
# inspect available context info, per | |
# https://docs.github.com/en/free-pro-team@latest/actions/reference/context-and-expression-syntax-for-github-actions. | |
# sample output: https://github.com/simonmichael/hledger/runs/1619227104 | |
# - name: Dump GitHub context | |
# env: | |
# GITHUB_CONTEXT: ${{ toJson(github) }} | |
# run: echo "$GITHUB_CONTEXT" | |
# - name: Dump job context | |
# env: | |
# JOB_CONTEXT: ${{ toJson(job) }} | |
# run: echo "$JOB_CONTEXT" | |
# - name: Dump steps context | |
# env: | |
# STEPS_CONTEXT: ${{ toJson(steps) }} | |
# run: echo "$STEPS_CONTEXT" | |
# - name: Dump runner context | |
# env: | |
# RUNNER_CONTEXT: ${{ toJson(runner) }} | |
# run: echo "$RUNNER_CONTEXT" | |
# - name: Dump strategy context | |
# env: | |
# STRATEGY_CONTEXT: ${{ toJson(strategy) }} | |
# run: echo "$STRATEGY_CONTEXT" | |
# - name: Dump matrix context | |
# env: | |
# MATRIX_CONTEXT: ${{ toJson(matrix) }} | |
# run: echo "$MATRIX_CONTEXT" |