Static analysis for UK financial regulatory citations in Python code.
Project description
watchfire
[!IMPORTANT] This package is still in development and is not production ready.
Static analysis for UK financial regulatory citations in Python code.
watchfire lets you annotate Python functions with citations to UK
financial regulation — CRR, the PRA Rulebook, PRA Policy Statements,
PRA Supervisory Statements — and then check those citations against a
bundled, versioned snapshot of the rulebook. The intent is to make the
mapping from compliance code to the regulation it implements executable:
runs in CI, lives next to the code, and breaks the build when it drifts.
Why this exists
Regulatory engineering teams in UK banks need an auditable trail from
every formula in their RWA / capital code back to a specific article of
the CRR or rule in the PRA Rulebook. Today that trail is produced by
hand in Word documents that drift the moment anyone changes the code.
watchfire puts the mapping where the code is, checks it on every commit,
and gives reviewers, auditors, and the PRA something verifiable to look
at instead of a stale spreadsheet.
The first real user is
OpenAfterHours/rwa_calculator
— a UK CRR / Basel 3.1 credit-risk RWA library whose formulas need to
trace back to specific articles. If you're building something similar,
watchfire is for you.
Install
uv add watchfire
# or
pip install watchfire
Python 3.11+.
Quickstart
Decorate a function with the regulation it implements:
from watchfire import cites
@cites("CRR Art. 153(1)(a)")
def corporate_rw(pd: float, lgd: float, maturity: float) -> float:
"""Risk-weight under the IRB approach for corporates."""
...
Add a [tool.watchfire] table to your pyproject.toml:
[tool.watchfire]
rulebook_version = "2024-07-09"
instruments = ["CRR", "PRA_RULEBOOK", "PS", "SS"]
source_paths = ["src"]
Run the checker:
$ uv run watchfire check
watchfire: checked 47 citation(s); no issues found.
If a citation fails to parse or points at something the bundled index
doesn't know about, watchfire check exits non-zero and prints a line
per finding with file, line number, and reason — suitable for CI:
src/myproj/sa.py:31: sovereign_rw: unknown_article: citation 'CRR Art. 999' points to CRR Article 999, which is not in the bundled rulebook index
watchfire: 1 failing finding(s), 0 unresolved, out of 12 resolved citation(s).
@cites is a no-op at runtime — it attaches the parsed citation to the
function as __watchfire__ and returns the function unchanged. No
wrapping, no overhead, nothing to debug.
Traceability matrix
watchfire matrix is the reverse lookup: given a project full of
decorated functions, group every citation by its article and list the
functions that cite it. The output is intended as an audit deliverable —
attach it to a PR or commit it as a CI artifact.
$ uv run watchfire matrix
CRR Art. 4(1)(75) Definitions: corporate 1 site
src/myproj/irb.py:21 is_corporate
CRR Art. 113 SA risk weights 1 site
src/myproj/sa.py:6 calculate_sa_rwa
CRR Art. 153 IRB risk weights 1 site
src/myproj/irb.py:7 corporate_rw CRR Art. 153(1)(a)
SS1/23, paragraph 2.5 (not in index) 1 site
src/myproj/irb.py:17 model_validation
watchfire matrix: 4 entries, 4 citation sites across 4 functions.
Useful flags:
--format {text,markdown,json}—markdownproduces a copy-paste table for PR comments;jsonis for downstream tooling.--specificity {article,full}— defaultarticlecollapses sub-paragraph detail into one row per article. Usefullfor the audit-grade view that keeps(1)(a)separate from(1)(b).--instrument CRR --article 153— narrow to one article. Answers "which functions cite Art. 153?".
watchfire matrix exits 0 unconditionally; it's informational. Parse
failures and unresolved citations are counted in the footer — fix them
with watchfire check.
Citation grammar
The parser accepts canonical UK regulatory citation strings. The shape
that comes out the other side is a frozen Citation dataclass; see
watchfire.Citation for the field list.
| Input | Meaning |
|---|---|
CRR Art. 153 |
Whole article |
CRR Article 153 |
(alternate spelling) |
CRR Art. 153(1) |
Paragraph |
CRR Art. 153(1)(a) |
Point |
CRR Art. 153(1)(a)(ii) |
Sub-point |
CRR Art. 4(1)(75) |
Numeric point (CRR definitions) |
PRA Rulebook, Credit Risk, 3.2 |
Rulebook section |
PS9/24 |
PRA Policy Statement, whole document |
SS1/23, paragraph 2.5 |
Supervisory Statement with paragraph reference |
Delegated Regulation 2018/171 Art. 3 |
UK on-shored EU Delegated Regulation |
The keyword for an article accepts Art, Art., Article, or article
in any case. Whitespace is normalised. Anything that doesn't parse is a
CitationParseError, which watchfire check reports with the offending
input — these are code-review events, not silent skips.
Configuration reference
[tool.watchfire]
# Snapshot of the rulebook to pin to. Decorators that omit `version=`
# inherit this pin. ISO-8601 date.
rulebook_version = "2024-07-09"
# Citation instruments allowed in this project. A citation whose
# instrument is not in this list is reported by `watchfire check`.
instruments = ["CRR", "PRA_RULEBOOK", "PS", "SS", "DELEGATED_REG"]
# Directories to walk when running `watchfire check` with no arguments.
source_paths = ["src"]
Public API
from watchfire import (
Citation, # frozen dataclass: instrument, article, paragraph, ...
parse_citation, # str -> Citation, raises CitationParseError
CitationParseError,
cites, # the @cites decorator
)
Everything else (watchfire.ast_walker, watchfire.index, watchfire.checks,
watchfire.cli) is internal and may change between releases.
Roadmap
watchfire v0.1 is intentionally a narrow vertical slice: get the
citation grammar right against real usage in rwa_calculator, ship the
decorator and CLI, then expand.
| Version | Adds |
|---|---|
| v0.1 | Citation grammar, @cites, watchfire check, bundled CRR index |
| v0.2 | watchfire matrix (traceability matrix), watchfire stale (rulebook diff) |
| v0.3+ | Automated scraping of legislation.gov.uk + the PRA Rulebook |
If you have feedback on the citation grammar specifically, please open an issue — the grammar is the foundation, and getting it wrong now is much cheaper to fix than getting it wrong later.
Licence
Apache 2.0. See LICENSE.
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 watchfire-0.2.0.tar.gz.
File metadata
- Download URL: watchfire-0.2.0.tar.gz
- Upload date:
- Size: 441.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.21 {"installer":{"name":"uv","version":"0.9.21","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0309d9f7347e366531a2b61470b7e5fa3f99f1ac9c04a3813ec0d74f19417ffa
|
|
| MD5 |
50b36e1cd50429307b1f928c9544ad36
|
|
| BLAKE2b-256 |
ebadb52cd9c2cae200b1070e5dde205926d1a9c99c948fcfe3c3f04799315335
|
File details
Details for the file watchfire-0.2.0-py3-none-any.whl.
File metadata
- Download URL: watchfire-0.2.0-py3-none-any.whl
- Upload date:
- Size: 327.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.21 {"installer":{"name":"uv","version":"0.9.21","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
12a32fd3ac801f526b3f4d3f84a424678c17c8139c12096da5ad299bb1d4836d
|
|
| MD5 |
d73afc532737cb03e957883dd022daf8
|
|
| BLAKE2b-256 |
21a28102d1510b96c491ef082d729bfbda4d8c0615e6412e6cd67f7e48836cdc
|