Skip to main content

pytest test-smell linter (Rust, via maturin)

Project description

zorilla

A small, fast, opinionated Rust CLI that detects syntactic test smells in pytest codebases. Built to compose with biston (structural test duplication) and with ruff's PT rules — zorilla owns only the gaps those two leave behind.

Status: v0.1 ready. Ships seven rules (ZR001–ZR007), inline and file-level suppression comments, text / json / sarif output formats, a list-rules / explain pair, and a pre-commit hook.

Installation

zorilla is distributed as a Python wheel built with maturin.

# once published
pip install zorilla
# or, for a local checkout (requires an activated venv)
maturin develop

This installs a zorilla binary on your PATH.

maturin develop needs an activated Python virtualenv — either source .venv/bin/activate first, or export VIRTUAL_ENV=/path/to/venv. Having the venv's bin/ on PATH is not sufficient.

Usage

zorilla check path/to/tests

Exit codes: 0 clean, 1 findings reported, 2 error.

Rules

Code Name Summary
ZR001 conditional-test-logic if / for / while / try in a test body
ZR002 sleep-in-test time.sleep / asyncio.sleep inside a test
ZR003 no-assertion Test function with no assertion or pytest.raises
ZR004 assertion-roulette Too many bare (message-less) asserts in one test
ZR005 mystery-guest Absolute path, URL, or ~-path literal inside a test
ZR006 patch-stack Too many stacked @patch / @mock.patch decorators
ZR007 empty-test Test body is empty (pass, ..., docstring-only)

Long-form docs (motivation, positive/negative examples, config knobs, suppression syntax) live under docs/rules/. You can also print them inline with zorilla explain ZR### — see below.

Worked example

Save as tests/test_demo.py:

import time

def test_branch_and_wait():
    if ready():
        time.sleep(1)
        assert done()

Run:

$ zorilla check .
tests/test_demo.py:4:5: ZR001 conditional-test-logic: test function has conditional logic (if/for/while/try)
tests/test_demo.py:5:9: ZR002 sleep-in-test: test calls sleep — wait on a condition instead
2 findings in 1 files discovered.

zorilla exits with status 1 because findings were reported. A clean run (or an empty directory) exits 0; an internal error exits 2.

Output formats

zorilla check --format json .

emits a JSON array, one object per finding — suitable for piping into jq or a CI aggregator:

[
  {
    "code": "ZR001",
    "message": "test function has conditional logic (if/for/while/try)",
    "file": "tests/test_demo.py",
    "line": 4,
    "column": 5,
    "severity": "warning"
  }
]
zorilla check --format sarif .

emits a SARIF 2.1.0 document that most code-scanning tools (GitHub code scanning, SonarQube, etc.) ingest directly:

{
  "$schema": "https://raw.githubusercontent.com/oasis-tcs/sarif-spec/master/Schemata/sarif-schema-2.1.0.json",
  "version": "2.1.0",
  "runs": [
    {
      "tool": { "driver": { "name": "zorilla", "version": "0.1.0" } },
      "results": [
        {
          "ruleId": "ZR001",
          "level": "warning",
          "message": { "text": "test function has conditional logic (if/for/while/try)" },
          "locations": [ /* ...physicalLocation... */ ]
        }
      ]
    }
  ]
}

JSON and SARIF output both omit the trailing human-readable summary line so stdout parses cleanly.

List and explain

$ zorilla list-rules
CODE  NAME                      DEFAULT
ZR001 conditional-test-logic    on
ZR002 sleep-in-test             on
ZR003 no-assertion              on
ZR004 assertion-roulette        on
ZR005 mystery-guest             on
ZR006 patch-stack               on
ZR007 empty-test                on
$ zorilla explain ZR003
# ZR003 — no-assertion
…

explain accepts the rule code in either case (ZR003 or zr003) and prints the bundled markdown. Unknown codes exit 2.

Pre-commit integration

Add zorilla to your project's .pre-commit-config.yaml:

repos:
  - repo: https://github.com/mojzis/zorilla
    rev: v0.1.0
    hooks:
      - id: zorilla

The hook entry point is zorilla check; pre-commit appends only the staged Python files as positional arguments, so zorilla lints each one directly (bypassing the include globs the way any explicit file path does). v0.1.0 is the target release tag — replace it with whichever tag is current when you wire the hook up.

Configuration

zorilla searches upward from the working directory for either zorilla.toml or a pyproject.toml containing [tool.zorilla]. The first match wins.

# pyproject.toml
[tool.zorilla]
include = ["tests/**/*.py", "**/test_*.py", "**/*_test.py", "**/conftest.py"]
exclude = ["**/fixtures/**"]

[tool.zorilla.rules.ZR004]
max_asserts = 4

[tool.zorilla.rules.ZR006]
max_patches = 3

Per-rule sections ([tool.zorilla.rules.ZRNNN]) accept enabled = false to disable the rule and any rule-specific knobs (max_asserts for ZR004, max_patches for ZR006, extra_helpers for ZR003, allowed_prefixes for ZR005). See docs/rules/ for the exhaustive list.

Suppression comments work per-line and per-file:

# zorilla: ignore-file                              <- silences the whole file
def test_x():
    if cond:  # zorilla: ignore[ZR001]              <- silences just this line
        ...

Developing

# Pre-commit gate
cargo fmt --all --check
cargo clippy --workspace --all-targets --all-features -- -D warnings
cargo test --workspace

# Maturin develop build
maturin develop

Workspace layout:

crates/
  zorilla-core/   # library
  zorilla-cli/    # binary (`zorilla`)

See CLAUDE.md for the development workflow and docs/plans/PLAN.md for the design doc driving the rule set.

License

MIT. See LICENSE.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

zorilla-0.1.0-py3-none-win_amd64.whl (1.2 MB view details)

Uploaded Python 3Windows x86-64

zorilla-0.1.0-py3-none-manylinux_2_28_x86_64.whl (1.3 MB view details)

Uploaded Python 3manylinux: glibc 2.28+ x86-64

zorilla-0.1.0-py3-none-macosx_11_0_arm64.whl (1.1 MB view details)

Uploaded Python 3macOS 11.0+ ARM64

File details

Details for the file zorilla-0.1.0-py3-none-win_amd64.whl.

File metadata

  • Download URL: zorilla-0.1.0-py3-none-win_amd64.whl
  • Upload date:
  • Size: 1.2 MB
  • Tags: Python 3, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for zorilla-0.1.0-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 6c75d72edfe1ab6886fc72f927d7f4de259ff57c28ab3fe830653d627d717ce4
MD5 60ac5b8febc7c52f0deea3c321235e4b
BLAKE2b-256 1b8e350e28e99e7ac04a54e7374666cda902d3258dfe3defc5c771466b554170

See more details on using hashes here.

Provenance

The following attestation bundles were made for zorilla-0.1.0-py3-none-win_amd64.whl:

Publisher: release.yml on mojzis/zorilla

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file zorilla-0.1.0-py3-none-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for zorilla-0.1.0-py3-none-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 ea846522676d36b71f698a8da3fc0a1edb1164a02abe463a274cf7ec13ac9815
MD5 9f8730b1eb6b532b2c22f47a7e91ac9d
BLAKE2b-256 9d6a049a00bc28fc45579cc8968ce19cbead0b12d0c7a85704247d4a00072592

See more details on using hashes here.

Provenance

The following attestation bundles were made for zorilla-0.1.0-py3-none-manylinux_2_28_x86_64.whl:

Publisher: release.yml on mojzis/zorilla

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file zorilla-0.1.0-py3-none-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for zorilla-0.1.0-py3-none-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 230817b7b88fb62de293687aff88582832fc6fef2515a54d732f8077c29ed10c
MD5 c61f07b29878097ad34b21bf7302506c
BLAKE2b-256 608b11db697b3777e2edda90386308b9d12856a8adb6eb4ccabbc88027a2a681

See more details on using hashes here.

Provenance

The following attestation bundles were made for zorilla-0.1.0-py3-none-macosx_11_0_arm64.whl:

Publisher: release.yml on mojzis/zorilla

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page