Test & Publish #768
Workflow file for this run
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
name: Test & Publish | |
on: | |
push: | |
branches: [ master ] | |
tags: | |
- '*' | |
pull_request: | |
branches: [ master ] | |
workflow_dispatch: | |
inputs: | |
debug_enabled: | |
description: 'Run the build with tmate debugging enabled (https://github.com/marketplace/actions/debugging-with-tmate)' | |
required: false | |
default: false | |
env: | |
current_python: '3.12' | |
poetry_version: '1.8.3' | |
nox_cmd: 'nox --verbose --error-on-missing-interpreters --session' | |
jobs: | |
check: | |
name: ${{ matrix.env.NOXENV }} | |
runs-on: ${{ matrix.os }}-latest | |
strategy: | |
fail-fast: false | |
matrix: | |
os: [Ubuntu] | |
env: | |
- NOXENV: check | |
- NOXENV: check_docs | |
- NOXENV: build_docs | |
env: ${{ matrix.env}} | |
steps: | |
- uses: actions/checkout@v3 | |
with: | |
submodules: true | |
- name: Get tag | |
uses: olegtarasov/get-tag@v2 | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Nox & co | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
poetry export --with dev --without-hashes --format constraints.txt --output constraints.txt | |
pip${{ env.current_python }} install --constraint constraints.txt nox nox-poetry | |
- name: Cache docs references cache | |
if: env.NOXENV == 'build_docs' | |
uses: actions/cache@v3 | |
with: | |
path: doc/_build/rinoh/*.rtc | |
key: ${{ env.NOXENV }}-references-${{ hashFiles('doc/_build/rinoh/*.rtc') }} | |
restore-keys: | | |
${{ env.NOXENV }}-references- | |
- name: Run Nox | |
run: ${{ env.nox_cmd }} ${{ env.NOXENV }} | |
- name: Determine directory to publish docs to | |
id: docs-publish-dir | |
if: env.NOXENV == 'build_docs' | |
uses: jannekem/run-python-script-action@v1 | |
with: | |
script: | | |
import os, re | |
github_ref = os.environ.get('GITHUB_REF') | |
m = re.match(r'^refs/tags/v([0-9]+\.[0-9]+\.[0-9]+(-dev\.[0-9]+)?)$', | |
github_ref) | |
if m: | |
target = m.group(1) | |
elif github_ref == 'refs/heads/master': | |
target = 'master' | |
else: | |
target = '' | |
set_output('target', target) | |
- name: Publish docs to GitHub pages | |
if: env.NOXENV == 'build_docs' && steps.docs-publish-dir.outputs.target != '' | |
uses: peaceiris/actions-gh-pages@v3 | |
with: | |
github_token: ${{ secrets.GITHUB_TOKEN }} | |
publish_dir: doc/_build/html | |
destination_dir: ${{ steps.docs-publish-dir.outputs.target }} | |
keep_files: false | |
outputs: | |
docs-target: ${{ steps.docs-publish-dir.outputs.target }} | |
docs-versions: | |
name: Update docs versions JSON | |
needs: check | |
if: needs.check.outputs.docs-target != '' | |
runs-on: Ubuntu-latest | |
steps: | |
- uses: actions/checkout@v3 | |
with: | |
ref: gh-pages | |
- name: Write versions to JSON file | |
uses: jannekem/run-python-script-action@v1 | |
with: | |
script: | | |
import json | |
from pathlib import Path | |
cwd = Path.cwd() | |
versions = sorted((item.name for item in cwd.iterdir() | |
if item.is_dir() and not item.name.startswith('.')), | |
reverse=True) | |
target_dir = Path('gh-pages') | |
target_dir.mkdir(parents=True) | |
legacy_target_file = target_dir / 'versions.json' | |
with legacy_target_file.open('w') as f: | |
json.dump(versions, f) | |
im_versions = [dict(version=version, title=version, aliases=[]) | |
for version in versions] | |
immaterial_target_file = target_dir / 'im_versions.json' | |
with immaterial_target_file.open('w') as f: | |
json.dump(im_versions, f) | |
- name: Publish versions JSON to GitHub pages | |
uses: peaceiris/actions-gh-pages@v3 | |
with: | |
github_token: ${{ secrets.GITHUB_TOKEN }} | |
publish_dir: gh-pages | |
keep_files: true | |
unit: | |
name: ${{ matrix.os }} / ${{ matrix.python-version }} unit tests${{ matrix.name }} | |
strategy: | |
fail-fast: false | |
matrix: | |
os: [Ubuntu, MacOS, Windows] | |
python-version: [3.8, 3.9, '3.10', 3.11, 3.12, 3.13.0-alpha - 3.13, pypy-3.9] | |
noxenv: [unit] | |
cov: [1] | |
name: [""] | |
include: | |
- { os: Ubuntu, python-version: 3.12, noxenv: unit_sphinx, cov: 0, name: " (Sphinx)" } | |
- { os: Ubuntu, python-version: 3.9, noxenv: unit_sphinx_py39, cov: 0, name: " (Sphinx)" } | |
runs-on: ${{ matrix.os }}-latest | |
env: | |
NOXENV: ${{ matrix.noxenv }} | |
WITH_COVERAGE: ${{ matrix.cov }} | |
steps: | |
- uses: actions/checkout@v3 | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Nox & co | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
poetry export --with dev --without-hashes --format constraints.txt --output constraints.txt | |
pip${{ env.current_python }} install --constraint constraints.txt nox nox-poetry | |
- name: Install Python ${{ matrix.python-version }} | |
if: matrix.python-version != env.current_python | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ matrix.python-version }} | |
- name: Set NOXENV for Python ${{ matrix.python-version }} | |
run: python .github/workflows/noxpy.py "${{ matrix.python-version }}" | |
- name: Run Nox | |
run: ${{ env.nox_cmd }} ${{ env.NOXENV }} -- -s | |
- name: Upload coverage statistics to codecov.io | |
if: env.WITH_COVERAGE == 1 | |
shell: bash | |
run: | | |
bash <(curl -s https://codecov.io/bash) -f coverage.xml -F ${{ env.NOXENV }},${{ matrix.python-version }},${{ runner.os }} | |
regression: | |
name: ${{ matrix.os }} / ${{ matrix.python-version }} regression tests${{ matrix.name }} | |
strategy: | |
fail-fast: false | |
matrix: | |
os: [Ubuntu, MacOS, Windows] | |
python-version: ['3.12'] | |
noxenv: [regression] | |
cov: [1] | |
name: [""] | |
include: | |
- { os: Ubuntu, python-version: 3.8, noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: '3.10', noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: 3.11, noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: 3.12, noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: 3.13.0-alpha - 3.13, noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: pypy-3.9, noxenv: regression, cov: 1 } | |
- { os: Ubuntu, python-version: 3.12, noxenv: regression_docutils, cov: 0, name: " (docutils)" } | |
- { os: Ubuntu, python-version: 3.12, noxenv: regression_sphinx, cov: 0, name: " (Sphinx)" } | |
- { os: Ubuntu, python-version: 3.9, noxenv: regression_sphinx_py39, cov: 0, name: " (Sphinx)" } | |
runs-on: ${{ matrix.os }}-latest | |
env: | |
NOXENV: ${{ matrix.noxenv }} | |
WITH_COVERAGE: ${{ matrix.cov }} | |
steps: | |
- uses: actions/checkout@v3 | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Nox & co | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
poetry export --with dev --without-hashes --format constraints.txt --output constraints.txt | |
pip${{ env.current_python }} install --constraint constraints.txt nox nox-poetry | |
- name: Install Python ${{ matrix.python-version }} | |
if: matrix.python-version != env.current_python | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ matrix.python-version }} | |
- name: Set NOXENV for Python ${{ matrix.python-version }} | |
run: python .github/workflows/noxpy.py "${{ matrix.python-version }}" | |
- name: Get Homebrew cache dir (macOS) | |
id: cache-dirs-macos | |
if: startsWith(runner.os, 'macOS') | |
run: | | |
echo "dir=$(brew --cache)" >> $GITHUB_OUTPUT | |
- name: Delete Homebrew cache dir to minimize cache size (macOS) | |
if: startsWith(runner.os, 'macOS') | |
run: | | |
rm -vrf "$(brew --cache)" | |
- name: Get Homebrew package versions (macOS) | |
id: homebrew-versions | |
if: startsWith(runner.os, 'macOS') | |
run: | | |
echo "mupdftools=$(brew info --json mupdf-tools | jq -r '.[0].versions.stable')" >> $GITHUB_OUTPUT | |
echo "imagemagick=$(brew info --json imagemagick | jq -r '.[0].versions.stable')" >> $GITHUB_OUTPUT | |
echo "graphviz=$(brew info --json graphviz | jq -r '.[0].versions.stable')" >> $GITHUB_OUTPUT | |
- name: Cache Homebrew downloads | |
if: startsWith(runner.os, 'macOS') | |
uses: actions/cache@v3 | |
with: | |
path: ${{ steps.cache-dirs-macos.outputs.dir }} | |
key: ${{ runner.os }}-homebrew-mupdftools${{ steps.homebrew-versions.outputs.mupdftools }}-im${{ steps.homebrew-versions.outputs.imagemagick }}-graphviz${{ steps.homebrew-versions.outputs.graphviz }} | |
- name: Get scoop cache dir (Windows) | |
id: cache-dirs-windows | |
if: startsWith(runner.os, 'Windows') | |
run: | | |
echo "dir=$(echo $env:USERPROFILE\scoop\cache)" >> $env:GITHUB_OUTPUT | |
- name: Cache scoop downloads | |
if: startsWith(runner.os, 'Windows') | |
uses: actions/cache@v3 | |
with: | |
path: ${{ steps.cache-dirs-windows.outputs.dir }} | |
key: ${{ runner.os }}-scoop | |
- name: Install tools required by the regression tests (Linux) | |
if: startsWith(runner.os, 'Linux') | |
run: | | |
sudo apt-get install mupdf-tools imagemagick | |
- name: Install tools required by the regression tests (macOS) | |
if: startsWith(runner.os, 'macOS') | |
run: | | |
brew install mupdf-tools imagemagick graphviz | |
- name: Set up scoop (Windows) | |
if: startsWith(runner.os, 'Windows') | |
uses: brechtm/setup-scoop@v2 | |
- name: Install tools required by the regression tests (Windows) | |
if: startsWith(runner.os, 'Windows') | |
run: | | |
scoop bucket add extras | |
scoop install mupdf | |
scoop install imagemagick | |
echo "$env:MAGICK_HOME" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | |
echo "MAGICK_HOME=$env:MAGICK_HOME" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | |
echo "MAGICK_CODER_MODULE_PATH=$env:MAGICK_CODER_MODULE_PATH" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append | |
- name: Run Nox | |
run: ${{ env.nox_cmd }} ${{ env.NOXENV }} -- -s | |
- name: Store test artifacts | |
if: failure() | |
uses: actions/upload-artifact@v3 | |
with: | |
name: ${{ matrix.os }} ${{ matrix.python-version }} regression tests${{ matrix.name }} | |
path: tests_regression/*_output/ | |
- name: Upload coverage statistics to codecov.io | |
if: env.WITH_COVERAGE == 1 | |
shell: bash | |
run: | | |
bash <(curl -s https://codecov.io/bash) -f coverage.xml -F ${{ env.NOXENV }},${{ matrix.python-version }},${{ runner.os }} | |
- name: Setup tmate debug session | |
if: ${{ github.event_name == 'workflow_dispatch' && github.event.inputs.debug_enabled && failure() }} | |
uses: mxschmitt/action-tmate@v3 | |
with: | |
limit-access-to-actor: true | |
check-tag: | |
needs: [check, unit, regression] | |
if: startsWith(github.ref, 'refs/tags/') | |
runs-on: ubuntu-latest | |
steps: | |
- uses: actions/checkout@v3 | |
with: | |
submodules: true | |
- name: Get tag | |
uses: olegtarasov/get-tag@v2 | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Poetry | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
- name: Get package version from Poetry | |
run: echo "PKG_VERSION=$(poetry version --short)" >> $GITHUB_ENV | |
- name: Check whether the tag matches the package version | |
uses: jannekem/run-python-script-action@v1 | |
with: | |
script: | | |
if '${{ env.GIT_TAG_NAME }}' != 'v${{ env.PKG_VERSION }}': | |
error("Git tag doesn't match the package version! Aborting.") | |
exit(1) | |
- name: Check whether the tag matches a stable version | |
id: check-stable | |
uses: jannekem/run-python-script-action@v1 | |
with: | |
script: | | |
import os, re | |
tag = os.environ.get('GIT_TAG_NAME') | |
match = re.match(r'^v[0-9]+\.[0-9]+\.[0-9]+$', tag) | |
output = 'true' if match else 'false' | |
print(f'stable = {output}') | |
set_output('stable', output) | |
- name: Check whether the tag matches a development version | |
id: check-develop | |
uses: jannekem/run-python-script-action@v1 | |
with: | |
script: | | |
import os, re | |
tag = os.environ.get('GIT_TAG_NAME') | |
match = re.match(r'^v[0-9]+\.[0-9]+\.[0-9]+-dev\.[0-9]+$', tag) | |
output = 'true' if match else 'false' | |
print(f'develop = {output}') | |
set_output('develop', output) | |
outputs: | |
stable: ${{ steps.check-stable.outputs.stable }} | |
develop: ${{ steps.check-develop.outputs.develop }} | |
publish-develop: | |
needs: check-tag | |
if: needs.check-tag.outputs.develop == 'true' | |
runs-on: ubuntu-latest | |
steps: | |
- uses: actions/checkout@v3 | |
with: | |
submodules: true | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Poetry | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
- name: Build the wheel | |
run: poetry build --format wheel | |
- name: Publish the wheel as a prerelease on GitHub | |
uses: ncipollo/release-action@v1 | |
with: | |
artifacts: "dist/rinohtype-*.whl" | |
token: ${{ secrets.GITHUB_TOKEN }} | |
prerelease: true | |
publish-release: | |
needs: check-tag | |
if: needs.check-tag.outputs.stable == 'true' | |
runs-on: ubuntu-latest | |
steps: | |
- uses: actions/checkout@v3 | |
with: | |
submodules: true | |
- name: Install Python ${{ env.current_python }} | |
uses: actions/setup-python@v4 | |
with: | |
python-version: ${{ env.current_python }} | |
- name: Install Poetry | |
run: | | |
pip${{ env.current_python }} install poetry==${{ env.poetry_version }} | |
- name: Publish to PyPI | |
env: | |
PYPI_TOKEN: ${{ secrets.PYPI_TOKEN }} | |
TESTPYPI_TOKEN: ${{ secrets.TESTPYPI_TOKEN }} | |
run: | | |
poetry publish --build --username __token__ --password "$PYPI_TOKEN" |