name: '๐Ÿ”ง CI' on: push: branches: [master] pull_request: merge_group: # Optimizes CI performance by canceling redundant workflow runs # If another push to the same PR or branch happens while this workflow is still running, # cancel the earlier run in favor of the next run. # # There's no point in testing an outdated version of the code. GitHub only allows # a limited number of job runners to be active at the same time, so it's better to cancel # pointless jobs early so that more useful jobs can run sooner. concurrency: group: ${{ github.workflow }}-${{ github.ref }} cancel-in-progress: true permissions: contents: read env: UV_FROZEN: "true" UV_NO_SYNC: "true" jobs: # This job analyzes which files changed and creates a dynamic test matrix # to only run tests/lints for the affected packages, improving CI efficiency build: name: 'Detect Changes & Set Matrix' runs-on: ubuntu-latest if: ${{ !contains(github.event.pull_request.labels.*.name, 'ci-ignore') }} steps: - name: '๐Ÿ“‹ Checkout Code' uses: actions/checkout@v4 - name: '๐Ÿ Setup Python 3.11' uses: actions/setup-python@v5 with: python-version: '3.11' - name: '๐Ÿ“‚ Get Changed Files' id: files uses: Ana06/get-changed-files@v2.3.0 - name: '๐Ÿ” Analyze Changed Files & Generate Build Matrix' id: set-matrix run: | python -m pip install packaging requests python .github/scripts/check_diff.py ${{ steps.files.outputs.all }} >> $GITHUB_OUTPUT outputs: lint: ${{ steps.set-matrix.outputs.lint }} test: ${{ steps.set-matrix.outputs.test }} extended-tests: ${{ steps.set-matrix.outputs.extended-tests }} compile-integration-tests: ${{ steps.set-matrix.outputs.compile-integration-tests }} dependencies: ${{ steps.set-matrix.outputs.dependencies }} test-doc-imports: ${{ steps.set-matrix.outputs.test-doc-imports }} test-pydantic: ${{ steps.set-matrix.outputs.test-pydantic }} # Run linting only on packages that have changed files lint: needs: [ build ] if: ${{ needs.build.outputs.lint != '[]' }} strategy: matrix: job-configs: ${{ fromJson(needs.build.outputs.lint) }} fail-fast: false uses: ./.github/workflows/_lint.yml with: working-directory: ${{ matrix.job-configs.working-directory }} python-version: ${{ matrix.job-configs.python-version }} secrets: inherit # Run unit tests only on packages that have changed files test: needs: [ build ] if: ${{ needs.build.outputs.test != '[]' }} strategy: matrix: job-configs: ${{ fromJson(needs.build.outputs.test) }} fail-fast: false uses: ./.github/workflows/_test.yml with: working-directory: ${{ matrix.job-configs.working-directory }} python-version: ${{ matrix.job-configs.python-version }} secrets: inherit # Test compatibility with different Pydantic versions for affected packages test-pydantic: needs: [ build ] if: ${{ needs.build.outputs.test-pydantic != '[]' }} strategy: matrix: job-configs: ${{ fromJson(needs.build.outputs.test-pydantic) }} fail-fast: false uses: ./.github/workflows/_test_pydantic.yml with: working-directory: ${{ matrix.job-configs.working-directory }} pydantic-version: ${{ matrix.job-configs.pydantic-version }} secrets: inherit test-doc-imports: needs: [ build ] if: ${{ needs.build.outputs.test-doc-imports != '[]' }} strategy: matrix: job-configs: ${{ fromJson(needs.build.outputs.test-doc-imports) }} fail-fast: false uses: ./.github/workflows/_test_doc_imports.yml with: python-version: ${{ matrix.job-configs.python-version }} secrets: inherit # Verify integration tests compile without actually running them (faster feedback) compile-integration-tests: needs: [ build ] if: ${{ needs.build.outputs.compile-integration-tests != '[]' }} strategy: matrix: job-configs: ${{ fromJson(needs.build.outputs.compile-integration-tests) }} fail-fast: false uses: ./.github/workflows/_compile_integration_test.yml with: working-directory: ${{ matrix.job-configs.working-directory }} python-version: ${{ matrix.job-configs.python-version }} secrets: inherit # Run extended test suites that require additional dependencies extended-tests: name: 'Extended Tests' needs: [ build ] if: ${{ needs.build.outputs.extended-tests != '[]' }} strategy: matrix: # note different variable for extended test dirs job-configs: ${{ fromJson(needs.build.outputs.extended-tests) }} fail-fast: false runs-on: ubuntu-latest timeout-minutes: 20 defaults: run: working-directory: ${{ matrix.job-configs.working-directory }} steps: - uses: actions/checkout@v4 - name: '๐Ÿ Set up Python ${{ matrix.job-configs.python-version }} + UV' uses: "./.github/actions/uv_setup" with: python-version: ${{ matrix.job-configs.python-version }} - name: '๐Ÿ“ฆ Install Dependencies & Run Extended Tests' shell: bash run: | echo "Running extended tests, installing dependencies with uv..." uv venv uv sync --group test VIRTUAL_ENV=.venv uv pip install -r extended_testing_deps.txt VIRTUAL_ENV=.venv make extended_tests - name: '๐Ÿงน Verify Clean Working Directory' shell: bash run: | set -eu STATUS="$(git status)" echo "$STATUS" # grep will exit non-zero if the target message isn't found, # and `set -e` above will cause the step to fail. echo "$STATUS" | grep 'nothing to commit, working tree clean' # Final status check - ensures all required jobs passed before allowing merge ci_success: name: 'โœ… CI Success' needs: [build, lint, test, compile-integration-tests, extended-tests, test-doc-imports, test-pydantic] if: | always() runs-on: ubuntu-latest env: JOBS_JSON: ${{ toJSON(needs) }} RESULTS_JSON: ${{ toJSON(needs.*.result) }} EXIT_CODE: ${{!contains(needs.*.result, 'failure') && !contains(needs.*.result, 'cancelled') && '0' || '1'}} steps: - name: '๐ŸŽ‰ All Checks Passed' run: | echo $JOBS_JSON echo $RESULTS_JSON echo "Exiting with $EXIT_CODE" exit $EXIT_CODE