name: CI # tests.yml workflow will run for all changes except: # any file or directory under web/ or doc/ # any Markdown (.md) file anywhere in the repository on: push: branches: [ master ] paths-ignore: - 'web/**' - 'doc/**' - '**.md' pull_request: branches: [ master ] paths-ignore: - 'web/**' - 'doc/**' - '**.md' permissions: read-all # save workspaces to speed up testing env: CAPA_SAVE_WORKSPACE: "True" jobs: changelog_format: runs-on: ubuntu-22.04 steps: - name: Checkout capa uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 # The sync GH action in capa-rules relies on a single '- *$' in the CHANGELOG file - name: Ensure CHANGELOG has '- *$' run: | number=$(grep '\- *$' CHANGELOG.md | wc -l) if [ $number != 1 ]; then exit 1; fi code_style: runs-on: ubuntu-22.04 steps: - name: Checkout capa uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 # use latest available python to take advantage of best performance - name: Set up Python 3.13 uses: actions/setup-python@0a5c61591373683505ea898e09a3ea4f39ef2b9c # v5.0.0 with: python-version: "3.13" - name: Install dependencies run: | pip install -r requirements.txt pip install -e .[dev,scripts] - name: Lint with ruff run: pre-commit run ruff - name: Lint with isort run: pre-commit run isort --show-diff-on-failure - name: Lint with black run: pre-commit run black --show-diff-on-failure - name: Lint with flake8 run: pre-commit run flake8 --hook-stage manual - name: Check types with mypy run: pre-commit run mypy --hook-stage manual - name: Check imports against dependencies run: pre-commit run deptry --hook-stage manual rule_linter: runs-on: ubuntu-22.04 steps: - name: Checkout capa with submodules uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 with: submodules: recursive - name: Set up Python 3.13 uses: actions/setup-python@0a5c61591373683505ea898e09a3ea4f39ef2b9c # v5.0.0 with: python-version: "3.13" - name: Install capa run: | pip install -r requirements.txt pip install -e .[dev,scripts] - name: Run rule linter run: python scripts/lint.py rules/ tests: name: Tests in ${{ matrix.python-version }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} needs: [code_style, rule_linter] strategy: fail-fast: false matrix: os: [ubuntu-22.04, ubuntu-22.04-arm, windows-2022, macos-15-intel, macos-14] # across all operating systems python-version: ["3.10", "3.13"] include: # on Ubuntu run these as well - os: ubuntu-22.04 python-version: "3.11" - os: ubuntu-22.04 python-version: "3.12" steps: - name: Checkout capa with submodules uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 with: submodules: recursive - name: Set up Python ${{ matrix.python-version }} uses: actions/setup-python@0a5c61591373683505ea898e09a3ea4f39ef2b9c # v5.0.0 with: python-version: ${{ matrix.python-version }} - name: Install pyyaml if: matrix.os == 'ubuntu-22.04' run: sudo apt-get install -y libyaml-dev - name: Install capa run: | pip install -r requirements.txt pip install -e .[dev,scripts] - name: Run tests (fast) # this set of tests runs about 80% of the cases in 20% of the time, # and should catch most errors quickly. run: pre-commit run pytest-fast --all-files --hook-stage manual - name: Run tests run: pytest -v tests/ binja-tests: name: Binary Ninja tests for ${{ matrix.python-version }} env: BN_SERIAL: ${{ secrets.BN_SERIAL }} runs-on: ubuntu-22.04 needs: [tests] strategy: fail-fast: false matrix: python-version: ["3.10", "3.13"] steps: - name: Checkout capa with submodules # do only run if BN_SERIAL is available, have to do this in every step, see https://github.com/orgs/community/discussions/26726#discussioncomment-3253118 if: ${{ env.BN_SERIAL != 0 }} uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 with: submodules: recursive - name: Set up Python ${{ matrix.python-version }} if: ${{ env.BN_SERIAL != 0 }} uses: actions/setup-python@0a5c61591373683505ea898e09a3ea4f39ef2b9c # v5.0.0 with: python-version: ${{ matrix.python-version }} - name: Install pyyaml if: ${{ env.BN_SERIAL != 0 }} run: sudo apt-get install -y libyaml-dev - name: Install capa if: ${{ env.BN_SERIAL != 0 }} run: | pip install -r requirements.txt pip install -e .[dev,scripts] - name: install Binary Ninja if: ${{ env.BN_SERIAL != 0 }} run: | mkdir ./.github/binja curl "https://raw.githubusercontent.com/Vector35/binaryninja-api/6812c97/scripts/download_headless.py" -o ./.github/binja/download_headless.py python ./.github/binja/download_headless.py --serial ${{ env.BN_SERIAL }} --output .github/binja/BinaryNinja-headless.zip unzip .github/binja/BinaryNinja-headless.zip -d .github/binja/ python .github/binja/binaryninja/scripts/install_api.py --install-on-root --silent - name: Run tests if: ${{ env.BN_SERIAL != 0 }} env: BN_LICENSE: ${{ secrets.BN_LICENSE }} run: pytest -v tests/test_binja_features.py # explicitly refer to the binja tests for performance. other tests run above. ghidra-tests: name: Ghidra tests for ${{ matrix.python-version }} runs-on: ubuntu-22.04 needs: [tests] strategy: fail-fast: false matrix: python-version: ["3.10", "3.13"] java-version: ["21"] ghidra-version: ["12.0"] public-version: ["PUBLIC_20251205"] # for ghidra releases ghidra-sha256: ['af43e8cfb2fa4490cf6020c3a2bde25c159d83f45236a0542688a024e8fc1941'] steps: - name: Checkout capa with submodules uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 with: submodules: true - name: Set up Python ${{ matrix.python-version }} uses: actions/setup-python@0a5c61591373683505ea898e09a3ea4f39ef2b9c # v5.0.0 with: python-version: ${{ matrix.python-version }} - name: Set up Java ${{ matrix.java-version }} uses: actions/setup-java@387ac29b308b003ca37ba93a6cab5eb57c8f5f93 # v4.0.0 with: distribution: 'temurin' java-version: ${{ matrix.java-version }} - name: Install Ghidra ${{ matrix.ghidra-version }} run: | mkdir ./.github/ghidra wget "https://github.com/NationalSecurityAgency/ghidra/releases/download/Ghidra_${{ matrix.ghidra-version }}_build/ghidra_${{ matrix.ghidra-version }}_${{ matrix.public-version }}.zip" -O ./.github/ghidra/ghidra_${{ matrix.ghidra-version }}_PUBLIC.zip echo "${{ matrix.ghidra-sha256 }} ./.github/ghidra/ghidra_${{ matrix.ghidra-version }}_PUBLIC.zip" | sha256sum -c - unzip .github/ghidra/ghidra_${{ matrix.ghidra-version }}_PUBLIC.zip -d .github/ghidra/ - name: Install pyyaml run: sudo apt-get install -y libyaml-dev - name: Install capa with Ghidra extra run: | pip install -e .[dev,ghidra] - name: Run tests env: GHIDRA_INSTALL_DIR: ${{ github.workspace }}/.github/ghidra/ghidra_${{ matrix.ghidra-version }}_PUBLIC run: pytest -v tests/test_ghidra_features.py