Skip to content

Removing pretty-format-json hook tampering with notebooks on github #332

Removing pretty-format-json hook tampering with notebooks on github

Removing pretty-format-json hook tampering with notebooks on github #332

Workflow file for this run

name: Build discrete-optimization
on:
push:
branches:
- "**"
tags:
- 'v[0-9]+.[0-9]+.[0-9]+'
pull_request:
workflow_dispatch:
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: true
jobs:
linters:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: create requirements.txt so that pip cache with setup-python works
run: echo "pre-commit" > requirements_precommit.txt
- uses: actions/setup-python@v4
with:
python-version: "3.8"
cache: "pip"
cache-dependency-path: requirements_precommit.txt
- name: install pre-commit
run: python -m pip install pre-commit
- name: get cached pre-commit hooks
uses: actions/cache@v3
with:
path: ~/.cache/pre-commit
key: pre-commit|${{ env.pythonLocation }}|${{ hashFiles('.pre-commit-config.yaml') }}
- name: pre-commit checks
run: pre-commit run --show-diff-on-failure --color=always --all-files
trigger:
# store trigger reason
runs-on: ubuntu-latest
outputs:
is_release: ${{ steps.reason.outputs.is_release }}
is_push_on_default_branch: ${{ steps.reason.outputs.is_push_on_default_branch }}
steps:
- id: reason
run: |
echo "is_release=${{ startsWith(github.ref, 'refs/tags/v') }}" >> $GITHUB_OUTPUT
echo "is_push_on_default_branch=${{ github.ref == format('refs/heads/{0}', github.event.repository.default_branch) }}" >> $GITHUB_OUTPUT
build:
runs-on: ubuntu-latest
needs: trigger
outputs:
do_version: ${{ steps.get_wheel_version.outputs.version }}
steps:
- name: Checkout source code
uses: actions/checkout@v3
- name: create requirements.txt so that pip cache with setup-python works
run:
echo "build" > requirements_build.txt
- uses: actions/setup-python@v4
with:
python-version: "3.8"
cache: "pip"
cache-dependency-path: requirements_build.txt
- name: Install build dependencies
run: pip install -U build
- name: Update version number according to pushed tag
if: needs.trigger.outputs.is_release == 'true'
run: |
VERSION=${GITHUB_REF/refs\/tags\/v/} # stripping "refs/tags/v"
echo $VERSION
# Replace in-place version number in package __init__.py, also used by pyproject.toml
sed -i -e "s/^__version__\s*=.*$/__version__ = \"${VERSION}\"/g" discrete_optimization/__init__.py
cat discrete_optimization/__init__.py
- name: Build discrete-optimization wheel
run: python -m build --wheel
- name: Upload as build artifacts
uses: actions/upload-artifact@v3
with:
name: wheels
path: dist/*.whl
- name: get wheel version and save it
id: get_wheel_version
run: |
wheelfile=$(ls ./dist/discrete_optimization*.whl)
version=$(python -c "print('$wheelfile'.split('-')[1])")
echo "version=$version"
echo "version=$version" >> $GITHUB_OUTPUT
test:
needs: build
strategy:
fail-fast: false
matrix:
os: ["ubuntu-latest", "macos-latest", "windows-latest"]
python-version: ["3.7", "3.8", "3.9"]
wo_gurobi: ["", "without gurobi"]
include:
- os: "ubuntu-latest"
minizinc_config_cmdline: export PATH=$PATH:$(pwd)/bin/squashfs-root/usr/bin; export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$(pwd)/bin/squashfs-root/usr/lib
minizinc_cache_path: $(pwd)/bin/squashfs-root
minizinc_url: https://github.com/MiniZinc/MiniZincIDE/releases/download/2.6.3/MiniZincIDE-2.6.3-x86_64.AppImage
minizinc_downloaded_filepath: bin/minizinc.AppImage
minizinc_install_cmdline: cd bin; sudo chmod +x minizinc.AppImage; sudo ./minizinc.AppImage --appimage-extract; cd ..
- os: "macos-latest"
minizinc_config_cmdline: export PATH=$PATH:$(pwd)/bin/MiniZincIDE.app/Contents/Resources
minizinc_cache_path: $(pwd)/bin/MiniZincIDE.app
minizinc_url: https://github.com/MiniZinc/MiniZincIDE/releases/download/2.6.3/MiniZincIDE-2.6.3-bundled.dmg
minizinc_downloaded_filepath: bin/minizinc.dmg
minizinc_install_cmdline: sudo hdiutil attach bin/minizinc.dmg; sudo cp -R /Volumes/MiniZinc*/MiniZincIDE.app bin/.
- os: "windows-latest"
minizinc_config_cmdline: export PATH=$PATH:~/AppData/Local/Programs/MiniZinc
minizinc_cache_path: ~/AppData/Local/Programs/MiniZinc
minizinc_url: https://github.com/MiniZinc/MiniZincIDE/releases/download/2.6.3/MiniZincIDE-2.6.3-bundled-setup-win64.exe
minizinc_downloaded_filepath: minizinc_setup.exe
minizinc_install_cmdline: cmd //c "minizinc_setup.exe /verysilent /currentuser /norestart /suppressmsgboxes /sp"
- coverage: false # generally no coverage to avoid multiple reports
- coverage: true # coverage only for one entry of the matrix
os: "ubuntu-latest"
python-version: "3.9"
wo_gurobi: ""
exclude:
- os: "windows-latest"
wo_gurobi: "without gurobi"
- os: "macos-latest"
wo_gurobi: "without gurobi"
- os: "ubuntu-latest"
wo_gurobi: "without gurobi"
python-version: 3.7
- os: "ubuntu-latest"
wo_gurobi: "without gurobi"
python-version: 3.8
runs-on: ${{ matrix.os }}
defaults:
run:
shell: bash
steps:
- name: Checkout discrete-optimization source code
uses: actions/checkout@v3
- run: rm -rf $RUNNER_TOOL_CACHE/Python/3.7.17
- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
with:
python-version: ${{ matrix.python-version }}
cache: "pip"
cache-dependency-path: pyproject.toml
- name: Download artifacts
uses: actions/download-artifact@v3
with:
name: wheels
path: wheels
- name: Install only discrete-optimization
run: |
python -m pip install -U pip
wheelfile=$(ls ./wheels/discrete_optimization*.whl)
pip install ${wheelfile}
- name: Check import work without minizinc if DO_SKIP_MZN_CHECK set
run: |
export DO_SKIP_MZN_CHECK=1
python -c "import discrete_optimization"
- name: Check import fails without minizinc if DO_SKIP_MZN_CHECK unset
run: |
python -c "
try:
import discrete_optimization
except RuntimeError:
pass
else:
raise AssertionError('We should not be able to import discrete_optimization without minizinc being installed.')
"
- name: Create bin/
run: mkdir -p bin
- name: get MininZinc path to cache
id: get-mzn-cache-path
run: |
echo "path=${{ matrix.minizinc_cache_path }}" >> $GITHUB_OUTPUT # expands variables
- name: Restore MiniZinc cache
id: cache-minizinc
uses: actions/cache@v3
with:
path: ${{ steps.get-mzn-cache-path.outputs.path }}
key: ${{ matrix.minizinc_url }}
- name: Download MiniZinc
if: steps.cache-minizinc.outputs.cache-hit != 'true'
run: |
curl -o "${{ matrix.minizinc_downloaded_filepath }}" -L ${{ matrix.minizinc_url }}
- name: Install MiniZinc
if: steps.cache-minizinc.outputs.cache-hit != 'true'
run: |
${{ matrix.minizinc_install_cmdline }}
- name: Test minizinc install
run: |
${{ matrix.minizinc_config_cmdline }}
minizinc --version
- name: Check imports are working
run: |
# configure minizinc
${{ matrix.minizinc_config_cmdline }}
# check imports
python tests/test_import_all_submodules.py
- name: Install test dependencies
run: |
wheelfile=$(ls ./wheels/discrete_optimization*.whl)
pip install ${wheelfile}[test]
if [ "${{ matrix.wo_gurobi }}" != "without gurobi" ]; then
echo "install gurobi"
pip install gurobipy
python -c "import gurobipy"
fi
- name: Restore tests data cache
id: cache-data
uses: actions/cache@v3
with:
path: ~/discrete_optimization_data
key: data-${{ hashFiles('discrete_optimization/datasets.py') }}
- name: Fetch data for tests
if: steps.cache-data.outputs.cache-hit != 'true'
run: |
${{ matrix.minizinc_config_cmdline }}
python -m discrete_optimization.datasets
- name: Test with pytest (no coverage)
if: ${{ !matrix.coverage }}
run: |
# configure minizinc
${{ matrix.minizinc_config_cmdline }}
# show library path used
pytest -s tests/show_do_path.py
# run test suite
MPLBACKEND="agg" NUMBA_BOUNDSCHECK=1 pytest \
-v --durations=0 --durations-min=10 \
tests
- name: Test with pytest (with coverage)
if: ${{ matrix.coverage }}
run: |
# configure minizinc
${{ matrix.minizinc_config_cmdline }}
# create a tmp directory from which running the tests
# so that "--cov discrete_optimization" look for package installed via the wheel
# instead of the source directory in the repository (which would lead to a coverage of 0%)
mkdir -p tmp && cd tmp
# show library path used
pytest -s ../tests/show_do_path.py
# run test suite
MPLBACKEND="agg" NUMBA_BOUNDSCHECK=1 pytest \
--cov discrete_optimization \
--cov-report xml:coverage.xml \
--cov-report html:coverage_html \
--cov-report term \
-v --durations=0 --durations-min=10 \
../tests
cd ..
- name: Upload coverage report as artifact
if: ${{ matrix.coverage }}
uses: actions/upload-artifact@v3
with:
name: coverage
path: |
tmp/coverage.xml
tmp/coverage_html
create-binder-env:
runs-on: ubuntu-latest
needs: [trigger, build]
env:
BINDER_ENV_DEFAULT_REF: binder # default reference: binder branch
outputs:
binder_env_ref: ${{ steps.get_binder_env_ref.outputs.binder_env_ref }}
steps:
- name: Initialize binder env reference
run: echo "BINDER_ENV_REF=$BINDER_ENV_DEFAULT_REF" >> $GITHUB_ENV
- name: Checkout binder branch
if: |
(needs.trigger.outputs.is_push_on_default_branch == 'true')
|| (needs.trigger.outputs.is_release == 'true')
uses: actions/checkout@v3
with:
ref: ${{ env.BINDER_ENV_DEFAULT_REF }}
- name: Update environment.yml and binder env reference
if: |
(needs.trigger.outputs.is_push_on_default_branch == 'true')
|| (needs.trigger.outputs.is_release == 'true')
run: |
if ${{ needs.trigger.outputs.is_release == 'true' }}; then
# RELEASE MODE
# Specify the proper discrete-optimization version
sed_replacement_pattern="\1- discrete-optimization==${{ needs.build.outputs.do_version }}"
if ${{ github.repository != 'airbus/discrete-optimization' && secrets.TEST_PYPI_API_TOKEN != '' }} == 'true'; then
# release to be deployed on TestPyPI
sed_replacement_pattern="${sed_replacement_pattern}\n\1- --extra-index-url https://test.pypi.org/simple/"
fi
sed_command="s|\(\s*\)-.*egg=discrete-optimization$|${sed_replacement_pattern}|"
echo $sed_command
sed -i -e "$sed_command" environment.yml
cat environment.yml
# Commit new environment
git config user.name "Actions"
git config user.email "[email protected]"
git commit environment.yml -m "Specify binder environment for release ${{ needs.build.outputs.do_version }}"
# Get sha1 to be used by binder for the environment, update the binder env reference
echo "BINDER_ENV_REF=$(git rev-parse --verify HEAD)" >> $GITHUB_ENV
# Revert for the default branch binder env
git revert HEAD -n
git commit -m "Specify binder environment for default branch"
else
# DEFAULT BRANCH MODE
# Specify the proper discrete-optimization version (current master sha1)
pip_spec="git+${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}@${GITHUB_SHA}#egg=discrete-optimization"
echo $pip_spec
sed -i -e "s|\(\s*\)-.*egg=discrete-optimization$|\1- ${pip_spec}|" environment.yml
cat environment.yml
# Commit new environment
git config user.name "Actions"
git config user.email "[email protected]"
git add environment.yml
if ! git diff-index --quiet HEAD -- environment.yml; then
# commit only if a difference (else would raise an error)
# no diff could happen if relaunching the whole workflow while this action was successful
git commit -m "Specify binder environment for default branch"
fi
fi
# Push binder branch with the updated environment
git push origin $BINDER_ENV_DEFAULT_REF
- name: Store binder env reference
id: get_binder_env_ref
run: |
echo "binder_env_ref=$BINDER_ENV_REF"
echo "binder_env_ref=$BINDER_ENV_REF" >> $GITHUB_OUTPUT
- uses: actions/checkout@v3 # checkout triggering branch to get scripts/trigger_binder.sh
- name: Trigger a build for default binder env ref on each BinderHub deployments in the mybinder.org federation
continue-on-error: true
if: |
(needs.trigger.outputs.is_push_on_default_branch == 'true')
|| (needs.trigger.outputs.is_release == 'true')
run: |
binder_env_full_ref=${GITHUB_REPOSITORY}/${BINDER_ENV_DEFAULT_REF}
echo Triggering binder environment build for ${binder_env_full_ref}
binderhubs="gke gke2 ovh ovh2 turing gesis"
return_code=0
for binderhub in $binderhubs; do
echo "** on ${binderhub}"
# go on even though the script crashes
bash scripts/trigger_binder.sh https://${binderhub}.mybinder.org/build/gh/${binder_env_full_ref} || ret_code=$?
# remember the potential crash
if [ $ret_code != 0 ]; then return_code=$ret_code; fi
done
# exit with last non-zero return code if any
exit $return_code
build-doc:
runs-on: ubuntu-latest
needs: create-binder-env
env:
python-version: "3.9"
minizinc_config_cmdline: export PATH=$PATH:$(pwd)/bin/squashfs-root/usr/bin; export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$(pwd)/bin/squashfs-root/usr/lib
minizinc_cache_path: $(pwd)/bin/squashfs-root
minizinc_url: https://github.com/MiniZinc/MiniZincIDE/releases/download/2.6.3/MiniZincIDE-2.6.3-x86_64.AppImage
minizinc_downloaded_filepath: bin/minizinc.AppImage
minizinc_install_cmdline: cd bin; sudo chmod +x minizinc.AppImage; sudo ./minizinc.AppImage --appimage-extract; cd ..
steps:
- name: Set env variables for github+binder links in doc
run: |
# binder environment repo and branch
AUTODOC_BINDER_ENV_GH_REPO_NAME="${GITHUB_REPOSITORY}"
AUTODOC_BINDER_ENV_GH_BRANCH="${{ needs.create-binder-env.outputs.binder_env_ref }}"
# notebooks source repo and branch depending if it is a commit push or a PR
if [[ $GITHUB_REF == refs/pull* ]];
then
AUTODOC_NOTEBOOKS_REPO_URL="${GITHUB_SERVER_URL}/${{ github.event.pull_request.head.repo.full_name }}"
AUTODOC_NOTEBOOKS_BRANCH=${GITHUB_HEAD_REF}
elif [[ $GITHUB_REF == refs/heads* ]];
then
AUTODOC_NOTEBOOKS_REPO_URL=${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}
AUTODOC_NOTEBOOKS_BRANCH=${GITHUB_REF/refs\/heads\//}
elif [[ $GITHUB_REF == refs/tags* ]];
then
AUTODOC_NOTEBOOKS_REPO_URL=${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}
AUTODOC_NOTEBOOKS_BRANCH=${GITHUB_REF/refs\/tags\//}
fi
# export in GITHUB_ENV for next steps
echo "AUTODOC_BINDER_ENV_GH_REPO_NAME=${AUTODOC_BINDER_ENV_GH_REPO_NAME}" >> $GITHUB_ENV
echo "AUTODOC_BINDER_ENV_GH_BRANCH=${AUTODOC_BINDER_ENV_GH_BRANCH}" >> $GITHUB_ENV
echo "AUTODOC_NOTEBOOKS_REPO_URL=${AUTODOC_NOTEBOOKS_REPO_URL}" >> $GITHUB_ENV
echo "AUTODOC_NOTEBOOKS_BRANCH=${AUTODOC_NOTEBOOKS_BRANCH}" >> $GITHUB_ENV
# check computed variables
echo "Binder env: ${AUTODOC_BINDER_ENV_GH_REPO_NAME}/${AUTODOC_BINDER_ENV_GH_BRANCH}"
echo "Notebooks source: ${AUTODOC_NOTEBOOKS_REPO_URL}/tree/${AUTODOC_NOTEBOOKS_BRANCH}"
- uses: actions/checkout@v3
with:
submodules: true
- name: Setup python
uses: actions/setup-python@v4
with:
python-version: ${{ env.python-version }}
cache: "pip"
cache-dependency-path: |
pyproject.toml
docs/requirements.txt
- name: Download artifacts
uses: actions/download-artifact@v3
with:
name: wheels
path: wheels
- name: Create bin/
run: mkdir -p bin
- name: get MininZinc path to cache
id: get-mzn-cache-path
run: |
echo "path=${{ env.minizinc_cache_path }}" >> $GITHUB_OUTPUT # expands variables
- name: Restore MiniZinc cache
id: cache-minizinc
uses: actions/cache@v3
with:
path: ${{ steps.get-mzn-cache-path.outputs.path }}
key: ${{ env.minizinc_url }}
- name: Download MiniZinc
if: steps.cache-minizinc.outputs.cache-hit != 'true'
run: |
curl -o "${{ env.minizinc_downloaded_filepath }}" -L ${{ env.minizinc_url }}
- name: Install MiniZinc
if: steps.cache-minizinc.outputs.cache-hit != 'true'
run: |
${{ env.minizinc_install_cmdline }}
- name: Install doc dependencies
run: |
python -m pip install -U pip
wheelfile=$(ls ./wheels/discrete_optimization*.whl)
pip install ${wheelfile}
pip install -r docs/requirements.txt
- name: generate documentation
run: |
# configure minzinc (beware: paths are relative to default working directory)
${{ env.minizinc_config_cmdline }}
# move to documentation directory
cd docs
# generate api doc source files
sphinx-apidoc -o source/api -f -T ../discrete_optimization
# generate available notebooks list
python generate_nb_index.py
# build doc html pages
sphinx-build -M html source build
# specify it is a nojekyll site
touch build/html/.nojekyll
- name: upload as artifact
uses: actions/upload-artifact@v3
with:
name: doc
path: docs/build/html
deploy-master-doc:
# for default branch
needs: [trigger, create-binder-env, build-doc, test]
if: needs.trigger.outputs.is_push_on_default_branch == 'true'
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Download artifacts
uses: actions/download-artifact@v3
with:
name: doc
path: docs/build/html
- name: Deploy documentation in a version subfolder on GH pages
uses: JamesIves/github-pages-deploy-action@v4
with:
branch: gh-pages # The branch the action should deploy to.
folder: docs/build/html # The folder the action should deploy.
target-folder: /${{ github.event.repository.default_branch }} # The folder the action should deploy to.
commit-message: publish documentation for ${{ github.event.repository.default_branch }}
deploy:
# for release tags
runs-on: ubuntu-latest
needs: [trigger, test]
if: needs.trigger.outputs.is_release == 'true'
steps:
- name: Download wheels artifact
uses: actions/download-artifact@v3
with:
name: wheels
path: wheels
- name: Create the github release
uses: ncipollo/release-action@v1
with:
artifacts: wheels/*.whl
generateReleaseNotes: true
- name: Publish package to TestPyPI (only for forks)
env:
TEST_PYPI_API_TOKEN: ${{ secrets.TEST_PYPI_API_TOKEN }}
if: github.repository != 'airbus/discrete-optimization' && env.TEST_PYPI_API_TOKEN != ''
uses: pypa/gh-action-pypi-publish@release/v1
with:
password: ${{ secrets.TEST_PYPI_API_TOKEN }}
repository_url: https://test.pypi.org/legacy/
packages_dir: wheels/
- name: Publish package to PyPI (main repo)
env:
PYPI_API_TOKEN: ${{ secrets.PYPI_API_TOKEN }}
if: github.repository == 'airbus/discrete-optimization' && env.PYPI_API_TOKEN != ''
uses: pypa/gh-action-pypi-publish@release/v1
with:
password: ${{ secrets.PYPI_API_TOKEN }}
packages_dir: wheels/
deploy-release-doc:
# for release tags
needs: [trigger, build, create-binder-env, build-doc, deploy]
if: needs.trigger.outputs.is_release == 'true'
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Trigger a prebuild of release environment on each BinderHub deployments in the mybinder.org federation (now that the release is available)
run: |
binder_env_full_ref=${GITHUB_REPOSITORY}/${{ needs.create-binder-env.outputs.binder_env_ref }}
echo Triggering binder environment build for ${binder_env_full_ref}
bash scripts/trigger_binder.sh https://gke.mybinder.org/build/gh/${binder_env_full_ref}
bash scripts/trigger_binder.sh https://ovh.mybinder.org/build/gh/${binder_env_full_ref}
bash scripts/trigger_binder.sh https://turing.mybinder.org/build/gh/${binder_env_full_ref}
bash scripts/trigger_binder.sh https://gesis.mybinder.org/build/gh/${binder_env_full_ref}
- name: Download artifacts
uses: actions/download-artifact@v3
with:
name: doc
path: docs/build/html
- name: Deploy documentation in a version subfolder on GH pages
uses: JamesIves/github-pages-deploy-action@v4
with:
branch: gh-pages # The branch the action should deploy to.
folder: docs/build/html # The folder the action should deploy.
target-folder: /${{ needs.build.outputs.do_version }} # The folder the action should deploy to.
commit-message: publish documentation for ${{ needs.build.outputs.do_version }}
update-doc-versions:
# triggers even if only one needed job succeeded (typically at least one will be skipped)
needs: [deploy-release-doc, deploy-master-doc]
if: |
always()
&& (
(needs.deploy-master-doc.result == 'success')
|| (needs.deploy-release-doc.result == 'success')
)
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
with:
ref: gh-pages
- uses: actions/setup-python@v4
- name: Generate versions.json
shell: python3 {0}
run: |
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)
target_file = target_dir / 'versions.json'
with target_file.open('w') as f:
json.dump(versions, f)
- name: Commit versions.json
uses: JamesIves/github-pages-deploy-action@v4
with:
branch: gh-pages # The branch the action should deploy to.
folder: gh-pages # The folder the action should deploy.
target-folder: / # The folder the action should deploy to.
commit-message: update versions.json
clean: false # do not remove other files (including whole doc versions)