Continuous Integration #217
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: "Continuous Integration" | |
on: | |
push: | |
branches: | |
- "**" | |
release: | |
types: | |
- published | |
jobs: | |
test: | |
name: "Test" | |
strategy: | |
matrix: | |
os: [ubuntu-latest, macos-latest] | |
python-version: ["3.9", "3.10", "3.11", "3.12", "3.13"] | |
runs-on: "${{ matrix.os }}" | |
steps: | |
# Check out the code | |
- uses: "actions/checkout@v4" | |
# We need node for some integration tests | |
- uses: "actions/setup-node@v4" | |
# Install python | |
- name: "Set up python" | |
uses: "actions/setup-python@v5" | |
with: | |
python-version: "${{ matrix.python-version }}" | |
- name: "Get Python Path" | |
id: get-py-path | |
shell: bash | |
run: | | |
echo "path=$(which python)" >> $GITHUB_OUTPUT | |
# Set the current month and year (used for cache key) | |
- name: "Get Date" | |
id: get-date | |
# Outputs e.g. "202007" | |
# tbh I have yet to find the docs where this output format is | |
# defined, but I copied this from the official cache action's README. | |
run: | | |
echo "date=$(/bin/date -u '+%Y%m')" >> $GITHUB_OUTPUT | |
shell: bash | |
# Generate the lockfile | |
- name: "Generate Cargo Lockfile" | |
run: "cargo generate-lockfile" | |
# Cache build dependencies | |
- name: "Cache Build Fragments" | |
id: "cache-build-fragments" | |
uses: "actions/cache@v4" | |
with: | |
path: | | |
~/.cargo/registry | |
~/.cargo/git | |
target | |
# Use the OS, the python version, and the hashed cargo lockfile as the | |
# cache key. The Python version shouldn't be necessary, but I have | |
# seen some weird failures in Windows CI where it gets the built | |
# python targets confused. The Python version is included at the | |
# end so it can be partially matched by cache keys in contexts | |
# where we're not iterating over python envs. | |
key: ${{ runner.os }}-${{ contains(runner.os, 'windows') && 'test-' || '' }}cargo-${{ hashFiles('**/Cargo.lock') }}-${{ matrix.python-version }} | |
# Cache `cargo install` built binaries | |
- name: "Cache Built Binaries" | |
id: "cache-binaries" | |
uses: "actions/cache@v4" | |
with: | |
path: "~/.cargo/bin" | |
# In theory, this should rebuild binaries once a month | |
key: "${{ runner.os }}-cargo-binaries-${{steps.get-date.outputs.date}}" | |
- name: Install wasm-pack | |
uses: jetli/[email protected] | |
with: | |
version: "v0.12.1" | |
- name: "Run Tests" | |
if: "${{ !contains(runner.os, 'windows') }}" | |
shell: bash | |
run: "cargo test --all-features" | |
- name: "Run Tests (Windows)" | |
if: "${{ contains(runner.os, 'windows') }}" | |
shell: bash | |
# Python behaves weirdly with setup.py develop in Windows, | |
# when it comes to loading DLLs, so on that platform we build and | |
# install the wheel and run the tests with that. | |
# Running `cargo test --features=wasm` runs all the regular lib | |
# tests plus the WASM integration tests, while excluding the | |
# python integration tests | |
run: | | |
cargo test --features=wasm | |
make develop-py-wheel | |
ls dist/*.whl | |
pip install dist/*.whl | |
echo "Running Tests" | |
python tests/test_py.py | |
env: | |
WINDOWS: "${{ contains(runner.os, 'windows') }}" | |
PYTHON: ${{ steps.get-py-path.outputs.path }} | |
build: | |
name: "Build Libs, WASM, and Python sdist" | |
needs: "test" | |
runs-on: ubuntu-latest | |
steps: | |
# Check out the code | |
- uses: "actions/checkout@v4" | |
# Install python | |
- name: "Set up python" | |
uses: "actions/setup-python@v5" | |
with: | |
python-version: "3.12" | |
- name: "Get Python Path" | |
id: get-py-path | |
shell: bash | |
run: | | |
echo "path=$(which python)" >> $GITHUB_OUTPUT | |
# Set the current month and year (used for cache key) | |
- name: "Get Date" | |
id: get-date | |
# Outputs e.g. "202007" | |
# tbh I have yet to find the docs where this output format is | |
# defined, but I copied this from the official cache action's README. | |
run: | | |
echo "date=$(/bin/date -u '+%Y%m')" >> $GITHUB_OUTPUT | |
shell: bash | |
# Generate the lockfile | |
- name: "Generate Cargo Lockfile" | |
run: "cargo generate-lockfile" | |
# Cache build dependencies | |
- name: "Cache Build Fragments" | |
id: "cache-build-fragments" | |
uses: "actions/cache@v4" | |
with: | |
path: | | |
~/.cargo/registry | |
~/.cargo/git | |
target | |
# This should partial match the caches generated for the tests, | |
# which include a python version at the end. | |
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }} | |
# Cache `cargo install` built binaries | |
- name: "Cache Built Binaries" | |
id: "cache-binaries" | |
uses: "actions/cache@v4" | |
with: | |
path: "~/.cargo/bin" | |
# In theory, this should rebuild binaries once a month | |
key: "${{ runner.os }}-cargo-binaries-${{steps.get-date.outputs.date}}" | |
- name: Install wasm-pack | |
uses: jetli/[email protected] | |
with: | |
version: "v0.12.1" | |
- name: "Build Rust/C Libraries" | |
run: "make build" | |
- name: "Check Rust target content" | |
run: ls target/release | |
- uses: "actions/upload-artifact@v3" | |
name: "Upload Rust/C Libraries" | |
with: | |
path: target/release/libjsonlogic_rs.* | |
name: libs | |
- name: "Build Python Source Dist" | |
run: "make build-py-sdist" | |
env: | |
WINDOWS: "${{ contains(runner.os, 'windows') }}" | |
PYTHON: ${{ steps.get-py-path.outputs.path }} | |
- uses: "actions/upload-artifact@v3" | |
name: "Upload Python sdist" | |
with: | |
path: dist/*.tar.gz | |
name: py-sdist | |
- name: "Build WASM Node Package" | |
run: "make build-wasm" | |
- uses: "actions/upload-artifact@v3" | |
name: "Upload node package" | |
with: | |
path: js/ | |
name: wasm-pkg | |
build-wheels: | |
name: > | |
Build Wheel | |
(${{ matrix.platform.os }}, ${{ matrix.platform.name }}, ${{ matrix.py }}) | |
# needs: test | |
runs-on: ${{ matrix.platform.os }} | |
strategy: | |
fail-fast: false | |
matrix: | |
platform: | |
- { os: ubuntu-latest, name: manylinux } | |
- { os: ubuntu-latest, name: musllinux } | |
- { os: macos-latest, name: macosx } | |
py: | |
- cp39 | |
- cp310 | |
- cp311 | |
- cp312 | |
- cp313 | |
steps: | |
- uses: actions/checkout@v4 | |
- name: Set up QEMU | |
if: runner.os == 'Linux' | |
uses: docker/setup-qemu-action@v3 | |
with: | |
platforms: all | |
- name: Build wheels | |
uses: pypa/[email protected] | |
env: | |
# configure cibuildwheel to build native archs ('auto'), and some | |
# emulated ones | |
CIBW_ARCHS_LINUX: x86_64 aarch64 | |
# cross-compiling wheels for discrete architectures not working OOTB | |
# see: https://github.com/PyO3/setuptools-rust/issues/206 | |
CIBW_ARCHS_MACOS: x86_64 universal2 arm64 | |
CIBW_ARCHS_WINDOWS: AMD64 | |
CIBW_BEFORE_ALL_LINUX: > | |
curl https://sh.rustup.rs -sSf | sh -s -- --default-toolchain stable -y | |
CIBW_BEFORE_ALL_MACOS: | | |
rustup target add aarch64-apple-darwin | |
rustup target add x86_64-apple-darwin | |
CIBW_ENVIRONMENT_LINUX: PATH=/root/.cargo/bin:$PATH | |
CIBW_ENVIRONMENT_MACOS: MACOSX_DEPLOYMENT_TARGET=10.12 | |
CIBW_BUILD: ${{ matrix.py }}-${{ matrix.platform.name }}_* | |
CIBW_TEST_COMMAND: python {project}/tests/test_py.py | |
- uses: actions/upload-artifact@v3 | |
with: | |
name: py-wheels | |
path: ./wheelhouse/*.whl | |
distribute: | |
name: "Distribute Cargo, WASM, and Python Sdist Packages" | |
needs: ["build", "build-wheels"] | |
runs-on: ubuntu-latest | |
if: github.event_name == 'release' && github.event.action == 'published' | |
steps: | |
# Check out the code | |
- uses: "actions/checkout@v4" | |
# Install python | |
- name: "Set up python" | |
uses: "actions/setup-python@v5" | |
with: | |
python-version: "3.12" | |
# Generate the lockfile | |
- name: "Generate Cargo Lockfile" | |
run: "cargo generate-lockfile" | |
- name: "Get Current Version" | |
id: get-version | |
shell: bash | |
run: | | |
echo "version=$(cargo pkgid | tr '#' '\n' | tail -n 1 | tr ':' ' ' | awk '{print $2}')" >> $GITHUB_OUTPUT | |
- name: "(DEBUG) log current version" | |
shell: bash | |
run: | | |
echo "${{ steps.get-version.outputs.version }}" | |
- name: "Check if new Cargo version" | |
id: cargo-version | |
shell: bash | |
run: | | |
echo "new=$(./scripts/newCargoVersion.sh)" >> $GITHUB_OUTPUT | |
- name: "Check if new NPM version" | |
id: npm-version | |
shell: bash | |
run: | | |
echo "new=$(./scripts/newNpmVersion.sh)" >> $GITHUB_OUTPUT | |
# Note we don't check for a new python version b/c there are so | |
# many python artifacts that it is impractical. Instead we just | |
# upload with a `--skip-existing` flag, so if it's already there | |
# it wont' be an error. | |
- name: "(DEBUG) new versions" | |
shell: bash | |
run: | | |
echo "Cargo: ${{ steps.cargo-version.outputs.new }}" | |
echo "NPM: ${{ steps.npm-version.outputs.new }}" | |
- name: "Persist new cargo state for subsequent jobs" | |
shell: bash | |
run: | | |
echo "${{ steps.cargo-version.outputs.new }}" > tmp-new-cargo-ver | |
- uses: "actions/upload-artifact@v3" | |
with: | |
path: "tmp-new-cargo-ver" | |
name: "new-cargo" | |
- name: "Cargo Publish" | |
if: "${{ steps.cargo-version.outputs.new == 'true' }}" | |
run: | | |
cargo publish --token "$CARGO_TOKEN" | |
env: | |
CARGO_TOKEN: "${{ secrets.CARGO_TOKEN }}" | |
- name: "Pull WASM Artifact" | |
uses: "actions/download-artifact@v1" | |
if: "${{ steps.npm-version.outputs.new == 'true' }}" | |
with: | |
name: wasm-pkg | |
path: dist-wasm | |
- name: "Publish NPM Package" | |
shell: bash | |
if: "${{ steps.npm-version.outputs.new == 'true' }}" | |
run: | | |
echo "//registry.npmjs.org/:_authToken=$NPM_TOKEN" > ~/.npmrc | |
npm publish dist-wasm/ --access public | |
env: | |
NPM_TOKEN: "${{ secrets.NPM_TOKEN }}" | |
- name: "Pull Python Sdist Artifact" | |
if: "${{ steps.cargo-version.outputs.new == 'true' }}" | |
uses: "actions/download-artifact@v1" | |
with: | |
name: py-sdist | |
path: dist-py | |
- name: "Publish Python Sdist" | |
if: "${{ steps.cargo-version.outputs.new == 'true' }}" | |
shell: bash | |
run: | | |
pip install twine | |
twine upload --skip-existing dist-py/* | |
env: | |
TWINE_USERNAME: "__token__" | |
TWINE_PASSWORD: "${{ secrets.PYPI_TOKEN }}" | |
distribute-py-wheels: | |
name: "Distribute Python Wheels" | |
needs: ["distribute"] | |
runs-on: ubuntu-latest | |
# upload to PyPI on every tag starting with 'v' | |
# if: github.event_name == 'push' && startsWith(github.event.ref, 'refs/tags/v') | |
# alternatively, to publish when a GitHub Release is created, use the following rule: | |
if: github.event_name == 'release' && github.event.action == 'published' | |
steps: | |
- uses: actions/download-artifact@v3 | |
with: | |
name: py-wheels | |
path: dist | |
- uses: pypa/gh-action-pypi-publish@release/v1 | |
with: | |
user: __token__ | |
password: ${{ secrets.PYPI_TOKEN }} | |
skip_existing: true | |
# To test: repository_url: https://test.pypi.org/legacy/ |