twister: Add unit tests for Pytest harness #92
  
    
      This file contains hidden or 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: Run tests with twister | |
| on: | |
| push: | |
| branches: | |
| - main | |
| - v*-branch | |
| - collab-* | |
| pull_request: | |
| branches: | |
| - main | |
| - v*-branch | |
| - collab-* | |
| schedule: | |
| # Run at 02:00 UTC on every Sunday | |
| - cron: '0 2 * * 0' | |
| permissions: | |
| contents: read | |
| concurrency: | |
| group: ${{ github.workflow }}-${{ github.event_name }}-${{ github.head_ref || github.ref }} | |
| cancel-in-progress: true | |
| jobs: | |
| twister-build-prep: | |
| if: github.repository_owner == 'zephyrproject-rtos' | |
| runs-on: ubuntu-24.04 | |
| outputs: | |
| subset: ${{ steps.output-services.outputs.subset }} | |
| size: ${{ steps.output-services.outputs.size }} | |
| fullrun: ${{ steps.output-services.outputs.fullrun }} | |
| env: | |
| MATRIX_SIZE: 10 | |
| PUSH_MATRIX_SIZE: 20 | |
| WEEKLY_MATRIX_SIZE: 200 | |
| BSIM_OUT_PATH: /opt/bsim/ | |
| BSIM_COMPONENTS_PATH: /opt/bsim/components | |
| TESTS_PER_BUILDER: 900 | |
| COMMIT_RANGE: ${{ github.event.pull_request.base.sha }}..${{ github.event.pull_request.head.sha }} | |
| BASE_REF: ${{ github.base_ref }} | |
| steps: | |
| - name: Checkout | |
| if: github.event_name == 'pull_request' | |
| uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | |
| with: | |
| ref: ${{ github.event.pull_request.head.sha }} | |
| fetch-depth: 0 | |
| path: zephyr | |
| persist-credentials: false | |
| - name: Set up Python | |
| if: github.event_name == 'pull_request' | |
| uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | |
| with: | |
| python-version: 3.12 | |
| cache: pip | |
| cache-dependency-path: scripts/requirements-actions.txt | |
| - name: install-packages | |
| working-directory: zephyr | |
| if: github.event_name == 'pull_request' | |
| run: | | |
| pip install -r scripts/requirements-actions.txt --require-hashes | |
| - name: Setup Zephyr project | |
| if: github.event_name == 'pull_request' | |
| uses: zephyrproject-rtos/action-zephyr-setup@c125c5ebeeadbd727fa740b407f862734af1e52a # v1.0.9 | |
| with: | |
| app-path: zephyr | |
| toolchains: all | |
| enable-ccache: false | |
| - name: Environment Setup | |
| working-directory: zephyr | |
| if: github.event_name == 'pull_request' | |
| run: | | |
| git config --global user.email "[email protected]" | |
| git config --global user.name "Zephyr Bot" | |
| rm -fr ".git/rebase-apply" | |
| rm -fr ".git/rebase-merge" | |
| git rebase origin/${BASE_REF} | |
| git clean -f -d | |
| git log --pretty=oneline | head -n 10 | |
| - name: Generate Test Plan with Twister | |
| working-directory: zephyr | |
| if: github.event_name == 'pull_request' | |
| id: test-plan | |
| run: | | |
| export ZEPHYR_BASE=${PWD} | |
| export ZEPHYR_TOOLCHAIN_VARIANT=zephyr | |
| python3 ./scripts/ci/test_plan.py -c origin/${BASE_REF}.. --pull-request -t $TESTS_PER_BUILDER | |
| if [ -s .testplan ]; then | |
| cat .testplan >> $GITHUB_ENV | |
| else | |
| echo "TWISTER_NODES=${MATRIX_SIZE}" >> $GITHUB_ENV | |
| fi | |
| rm -f testplan.json .testplan | |
| - name: Determine matrix size | |
| id: output-services | |
| run: | | |
| if [ "${{github.event_name}}" = "push" ]; then | |
| subset="[$(seq -s',' 1 ${PUSH_MATRIX_SIZE})]" | |
| size=${MATRIX_SIZE} | |
| elif [ "${{github.event_name}}" = "pull_request" ]; then | |
| if [ -n "${TWISTER_NODES}" ]; then | |
| subset="[$(seq -s',' 1 ${TWISTER_NODES})]" | |
| else | |
| subset="[$(seq -s',' 1 ${MATRIX_SIZE})]" | |
| fi | |
| size=${TWISTER_NODES} | |
| elif [ "${{github.event_name}}" = "schedule" -a "${{github.repository}}" = "zephyrproject-rtos/zephyr" ]; then | |
| subset="[$(seq -s',' 1 ${WEEKLY_MATRIX_SIZE})]" | |
| size=${WEEKLY_MATRIX_SIZE} | |
| else | |
| size=0 | |
| fi | |
| echo "subset=${subset}" >> $GITHUB_OUTPUT | |
| echo "size=${size}" >> $GITHUB_OUTPUT | |
| echo "fullrun=${TWISTER_FULL}" >> $GITHUB_OUTPUT | |
| twister-build: | |
| runs-on: | |
| group: zephyr-runner-v2-linux-x64-4xlarge | |
| needs: twister-build-prep | |
| if: needs.twister-build-prep.outputs.size != 0 | |
| container: | |
| image: ghcr.io/zephyrproject-rtos/ci-repo-cache:v0.28.6.20251003 | |
| options: '--entrypoint /bin/bash' | |
| strategy: | |
| fail-fast: false | |
| matrix: | |
| subset: ${{fromJSON(needs.twister-build-prep.outputs.subset)}} | |
| timeout-minutes: 1440 | |
| env: | |
| CCACHE_DIR: /node-cache/ccache-zephyr | |
| CCACHE_REMOTE_STORAGE: "redis://cache-*.keydb-cache.svc.cluster.local|shards=1,2,3" | |
| CCACHE_REMOTE_ONLY: "true" | |
| # `--specs` is ignored because ccache is unable to resolve the toolchain specs file path. | |
| CCACHE_IGNOREOPTIONS: '-specs=* --specs=*' | |
| BSIM_OUT_PATH: /opt/bsim/ | |
| BSIM_COMPONENTS_PATH: /opt/bsim/components | |
| TWISTER_COMMON: ' --test-config tests/test_config_ci.yaml --force-color --inline-logs -v -N -M --retry-failed 3 --timeout-multiplier 2 ' | |
| WEEKLY_OPTIONS: ' -M --build-only --all --show-footprint --report-filtered -j 32' | |
| PR_OPTIONS: ' --clobber-output --integration -j 16' | |
| PUSH_OPTIONS: ' --clobber-output -M --show-footprint --report-filtered -j 16' | |
| COMMIT_RANGE: ${{ github.event.pull_request.base.sha }}..${{ github.event.pull_request.head.sha }} | |
| BASE_REF: ${{ github.base_ref }} | |
| LLVM_TOOLCHAIN_PATH: /usr/lib/llvm-20 | |
| steps: | |
| - name: Print cloud service information | |
| run: | | |
| echo "ZEPHYR_RUNNER_CLOUD_PROVIDER = ${ZEPHYR_RUNNER_CLOUD_PROVIDER}" | |
| echo "ZEPHYR_RUNNER_CLOUD_NODE = ${ZEPHYR_RUNNER_CLOUD_NODE}" | |
| echo "ZEPHYR_RUNNER_CLOUD_POD = ${ZEPHYR_RUNNER_CLOUD_POD}" | |
| - name: Apply container owner mismatch workaround | |
| run: | | |
| # FIXME: The owner UID of the GITHUB_WORKSPACE directory may not | |
| # match the container user UID because of the way GitHub | |
| # Actions runner is implemented. Remove this workaround when | |
| # GitHub comes up with a fundamental fix for this problem. | |
| git config --global --add safe.directory ${GITHUB_WORKSPACE} | |
| - name: Clone cached Zephyr repository | |
| continue-on-error: true | |
| run: | | |
| git clone --shared /repo-cache/zephyrproject/zephyr . | |
| git remote set-url origin ${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY} | |
| - name: Checkout | |
| uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | |
| with: | |
| ref: ${{ github.event.pull_request.head.sha }} | |
| fetch-depth: 0 | |
| persist-credentials: false | |
| - name: Environment Setup | |
| run: | | |
| if [ "${{github.event_name}}" = "pull_request" ]; then | |
| git config --global user.email "[email protected]" | |
| git config --global user.name "Zephyr Builder" | |
| rm -fr ".git/rebase-apply" | |
| rm -fr ".git/rebase-merge" | |
| git rebase origin/${BASE_REF} | |
| git clean -f -d | |
| git log --pretty=oneline | head -n 10 | |
| fi | |
| echo "$HOME/.local/bin" >> $GITHUB_PATH | |
| echo "$HOME/.cargo/bin" >> $GITHUB_PATH | |
| west init -l . || true | |
| west config manifest.group-filter -- +ci,+optional | |
| west config --global update.narrow true | |
| west update --path-cache /repo-cache/zephyrproject 2>&1 1> west.update.log || west update --path-cache /repo-cache/zephyrproject 2>&1 1> west.update.log || ( rm -rf ../modules ../bootloader ../tools && west update --path-cache /repo-cache/zephyrproject) | |
| west forall -c 'git reset --hard HEAD' | |
| echo "ZEPHYR_SDK_INSTALL_DIR=/opt/toolchains/zephyr-sdk-$( cat SDK_VERSION )" >> $GITHUB_ENV | |
| - name: Check Environment | |
| run: | | |
| cmake --version | |
| gcc --version | |
| cargo --version | |
| rustup target list --installed | |
| ls -la | |
| echo "github.ref: ${{ github.ref }}" | |
| echo "github.base_ref: ${{ github.base_ref }}" | |
| echo "github.ref_name: ${{ github.ref_name }}" | |
| - name: Install Python packages | |
| run: | | |
| pip install -r scripts/requirements-actions.txt --require-hashes | |
| west packages pip --install | |
| - name: Set up ccache | |
| run: | | |
| mkdir -p ${CCACHE_DIR} | |
| ccache -M 10G | |
| ccache -p | |
| ccache -z -s -vv | |
| - name: Update BabbleSim to manifest revision | |
| run: | | |
| export BSIM_VERSION=$( west list bsim -f {revision} ) | |
| echo "Manifest points to bsim sha $BSIM_VERSION" | |
| cd /opt/bsim_west/bsim | |
| git fetch -n origin ${BSIM_VERSION} | |
| git -c advice.detachedHead=false checkout ${BSIM_VERSION} | |
| west update | |
| make everything -s -j 8 | |
| - if: github.event_name == 'push' | |
| name: Run Tests with Twister (Push) | |
| id: run_twister | |
| run: | | |
| export ZEPHYR_BASE=${PWD} | |
| export ZEPHYR_TOOLCHAIN_VARIANT=zephyr | |
| ./scripts/twister --subset ${{matrix.subset}}/${{ strategy.job-total }} ${TWISTER_COMMON} ${PUSH_OPTIONS} | |
| if [ "${{matrix.subset}}" = "1" ]; then | |
| ./scripts/zephyr_module.py --twister-out module_tests.args | |
| if [ -s module_tests.args ]; then | |
| ./scripts/twister +module_tests.args --outdir module_tests ${TWISTER_COMMON} ${PUSH_OPTIONS} | |
| fi | |
| fi | |
| - if: github.event_name == 'pull_request' | |
| name: Run Tests with Twister (Pull Request) | |
| id: run_twister_pr | |
| run: | | |
| rm -f testplan.json | |
| export ZEPHYR_BASE=${PWD} | |
| export ZEPHYR_TOOLCHAIN_VARIANT=zephyr | |
| python3 ./scripts/ci/test_plan.py -c origin/${BASE_REF}.. --pull-request | |
| ./scripts/twister --subset ${{matrix.subset}}/${{ strategy.job-total }} --load-tests testplan.json ${TWISTER_COMMON} ${PR_OPTIONS} | |
| if [ "${{matrix.subset}}" = "1" -a ${{needs.twister-build-prep.outputs.fullrun}} = 'True' ]; then | |
| ./scripts/zephyr_module.py --twister-out module_tests.args | |
| if [ -s module_tests.args ]; then | |
| ./scripts/twister +module_tests.args --outdir module_tests ${TWISTER_COMMON} ${PR_OPTIONS} | |
| fi | |
| fi | |
| - if: github.event_name == 'schedule' | |
| name: Run Tests with Twister (Weekly) | |
| id: run_twister_sched | |
| run: | | |
| export ZEPHYR_BASE=${PWD} | |
| export ZEPHYR_TOOLCHAIN_VARIANT=zephyr | |
| ./scripts/twister --subset ${{matrix.subset}}/${{ strategy.job-total }} ${TWISTER_COMMON} ${WEEKLY_OPTIONS} | |
| if [ "${{matrix.subset}}" = "1" ]; then | |
| ./scripts/zephyr_module.py --twister-out module_tests.args | |
| if [ -s module_tests.args ]; then | |
| ./scripts/twister +module_tests.args --outdir module_tests ${TWISTER_COMMON} ${WEEKLY_OPTIONS} | |
| fi | |
| fi | |
| - name: Print ccache stats | |
| if: always() | |
| run: | | |
| ccache -s -vv | |
| - name: Upload Unit Test Results | |
| if: always() | |
| uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | |
| with: | |
| name: Unit Test Results (Subset ${{ matrix.subset }}) | |
| if-no-files-found: ignore | |
| path: | | |
| twister-out/twister.xml | |
| twister-out/twister.json | |
| module_tests/twister.xml | |
| testplan.json | |
| - if: matrix.subset == 1 && github.event_name == 'push' | |
| name: Save the list of Python packages | |
| shell: bash | |
| run: | | |
| FREEZE_FILE="frozen-requirements.txt" | |
| timestamp="$(date)" | |
| version="$(git describe --abbrev=12 --always)" | |
| echo -e "# Generated at $timestamp ($version)\n" > $FREEZE_FILE | |
| pip freeze | tee -a $FREEZE_FILE | |
| - if: matrix.subset == 1 && github.event_name == 'push' | |
| name: Upload the list of Python packages | |
| uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | |
| with: | |
| name: Frozen PIP package set | |
| path: | | |
| frozen-requirements.txt | |
| twister-test-results: | |
| name: "Publish Unit Tests Results" | |
| needs: | |
| - twister-build | |
| runs-on: ubuntu-24.04 | |
| permissions: | |
| checks: write # to create the check run entry with Twister test results | |
| # the build-and-test job might be skipped, we don't need to run this job then | |
| if: success() || failure() | |
| steps: | |
| - name: Check out source code | |
| uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | |
| with: | |
| ref: ${{ github.event.pull_request.head.sha }} | |
| fetch-depth: 0 | |
| persist-credentials: false | |
| - name: Set up Python | |
| uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | |
| with: | |
| python-version: 3.12 | |
| cache: pip | |
| cache-dependency-path: scripts/requirements-actions.txt | |
| - name: Install Python packages | |
| run: | | |
| pip install -r scripts/requirements-actions.txt --require-hashes | |
| - name: Download Artifacts | |
| uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | |
| with: | |
| path: artifacts | |
| - name: Merge Test Results | |
| run: | | |
| junitparser merge --glob 'artifacts/*/twister.xml' 'artifacts/*/*/twister.xml' junit.xml | |
| junit2html junit.xml junit.html | |
| - name: Upload Unit Test Results | |
| if: always() | |
| uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | |
| with: | |
| name: Unit Test Results | |
| if-no-files-found: ignore | |
| path: | | |
| junit.html | |
| junit.xml | |
| - name: Upload test results to Codecov | |
| if: ${{ !cancelled() && (github.event_name == 'push') }} | |
| uses: codecov/test-results-action@47f89e9acb64b76debcd5ea40642d25a4adced9f # v1.1.1 | |
| with: | |
| token: ${{ secrets.CODECOV_TOKEN }} | |
| - name: Publish Unit Test Results | |
| uses: EnricoMi/publish-unit-test-result-action@3a74b2957438d0b6e2e61d67b05318aa25c9e6c6 # v2.20.0 | |
| with: | |
| check_name: Unit Test Results | |
| files: "**/twister.xml" | |
| comment_mode: off | |
| - name: Analyze Twister Reports | |
| if: needs.twister-build.result == 'failure' | |
| run: | | |
| ./scripts/ci/twister_report_analyzer.py artifacts/*/*/twister.json --long-summary --platforms --output-md errors.md | |
| if [[ -s "errors.md" ]]; then | |
| echo '### Error Summary! 🚀' >> $GITHUB_STEP_SUMMARY | |
| cat errors.md >> $GITHUB_STEP_SUMMARY | |
| fi | |
| - name: Upload Twister Analysis Results | |
| if: needs.twister-build.result == 'failure' | |
| uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | |
| with: | |
| name: Twister Analysis Results | |
| if-no-files-found: ignore | |
| path: | | |
| twister_report_summary.json | |
| twister-status-check: | |
| if: always() | |
| name: "Check Twister Status" | |
| needs: | |
| - twister-build-prep | |
| - twister-build | |
| uses: ./.github/workflows/ready-to-merge.yml | |
| with: | |
| needs_context: ${{ toJson(needs) }} |