Static checks distilled from real upstream bugs that off-the-shelf linters miss.
Project description
wildlint
Static checks distilled from real upstream bugs — the kind off-the-shelf linters miss because they look like ordinary, working code.
Every rule here was born from a concrete bug that was found and fixed in a public project, then generalized to the smallest static check that still catches the class without flooding you with false positives. If a bug could not be turned into a low-noise rule, it is documented as not-shipped rather than added as noise (see Not shipped).
Install
pip install wildlint
Use
wildlint path/to/code # scan a file or directory (default: .)
wildlint --select WL001,WL002 src/
wildlint --pedantic src/ # also run opt-in, higher-false-positive rules
Exits non-zero when anything is found, so it drops straight into CI or a pre-commit hook.
pre-commit
# .pre-commit-config.yaml
repos:
- repo: https://github.com/patchwright/wildlint
rev: v0.2.0
hooks:
- id: wildlint
CI (GitHub Actions)
- run: pip install wildlint
- run: wildlint src/
Rules
| Code | Tier | Catches | Distilled from |
|---|---|---|---|
| WL001 | default | x.replace(P, "") guarded by x.startswith(P)/endswith(P) — removes every occurrence, silently corrupting values that contain the marker twice. Meant str.removeprefix/removesuffix. |
nephila/giturlparse#149 |
| WL002 | pedantic | s.split(' ') where s.split() was meant — keeps empty tokens and skips whitespace collapsing/trimming, leaking blanks downstream. Advisory and opt-in: only an exact single-space literal fires, and it's frequently intentional. |
derek73/python-nameparser#164 |
| WL003 | pedantic | x[-k] with k >= 2 — IndexError when the sequence is shorter than k. Opt-in because deep negative indexing is often provably safe from context the checker can't see. |
savoirfairelinux/num2words#661 |
The default tier is WL001 only — it has effectively zero false positives. WL002
and WL003 are opt-in via --pedantic: real bug classes, but they also fire on
legitimate code, so the default stays strictly precision.
Each rule is verified against the actual pre-fix source of the project it came
from — see the tests, and the rule docstrings in src/wildlint/checkers.py.
Property-test templates
Some bug classes have no stable AST signature — the same wrong behaviour is
reached by different code each time, so any static rule broad enough to catch
them all also flags mountains of correct code. The archetype is the
rounding-rollover bug in number / byte / SI-prefix humanizers
(boltons#403,
millify#13,
numerize#17,
si-prefix#17): four distinct
implementations of one invariant break (<=-vs-<, a missing carry after
rounding, rounding an unrounded boundary). millify(999999) returns '1000k'
instead of '1M'.
What they share is a falsifiable property: a humanizer must never emit a
mantissa >= base while a larger unit is still available. wildlint ships that
check two ways.
Run it directly (dependency-free, in your own test suite or CI):
from wildlint.property_templates import find_rollover
from millify import millify
def test_no_rounding_rollover():
violations = find_rollover(millify, base=1000) # 1000=SI, 1024=bytes
assert not violations, "\n".join(str(v) for v in violations)
find_rollover sweeps the dangerous boundary inputs (values that round up
across a unit boundary) and returns the concrete violations. Pass units=[...]
(small→large) for an exact check that won't flag legitimate overflow at the
largest unit.
Or render a paste-ready template:
wildlint --template rollover --func millify --import-from millify --base 1000
| Code | Catches | Distilled from |
|---|---|---|
| WP001 | A humanizer emits a mantissa >= base while a larger unit is available ('1000k' instead of '1M') because the unit is chosen before the mantissa is rounded. |
boltons#403, millify#13, numerize#17, si-prefix#17 |
Bugs considered but not shipped
Some real bugs do not generalize into a low-false-positive static rule. They are
recorded in NON_GENERALIZED in checkers.py so the reasoning is preserved:
- break-vs-continue (mnamer#371) — whether
breakshould becontinueis entirely loop-intent dependent. - sign-doubling (humanize#326) — a numeric-formatting concern, not a syntactic pattern.
- validation-branch-order (validators#463) — specific to one parser's control flow.
- radix-from-ignored-param (shortuuid#115) — requires matching a docstring contract to the implementation.
- rng-from-unordered-set — iterating a set into a
randompopulation (directly, or vialist(some_set)feedingrandom.choicesweights) is non-deterministic across processes:PYTHONHASHSEEDvaries per worker, so set iteration order — and item↔weight alignment — changes run to run. The bare form (random.choice({1,2,3})) is rare; the real class (set→list→positional use) is only visible cross-process and is best caught by a reproducibility property test (run twice under differingPYTHONHASHSEED, assert identical output), not a static rule.
Adding a rule
A checker is any object with code, name, tier, and
check(tree, path) -> list[Finding]. Append an instance to CHECKERS in
checkers.py and add positive/negative tests mirroring the wild bug. That's the
whole extension surface — the suite grows one real bug at a time.
License
MIT.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file wildlint-0.2.0.tar.gz.
File metadata
- Download URL: wildlint-0.2.0.tar.gz
- Upload date:
- Size: 18.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
84f4ac8a826417f1f0648442bb5fc511f22c42c822efb9d2b7a142581ad26df2
|
|
| MD5 |
eab1a01f44d23e5a829bfbab07ecebd4
|
|
| BLAKE2b-256 |
d76f70837ccb24df73c7ffec3fde06f41a7b4248ccf4f4013c9780428dea4bb2
|
Provenance
The following attestation bundles were made for wildlint-0.2.0.tar.gz:
Publisher:
release.yml on patchwright/wildlint
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
wildlint-0.2.0.tar.gz -
Subject digest:
84f4ac8a826417f1f0648442bb5fc511f22c42c822efb9d2b7a142581ad26df2 - Sigstore transparency entry: 1850170709
- Sigstore integration time:
-
Permalink:
patchwright/wildlint@c7607e0c6010f11eb1781e20241d2edccffe0505 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/patchwright
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@c7607e0c6010f11eb1781e20241d2edccffe0505 -
Trigger Event:
push
-
Statement type:
File details
Details for the file wildlint-0.2.0-py3-none-any.whl.
File metadata
- Download URL: wildlint-0.2.0-py3-none-any.whl
- Upload date:
- Size: 15.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
690c9fd079ca64e9f02c32b11f797ab78322436876374f7acea7ebfcf10cd50d
|
|
| MD5 |
a8eac70e8a8ee50b84f3f3abc83ed544
|
|
| BLAKE2b-256 |
ed0193d71fdf7f2957cf5c5c44e70000c10fedc57be396beab5fb9df3f638167
|
Provenance
The following attestation bundles were made for wildlint-0.2.0-py3-none-any.whl:
Publisher:
release.yml on patchwright/wildlint
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
wildlint-0.2.0-py3-none-any.whl -
Subject digest:
690c9fd079ca64e9f02c32b11f797ab78322436876374f7acea7ebfcf10cd50d - Sigstore transparency entry: 1850170808
- Sigstore integration time:
-
Permalink:
patchwright/wildlint@c7607e0c6010f11eb1781e20241d2edccffe0505 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/patchwright
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@c7607e0c6010f11eb1781e20241d2edccffe0505 -
Trigger Event:
push
-
Statement type: