add option to generate test coverage report without codecov #1
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 Python package | ||
on: | ||
workflow_call: | ||
inputs: | ||
envs: | ||
description: Array of tox environments to test | ||
required: true | ||
type: string | ||
libraries: | ||
description: Additional packages to install | ||
required: false | ||
default: '' | ||
type: string | ||
posargs: | ||
description: Positional arguments for the underlying tox test command | ||
required: false | ||
default: '' | ||
type: string | ||
toxdeps: | ||
description: Tox dependencies | ||
required: false | ||
default: '' | ||
type: string | ||
toxargs: | ||
description: Positional arguments for tox | ||
required: false | ||
default: '' | ||
type: string | ||
pytest: | ||
description: Whether pytest is run | ||
required: false | ||
default: true | ||
type: boolean | ||
pytest-results-summary: | ||
description: Whether to report test summary | ||
required: false | ||
default: false | ||
type: boolean | ||
coverage: | ||
description: Coverage providers to upload to | ||
required: false | ||
default: '' | ||
type: string | ||
conda: | ||
description: Whether to test with conda | ||
required: false | ||
default: 'auto' | ||
type: string | ||
setenv: | ||
description: A map of environment variables to be available when testing | ||
required: false | ||
default: '' | ||
type: string | ||
display: | ||
description: Whether to setup a headless display | ||
required: false | ||
default: false | ||
type: boolean | ||
cache-path: | ||
description: A list of files, directories, and wildcard patterns to cache and restore | ||
required: false | ||
default: '' | ||
type: string | ||
cache-key: | ||
description: An explicit key for restoring and saving the cache | ||
required: false | ||
default: '' | ||
type: string | ||
cache-restore-keys: | ||
description: An ordered list of keys to use for restoring the cache if no cache hit occurred for key | ||
required: false | ||
default: '' | ||
type: string | ||
artifact-path: | ||
description: A list of files, directories, and wildcard patterns to upload as artifacts | ||
required: false | ||
default: '' | ||
type: string | ||
runs-on: | ||
description: Which runner image to use for each OS | ||
required: false | ||
default: '' | ||
type: string | ||
default_python: | ||
description: Default version of Python | ||
required: false | ||
default: '3.x' | ||
type: string | ||
fail-fast: | ||
description: Whether to cancel all in-progress jobs if any job fails | ||
required: false | ||
default: false | ||
type: boolean | ||
timeout-minutes: | ||
description: The maximum number of minutes to let a job run before GitHub automatically cancels it | ||
required: false | ||
default: 360 | ||
type: number | ||
submodules: | ||
description: Whether to checkout submodules | ||
required: false | ||
default: true | ||
type: boolean | ||
checkout_ref: | ||
description: The ref to checkout | ||
required: false | ||
default: '' | ||
type: string | ||
secrets: | ||
CODECOV_TOKEN: | ||
description: Codecov upload token | ||
required: false | ||
jobs: | ||
envs: | ||
name: Load tox environments | ||
runs-on: ubuntu-latest | ||
steps: | ||
- uses: actions/setup-python@0b93645e9fea7318ecaed2b359559ac225c90a2b # v5.3.0 | ||
with: | ||
python-version: '3.12' | ||
- run: python -m pip install PyYAML click packaging | ||
- run: echo $TOX_MATRIX_SCRIPT | base64 --decode > tox_matrix.py | ||
env: | ||
TOX_MATRIX_SCRIPT: import json
import os
import re

import click
import yaml
from packaging.version import InvalidVersion, Version


@click.command()
@click.option("--envs", default="")
@click.option("--libraries", default="")
@click.option("--posargs", default="")
@click.option("--toxdeps", default="")
@click.option("--toxargs", default="")
@click.option("--pytest", default="true")
@click.option("--pytest-results-summary", default="false")
@click.option("--coverage", default="")
@click.option("--conda", default="auto")
@click.option("--setenv", default="")
@click.option("--display", default="false")
@click.option("--cache-path", default="")
@click.option("--cache-key", default="")
@click.option("--cache-restore-keys", default="")
@click.option("--artifact-path", default="")
@click.option("--runs-on", default="")
@click.option("--default-python", default="")
@click.option("--timeout-minutes", default="360")
def load_tox_targets(envs, libraries, posargs, toxdeps, toxargs, pytest, pytest_results_summary,
                     coverage, conda, setenv, display, cache_path, cache_key,
                     cache_restore_keys, artifact_path, runs_on, default_python, timeout_minutes):
    """Script to load tox targets for GitHub Actions workflow."""
    # Load envs config
    envs = yaml.load(envs, Loader=yaml.BaseLoader)
    print(json.dumps(envs, indent=2))

    # Load global libraries config
    global_libraries = {
        "brew": [],
        "brew-cask": [],
        "apt": [],
        "choco": [],
    }
    libraries = yaml.load(libraries, Loader=yaml.BaseLoader)
    if libraries is not None:
        global_libraries.update(libraries)
    print(json.dumps(global_libraries, indent=2))

    # Default images to use for runners
    default_runs_on = {
        "linux": "ubuntu-latest",
        "macos": "macos-latest",
        "windows": "windows-latest",
    }
    custom_runs_on = yaml.load(runs_on, Loader=yaml.BaseLoader)
    if isinstance(custom_runs_on, dict):
        default_runs_on.update(custom_runs_on)
    print(json.dumps(default_runs_on, indent=2))

    # Default string parameters which can be overwritten by each env
    string_parameters = {
        "posargs": posargs,
        "toxdeps": toxdeps,
        "toxargs": toxargs,
        "pytest": pytest,
        "pytest-results-summary": pytest_results_summary,
        "coverage": coverage,
        "conda": conda,
        "setenv": setenv,
        "display": display,
        "cache-path": cache_path,
        "cache-key": cache_key,
        "cache-restore-keys": cache_restore_keys,
        "artifact-path": artifact_path,
        "timeout-minutes": timeout_minutes,
    }

    # Create matrix
    matrix = {"include": []}
    for env in envs:
        matrix["include"].append(get_matrix_item(
            env,
            global_libraries=global_libraries,
            global_string_parameters=string_parameters,
            runs_on=default_runs_on,
            default_python=default_python,
        ))

    # Output matrix
    print(json.dumps(matrix, indent=2))
    with open(os.environ["GITHUB_OUTPUT"], "a") as f:
        f.write(f"matrix={json.dumps(matrix)}\n")


def get_matrix_item(env, global_libraries, global_string_parameters,
                    runs_on, default_python):

    # define spec for each matrix include (+ global_string_parameters)
    item = {
        "os": None,
        "toxenv": None,
        "python_version": None,
        "name": None,
        "pytest_flag": None,
        "libraries_brew": None,
        "libraries_brew_cask": None,
        "libraries_apt": None,
        "libraries_choco": None,
        "cache-path": None,
        "cache-key": None,
        "cache-restore-keys": None,
        "artifact-name": None,
        "artifact-path": None,
        "timeout-minutes": None,
    }
    for string_param, default in global_string_parameters.items():
        env_value = env.get(string_param)
        item[string_param] = default if env_value is None else env_value

    # set os and toxenv
    for k, v in runs_on.items():
        if k in env:
            platform = k
            item["os"] = env.get("runs-on", v)
            item["toxenv"] = env[k]
    assert item["os"] is not None and item["toxenv"] is not None

    # set python_version
    python_version = env.get("python-version")
    m = re.search("^py(2|3)([0-9]+)", item["toxenv"])
    if python_version is not None:
        item["python_version"] = python_version
    elif m is not None:
        major, minor = m.groups()
        item["python_version"] = f"{major}.{minor}"
    else:
        item["python_version"] = env.get("default_python") or default_python

    # if Python is <3.10 we can't use macos-latest which is arm64
    try:
        if Version(item["python_version"]) < Version('3.10') and item["os"] == "macos-latest":
            item["os"] = "macos-12"
    except InvalidVersion:
        # python_version might be for example 'pypy-3.10' which won't parse
        pass

    # set name
    item["name"] = env.get("name") or f'{item["toxenv"]} ({item["os"]})'

    # set artifact-name (replace invalid path characters)
    item["artifact-name"] = re.sub(r"[\\ /:<>|*?\"']", "-", item["name"])
    item["artifact-name"] = re.sub(r"-+", "-", item["artifact-name"])

    # set pytest_flag
    item["pytest_flag"] = ""
    sep = r"\\" if platform == "windows" else "/"
    if item["pytest"] == "true":
        if "codecov" in item.get("coverage", ""):
            item["pytest_flag"] += (
                rf"--cov --cov-report=xml:${{GITHUB_WORKSPACE}}{sep}coverage.xml "
            )
        elif "github" in item.get("coverage", ""):
            item["pytest_flag"] += "--cov "

        if item["pytest-results-summary"] == "true":
            item["pytest_flag"] += rf"--junitxml ${{GITHUB_WORKSPACE}}{sep}results.xml "

    # set libraries
    env_libraries = env.get("libraries")
    if isinstance(env_libraries, str) and len(env_libraries.strip()) == 0:
        env_libraries = {}  # no libraries requested for environment
    libraries = global_libraries if env_libraries is None else env_libraries
    for manager in ["brew", "brew_cask", "apt", "choco"]:
        item[f"libraries_{manager}"] = " ".join(libraries.get(manager, []))

    # set "auto" conda value
    if item["conda"] == "auto":
        item["conda"] = "true" if "conda" in item["toxenv"] else "false"

    # inject toxdeps for conda
    if item["conda"] == "true" and "tox-conda" not in item["toxdeps"].lower():
        item["toxdeps"] = ("tox-conda " + item["toxdeps"]).strip()

    # make timeout-minutes a number
    item["timeout-minutes"] = int(item["timeout-minutes"])

    # verify values
    assert item["pytest"] in {"true", "false"}
    assert item["conda"] in {"true", "false"}
    assert item["display"] in {"true", "false"}

    return item


if __name__ == "__main__":
    load_tox_targets()
 | ||
- run: cat tox_matrix.py | ||
- id: set-outputs | ||
run: | | ||
python tox_matrix.py --envs "${{ inputs.envs }}" --libraries "${{ inputs.libraries }}" \ | ||
--posargs "${{ inputs.posargs }}" --toxdeps "${{ inputs.toxdeps }}" \ | ||
--toxargs "${{ inputs.toxargs }}" --pytest "${{ inputs.pytest }}" \ | ||
--pytest-results-summary "${{ inputs.pytest-results-summary }}" \ | ||
--coverage "${{ inputs.coverage }}" --conda "${{ inputs.conda }}" \ | ||
--setenv "${{ inputs.setenv }}" \ | ||
--display "${{ inputs.display }}" --cache-path "${{ inputs.cache-path }}" \ | ||
--cache-key "${{ inputs.cache-key }}" --cache-restore-keys "${{ inputs.cache-restore-keys }}" \ | ||
--artifact-path "${{ inputs.artifact-path }}" \ | ||
--runs-on "${{ inputs.runs-on }}" --default-python "${{ inputs.default_python }}" \ | ||
--timeout-minutes "${{ inputs.timeout-minutes }}" | ||
shell: sh | ||
outputs: | ||
matrix: ${{ steps.set-outputs.outputs.matrix }} | ||
tox: | ||
name: ${{ matrix.name }} | ||
needs: [envs] | ||
runs-on: ${{ matrix.os }} | ||
timeout-minutes: ${{ matrix.timeout-minutes }} | ||
strategy: | ||
fail-fast: ${{ inputs.fail-fast }} | ||
matrix: ${{fromJSON(needs.envs.outputs.matrix)}} | ||
defaults: | ||
run: | ||
shell: bash -l {0} | ||
steps: | ||
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | ||
with: | ||
fetch-depth: 0 | ||
lfs: true | ||
submodules: ${{ inputs.submodules }} | ||
ref: ${{ inputs.checkout_ref }} | ||
- name: Cache ${{ matrix.cache_key }} | ||
if: ${{ matrix.cache-path != '' && matrix.cache-key != '' }} | ||
uses: actions/cache@6849a6489940f00c2f30c0fb92c6274307ccb58a # v4.1.2 | ||
with: | ||
path: ${{ matrix.cache-path }} | ||
key: ${{ matrix.cache-key }} | ||
restore-keys: ${{ matrix.cache-restore-keys }} | ||
- name: Install dependencies | ||
uses: ConorMacBride/install-package@3e7ad059e07782ee54fa35f827df52aae0626f30 # v1.1.0 | ||
with: | ||
brew: ${{ matrix.libraries_brew }} | ||
brew-cask: ${{ matrix.libraries_brew_cask }} | ||
apt: ${{ matrix.libraries_apt }} | ||
choco: ${{ matrix.libraries_choco }} | ||
- name: Setup Python ${{ matrix.python_version }} | ||
if: ${{ matrix.conda != 'true' }} | ||
uses: actions/setup-python@0b93645e9fea7318ecaed2b359559ac225c90a2b # v5.3.0 | ||
with: | ||
python-version: ${{ matrix.python_version }} | ||
allow-prereleases: true | ||
- name: Setup conda | ||
if: ${{ matrix.conda == 'true' }} | ||
uses: mamba-org/setup-micromamba@06375d89d211a1232ef63355742e9e2e564bc7f7 # v2.0.2 | ||
with: | ||
environment-name: test | ||
condarc: | | ||
channels: | ||
- conda-forge | ||
create-args: >- | ||
conda | ||
python=${{ matrix.python_version }} | ||
tox | ||
init-shell: bash | ||
cache-environment: true | ||
cache-downloads: true | ||
- id: set-env | ||
if: ${{ matrix.setenv != '' }} | ||
run: | | ||
python -m pip install PyYAML | ||
echo $SET_ENV_SCRIPT | base64 --decode > set_env.py | ||
python set_env.py "${{ matrix.setenv }}" | ||
rm set_env.py | ||
env: | ||
SET_ENV_SCRIPT: aW1wb3J0IGpzb24KaW1wb3J0IG9zCmltcG9ydCBzeXMKCmltcG9ydCB5YW1sCgpHSVRIVUJfRU5WID0gb3MuZ2V0ZW52KCJHSVRIVUJfRU5WIikKaWYgR0lUSFVCX0VOViBpcyBOb25lOgogICAgcmFpc2UgVmFsdWVFcnJvcigiR0lUSFVCX0VOViBub3Qgc2V0LiBNdXN0IGJlIHJ1biBpbnNpZGUgR2l0SHViIEFjdGlvbnMuIikKCkRFTElNSVRFUiA9ICJFT0YiCgoKZGVmIHNldF9lbnYoZW52KToKCiAgICBlbnYgPSB5YW1sLmxvYWQoZW52LCBMb2FkZXI9eWFtbC5CYXNlTG9hZGVyKQogICAgcHJpbnQoanNvbi5kdW1wcyhlbnYsIGluZGVudD0yKSkKCiAgICBpZiBub3QgaXNpbnN0YW5jZShlbnYsIGRpY3QpOgogICAgICAgIHRpdGxlID0gImBlbnZgIG11c3QgYmUgbWFwcGluZyIKICAgICAgICBtZXNzYWdlID0gZiJgZW52YCBtdXN0IGJlIG1hcHBpbmcgb2YgZW52IHZhcmlhYmxlcyB0byB2YWx1ZXMsIGdvdCB0eXBlIHt0eXBlKGVudil9IgogICAgICAgIHByaW50KGYiOjplcnJvciB0aXRsZT17dGl0bGV9Ojp7bWVzc2FnZX0iKQogICAgICAgIGV4aXQoMSkKCiAgICBmb3IgaywgdiBpbiBlbnYuaXRlbXMoKToKCiAgICAgICAgaWYgbm90IGlzaW5zdGFuY2Uodiwgc3RyKToKICAgICAgICAgICAgdGl0bGUgPSAiYGVudmAgdmFsdWVzIG11c3QgYmUgc3RyaW5ncyIKICAgICAgICAgICAgbWVzc2FnZSA9IGYiYGVudmAgdmFsdWVzIG11c3QgYmUgc3RyaW5ncywgYnV0IHZhbHVlIG9mIHtrfSBoYXMgdHlwZSB7dHlwZSh2KX0iCiAgICAgICAgICAgIHByaW50KGYiOjplcnJvciB0aXRsZT17dGl0bGV9Ojp7bWVzc2FnZX0iKQogICAgICAgICAgICBleGl0KDEpCgogICAgICAgIHYgPSB2LnNwbGl0KCJcbiIpCgogICAgICAgIHdpdGggb3BlbihHSVRIVUJfRU5WLCAiYSIpIGFzIGY6CiAgICAgICAgICAgIGlmIGxlbih2KSA9PSAxOgogICAgICAgICAgICAgICAgZi53cml0ZShmIntrfT17dlswXX1cbiIpCiAgICAgICAgICAgIGVsc2U6CiAgICAgICAgICAgICAgICBmb3IgbGluZSBpbiB2OgogICAgICAgICAgICAgICAgICAgIGFzc2VydCBsaW5lLnN0cmlwKCkgIT0gREVMSU1JVEVSCiAgICAgICAgICAgICAgICBmLndyaXRlKGYie2t9PDx7REVMSU1JVEVSfVxuIikKICAgICAgICAgICAgICAgIGZvciBsaW5lIGluIHY6CiAgICAgICAgICAgICAgICAgICAgZi53cml0ZShmIntsaW5lfVxuIikKICAgICAgICAgICAgICAgIGYud3JpdGUoZiJ7REVMSU1JVEVSfVxuIikKCiAgICAgICAgcHJpbnQoZiJ7a30gd3JpdHRlbiB0byBHSVRIVUJfRU5WIikKCgppZiBfX25hbWVfXyA9PSAiX19tYWluX18iOgogICAgc2V0X2VudihzeXMuYXJndlsxXSkK | ||
- name: Setup headless display | ||
if: ${{ matrix.display == 'true' }} | ||
uses: pyvista/setup-headless-display-action@4cf5c603091e085da8830e1480355ff03f3e171b # v2 | ||
- name: Install tox | ||
run: python -m pip install --upgrade tox ${{ matrix.toxdeps }} | ||
- run: python -m tox -e ${{ matrix.toxenv }} ${{ matrix.toxargs }} -- ${{ matrix.pytest_flag }} ${{ matrix.posargs }} | ||
- if: ${{ (success() || failure()) && matrix.artifact-path != '' }} | ||
uses: actions/upload-artifact@b4b15b8c7c6ac21ea08fcf65892d2ee8f75cf882 # v4.4.3 | ||
with: | ||
name: ${{ matrix.artifact-name }} | ||
path: ${{ matrix.artifact-path }} | ||
- if: ${{ (success() || failure()) && matrix.pytest-results-summary == 'true' && matrix.pytest == 'true' }} | ||
uses: test-summary/action@31493c76ec9e7aa675f1585d3ed6f1da69269a86 # v2.4 | ||
with: | ||
paths: "**/results.xml" | ||
- name: Upload to Codecov | ||
# Even if tox fails, upload coverage | ||
if: ${{ (success() || failure()) && contains(matrix.coverage, 'codecov') && matrix.pytest == 'true' }} | ||
uses: codecov/codecov-action@015f24e6818733317a2da2edd6290ab26238649a # v5.0.7 | ||
with: | ||
token: ${{ secrets.CODECOV_TOKEN }} | ||
- name: Upload coverage data to GitHub | ||
if: ${{ (success() || failure()) && contains(matrix.coverage, 'github') && matrix.pytest == 'true' }} | ||
uses: actions/upload-artifact@v4 | ||
with: | ||
name: .coverage.${{ github.sha }}-${{ runner.os }}-${{ runner.arch }}-${{ matrix.toxenv }} | ||
path: **/.coverage | ||
report_overall_test_coverage: | ||
needs: [ tox ] | ||
if: always() | ||
name: report overall test coverage | ||
runs-on: ubuntu-latest | ||
steps: | ||
- uses: actions/checkout@v4 | ||
with: | ||
lfs: true | ||
submodules: ${{ inputs.submodules }} | ||
ref: ${{ inputs.checkout_ref }} | ||
- uses: actions/download-artifact@v4 | ||
with: | ||
pattern: .coverage.* | ||
merge-multiple: true | ||
- id: check_downloaded_files | ||
run: | | ||
[ "$(ls -A .coverage*)" ] && exit 0 || exit 1 | ||
continue-on-error: true | ||
- if: steps.check_downloaded_files.outcome == 'success' | ||
uses: actions/setup-python@v5 | ||
with: | ||
python-version: "3.12" | ||
- if: steps.check_downloaded_files.outcome == 'success' | ||
name: generate coverage report | ||
run: | | ||
python -Im pip install --upgrade coverage[toml] | ||
python -Im coverage combine | ||
python -Im coverage report -i -m --format=markdown >> $GITHUB_STEP_SUMMARY | ||
- if: steps.check_downloaded_files.outcome == 'success' | ||
uses: actions/upload-artifact@v4 | ||
with: | ||
name: .coverage | ||
path: .coverage |