Improve compilation times #1137
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
# Copyright (C) 2022 Roberto Rossini (roberros@uio.no) | |
# SPDX-License-Identifier: MIT | |
name: MacOS CI | |
on: | |
push: | |
branches: [main] | |
paths: | |
- ".github/workflows/macos-ci.yml" | |
- "cmake/**" | |
- "examples/**" | |
- "src/**" | |
- "test/integration/**" | |
- "test/units/**" | |
- "CMakeLists.txt" | |
- "conanfile.py" | |
tags: | |
- "v*.*.*" | |
pull_request: | |
paths: | |
- ".github/workflows/macos-ci.yml" | |
- "cmake/**" | |
- "examples/**" | |
- "src/**" | |
- "test/integration/**" | |
- "test/units/**" | |
- "CMakeLists.txt" | |
- "conanfile.py" | |
# https://stackoverflow.com/a/72408109 | |
concurrency: | |
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | |
cancel-in-progress: true | |
env: | |
CCACHE_DIR: "${{ github.workspace }}/ccache-cache" | |
CCACHE_COMPILERCHECK: "content" | |
CCACHE_COMPRESSLEVEL: "1" | |
CCACHE_MAXSIZE: "250M" | |
CONAN_HOME: "${{ github.workspace }}/.conan2" | |
HICTK_CI: "1" | |
HOMEBREW_NO_AUTO_UPDATE: "1" | |
defaults: | |
run: | |
shell: bash | |
jobs: | |
matrix-factory: | |
name: Generate job matrix | |
runs-on: ubuntu-latest | |
outputs: | |
matrix: ${{ steps.set-result.outputs.result }} | |
steps: | |
- uses: actions/github-script@v7 | |
id: set-result | |
with: | |
script: | | |
// Documentation | |
// https://docs.github.com/en/actions/learn-github-actions/contexts#fromjson | |
// https://github.com/actions/runner/issues/982#issuecomment-809360765 | |
var includes = [] | |
includes.push({ compiler_name: 'apple-clang', compiler_version: '14', os: 'macos-12', conan: '2.4.*', cmake: '3.29.*', build_type: 'Release', developer_mode: 'OFF' }) | |
includes.push({ compiler_name: 'apple-clang', compiler_version: '15', os: 'macos-13', conan: '2.4.*', cmake: '3.29.*', build_type: 'Release', developer_mode: 'OFF' }) | |
includes.push({ compiler_name: 'apple-clang', compiler_version: '15', os: 'macos-14', conan: '2.4.*', cmake: '3.29.*', build_type: 'Release', developer_mode: 'OFF' }) | |
includes.push({ compiler_name: 'apple-clang', compiler_version: '15', os: 'macos-13', conan: '2.4.*', cmake: '3.29.*', build_type: 'Debug', developer_mode: 'OFF' }) | |
includes.push({ compiler_name: 'apple-clang', compiler_version: '15', os: 'macos-14', conan: '2.4.*', cmake: '3.29.*', build_type: 'Debug', developer_mode: 'OFF' }) | |
return { include: includes } | |
cache-test-dataset: | |
uses: paulsengroup/hictk/.github/workflows/cache-test-dataset.yml@main | |
name: Cache test dataset | |
build-project: | |
name: Build project | |
needs: matrix-factory | |
runs-on: ${{ matrix.os }} | |
strategy: | |
fail-fast: false | |
matrix: ${{ fromJSON(needs.matrix-factory.outputs.matrix) }} | |
outputs: | |
ccache-old-cache-key: ${{ steps.cache-ccache.outputs.cache-matched-key }} | |
env: | |
CCACHE_DIR: "${{ github.workspace }}/ccache-cache" | |
CCACHE_COMPILERCHECK: "content" | |
CCACHE_COMPRESSLEVEL: "1" | |
CCACHE_MAXSIZE: "250M" | |
CONAN_HOME: "${{ github.workspace }}/.conan2" | |
HOMEBREW_NO_AUTO_UPDATE: "1" | |
steps: | |
- uses: actions/checkout@v4 | |
- name: Generate requirements.txt for pip | |
run: | | |
echo 'conan==${{ matrix.conan }}' > requirements.txt | |
echo 'cmake==${{ matrix.cmake }}' >> requirements.txt | |
- uses: actions/setup-python@v5 | |
with: | |
python-version: "3.12" | |
cache: "pip" | |
- name: Detect number available CPUs | |
run: | | |
ncpus=$(python -c 'import multiprocessing as mp; print(mp.cpu_count())') | |
echo "NPROC=$ncpus" >> $GITHUB_ENV | |
echo "CMAKE_BUILD_PARALLEL_LEVEL=$ncpus" >> $GITHUB_ENV | |
echo "CTEST_PARALLEL_LEVEL=$ncpus" >> $GITHUB_ENV | |
- name: Install build deps | |
run: | | |
pip install -r requirements.txt | |
brew install ccache | |
- name: Generate cache key | |
id: cache-key | |
run: | | |
set -u | |
os="${{ matrix.os }}" | |
compiler="${{ matrix.compiler_name }}" | |
compiler_version="${{ matrix.compiler_version }}" | |
build_type="${{ matrix.build_type }}" | |
conanfile_hash="${{ hashFiles('conanfile.py') }}" | |
# This can be used by to always update a cache entry (useful e.g. for ccache) | |
current_date="$(date '+%s')" | |
conan_key_prefix="conan-$os-$compiler-$compiler_version-$conanfile_hash-$build_type" | |
ccache_key_prefix="ccache-$os-$compiler-$compiler_version-$GITHUB_REF-$conanfile_hash-$build_type" | |
echo "conan-key=$conan_key_prefix" | tee -a $GITHUB_OUTPUT | |
echo "ccache-key=${ccache_key_prefix}-${current_date}" | tee -a $GITHUB_OUTPUT | |
echo "ccache-restore-key=$ccache_key_prefix" | tee -a $GITHUB_OUTPUT | |
- name: Restore Conan cache | |
id: cache-conan | |
uses: actions/cache/restore@v4 | |
with: | |
key: ${{ steps.cache-key.outputs.conan-key }} | |
path: ${{ env.CONAN_HOME }}/p | |
- name: Configure Conan | |
run: conan profile detect --force | |
- name: Clean Conan cache (pre-build) | |
if: steps.cache-conan.outputs.cache-hit != 'true' | |
run: | | |
conan cache clean "*" --build | |
conan cache clean "*" --download | |
conan cache clean "*" --source | |
conan remove --confirm "*" | |
- name: Install build dependencies | |
run: | | |
conan install . \ | |
--build=missing \ | |
-pr default \ | |
-s "build_type=${{ matrix.build_type }}" \ | |
-s "compiler=${{ matrix.compiler_name }}" \ | |
-s "compiler.version=${{ matrix.compiler_version }}" \ | |
-s compiler.libcxx=libc++ \ | |
-s compiler.cppstd=17 \ | |
--output-folder="${{ github.workspace }}/build" | |
- name: Clean Conan cache (post-build) | |
if: steps.cache-conan.outputs.cache-hit != 'true' | |
run: | | |
conan cache clean "*" --build | |
conan cache clean "*" --download | |
conan cache clean "*" --source | |
- name: Save Conan cache | |
uses: actions/cache/save@v4 | |
if: steps.cache-conan.outputs.cache-hit != 'true' | |
with: | |
key: ${{ steps.cache-key.outputs.conan-key }} | |
path: ${{ env.CONAN_HOME }}/p | |
env: | |
ZSTD_CLEVEL: 19 | |
- name: Configure project | |
run: | | |
cmake -DCMAKE_BUILD_TYPE=${{ matrix.build_type }} \ | |
-DCMAKE_PREFIX_PATH="${{ github.workspace }}/build" \ | |
-DENABLE_DEVELOPER_MODE=${{ matrix.developer_mode }} \ | |
-DHICTK_ENABLE_TESTING=ON \ | |
-DHICTK_BUILD_EXAMPLES=ON \ | |
-DOPT_ENABLE_CLANG_TIDY=OFF \ | |
-DOPT_ENABLE_CPPCHECK=OFF \ | |
-DHICTK_DOWNLOAD_TEST_DATASET=OFF \ | |
-DHICTK_ENABLE_GIT_VERSION_TRACKING=OFF \ | |
-DCMAKE_INSTALL_PREFIX=dest \ | |
-S "${{ github.workspace }}" \ | |
-B "${{ github.workspace }}/build" | |
- name: Restore Ccache folder | |
id: cache-ccache | |
uses: actions/cache/restore@v4 | |
with: | |
key: ${{ steps.cache-key.outputs.ccache-restore-key }} | |
path: ${{ env.CCACHE_DIR }} | |
- name: Reset Ccache stats | |
run: ccache --zero-stats | |
- name: Build project | |
run: cmake --build ${{ github.workspace }}/build | |
- name: Package binaries | |
run: | | |
cmake --install build | |
gtar -cf - -C dest/ bin | zstd -T0 -13 -o binaries.tar.zst | |
- name: Package unit tests | |
run: | | |
rm -r build/src | |
gtar -cf - build/ | zstd -T0 -13 -o unit-tests.tar.zst | |
- name: Upload unit tests | |
uses: actions/upload-artifact@v4 | |
with: | |
name: "unit-tests-${{ matrix.os }}-\ | |
${{ matrix.compiler_name }}-\ | |
${{ matrix.compiler_version }}-\ | |
${{ matrix.build_type }}-\ | |
${{ matrix.developer_mode }}" | |
path: unit-tests.tar.zst | |
if-no-files-found: error | |
retention-days: 1 | |
- name: Upload binaries | |
uses: actions/upload-artifact@v4 | |
with: | |
name: "binaries-${{ matrix.os }}-\ | |
${{ matrix.compiler_name }}-\ | |
${{ matrix.compiler_version }}-\ | |
${{ matrix.build_type }}-\ | |
${{ matrix.developer_mode }}" | |
path: binaries.tar.zst | |
if-no-files-found: error | |
retention-days: 1 | |
- name: Print Ccache statistics (pre-cleanup) | |
run: | | |
ccache --show-stats \ | |
--show-compression \ | |
--verbose | |
- name: Cleanup Ccache folder | |
run: | | |
ccache --evict-older-than=14400s # 4h | |
ccache --recompress=19 --recompress-threads="$NPROC" | |
ccache --cleanup | |
- name: Print Ccache statistics (post-cleanup) | |
run: | | |
ccache --show-stats \ | |
--show-compression \ | |
--verbose | |
- name: Save Ccache folder | |
uses: actions/cache/save@v4 | |
with: | |
key: ${{ steps.cache-key.outputs.ccache-key }} | |
path: ${{ env.CCACHE_DIR }} | |
env: | |
ZSTD_CLEVEL: 1 | |
clean-stale-cache: | |
needs: [build-project] | |
uses: paulsengroup/hictk/.github/workflows/evict-gha-cache.yml@main | |
name: Clean stale Ccache cache | |
permissions: | |
actions: write | |
if: needs.build-project.outputs.ccache-old-cache-key != '' | |
with: | |
cache-key: "${{ needs.build-project.outputs.ccache-old-cache-key }}" | |
run-unit-tests: | |
name: Run unit tests | |
needs: [matrix-factory, cache-test-dataset, build-project] | |
runs-on: ${{ matrix.os }} | |
strategy: | |
fail-fast: false | |
matrix: ${{ fromJSON(needs.matrix-factory.outputs.matrix) }} | |
env: | |
HICTK_CI: "1" | |
steps: | |
- name: Checkout repo | |
uses: actions/checkout@v4 | |
- name: Restore test dataset | |
uses: actions/cache/restore@v4 | |
with: | |
key: ${{ needs.cache-test-dataset.outputs.cache-key }} | |
path: test/data/hictk_test_data.tar.zst | |
fail-on-cache-miss: true | |
- name: Download unit tests artifact | |
uses: actions/download-artifact@v4 | |
with: | |
name: "unit-tests-${{ matrix.os }}-\ | |
${{ matrix.compiler_name }}-\ | |
${{ matrix.compiler_version }}-\ | |
${{ matrix.build_type }}-\ | |
${{ matrix.developer_mode }}" | |
- name: Extract binaries test dataset | |
run: | | |
gtar -xf unit-tests.tar.zst | |
gtar -xf test/data/hictk_test_data.tar.zst | |
- name: Generate requirements.txt for pip | |
run: | | |
echo 'cmake==${{ matrix.cmake }}' > requirements.txt | |
- uses: actions/setup-python@v5 | |
with: | |
python-version: "3.12" | |
cache: "pip" | |
- name: Install test dependencies | |
run: pip install -r requirements.txt | |
- name: Run unit tests | |
run: | | |
ctest --test-dir build/ \ | |
--schedule-random \ | |
--output-on-failure \ | |
--no-tests=error \ | |
--timeout 400 2>&1 | | |
head -n 1000 | |
run-integration-tests: | |
name: Run integration tests | |
needs: [matrix-factory, cache-test-dataset, build-project] | |
runs-on: ${{ matrix.os }} | |
strategy: | |
fail-fast: false | |
matrix: ${{ fromJSON(needs.matrix-factory.outputs.matrix) }} | |
env: | |
HICTK_CI: "1" | |
steps: | |
- name: Checkout repo | |
uses: actions/checkout@v4 | |
- uses: actions/setup-python@v5 | |
with: | |
python-version: "3.12" | |
cache-dependency-path: "test/integration/pyproject.toml" | |
cache: "pip" | |
- name: Restore test dataset | |
uses: actions/cache/restore@v4 | |
with: | |
key: ${{ needs.cache-test-dataset.outputs.cache-key }} | |
path: test/data/hictk_test_data.tar.zst | |
fail-on-cache-miss: true | |
- name: Download binaries artifact | |
uses: actions/download-artifact@v4 | |
with: | |
name: "binaries-${{ matrix.os }}-\ | |
${{ matrix.compiler_name }}-\ | |
${{ matrix.compiler_version }}-\ | |
${{ matrix.build_type }}-\ | |
${{ matrix.developer_mode }}" | |
- name: Extract binaries test dataset | |
run: | | |
gtar -xf binaries.tar.zst | |
gtar -xf test/data/hictk_test_data.tar.zst | |
- name: Install test suite | |
run: | | |
python3.12 -m venv venv --upgrade | |
venv/bin/pip install test/integration | |
- name: Detect number available CPUs | |
run: | | |
ncpus=$(python -c 'import multiprocessing as mp; print(mp.cpu_count())') | |
echo "NPROC=$ncpus" | tee -a $GITHUB_ENV | |
- name: Test hictk balance | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.balance.json \ | |
--suites=balance | |
- name: Test hictk convert | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.convert.json \ | |
--suites=convert | |
- name: Test hictk dump | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.dump.json \ | |
--suites=dump | |
- name: Test hictk fix-mcool | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.fix-mcool.json \ | |
--suites=fix-mcool | |
- name: Test hictk load | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.load.json \ | |
--suites=load | |
- name: Test hictk merge | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.merge.json \ | |
--suites=merge | |
- name: Test hictk metadata | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.metadata.json \ | |
--suites=metadata | |
- name: Test hictk rename-chromosomes | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.rename-chromosomes.json \ | |
--suites=rename-chromosomes | |
- name: Test hictk validate | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.validate.json \ | |
--suites=validate | |
- name: Test hictk zoomify | |
run: | | |
venv/bin/hictk_integration_suite \ | |
bin/hictk \ | |
test/integration/config.toml \ | |
--data-dir test/data \ | |
--threads "$NPROC" \ | |
--result-file integration-test-report.zoomify.json \ | |
--suites=zoomify | |
macos-ci-status-check: | |
name: Status Check (MacOS CI) | |
if: ${{ always() }} | |
runs-on: ubuntu-latest | |
needs: | |
- matrix-factory | |
- cache-test-dataset | |
- build-project | |
- run-unit-tests | |
- run-integration-tests | |
steps: | |
- name: Collect job results | |
if: | | |
needs.matrix-factory.result != 'success' || | |
needs.cache-test-dataset.result != 'success' || | |
needs.build-project.result != 'success' || | |
needs.run-unit-tests.result != 'success' || | |
needs.run-integration-tests.result != 'success' | |
run: exit 1 |