Skip to main content

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 citations to the function as a tuple[Citation, ...] on __watchfire__ and returns the function unchanged. No wrapping, no overhead, nothing to debug.

Multiple @cites decorators stack when the same rule lives in more than one instrument (for example a CRR article and its corresponding PRA Policy Statement). The outermost decorator is the primary citation and appears first in __watchfire__; watchfire check reports one finding per decorator.

@cites("CRR Art. 163")            # outer / primary
@cites("PS1/26, paragraph 163")  # inner / secondary
def apply_pd_floor(...): ...

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}markdown produces a copy-paste table for PR comments; json is for downstream tooling.
  • --specificity {article,full} — default article collapses sub-paragraph detail into one row per article. Use full for 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)
CRR Art. 92a Inserted article with letter suffix
CRR Art. 153(1a) Inserted paragraph with letter suffix
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
PS1/26, paragraph 123B Alphanumeric paragraph identifier
PS1/26 Art. 111(1)(a) PS-attached rulebook instrument with CRR-style article
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. Letter suffixes are accepted on article numbers, on CRR paragraph numbers, and on dotted segments of PS/SS paragraph identifiers; each segment must still start with digits. PRA Rulebook section paths are digits-only. 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


Download files

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

Source Distribution

watchfire-0.3.0.tar.gz (850.0 kB view details)

Uploaded Source

Built Distribution

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

watchfire-0.3.0-py3-none-any.whl (731.1 kB view details)

Uploaded Python 3

File details

Details for the file watchfire-0.3.0.tar.gz.

File metadata

  • Download URL: watchfire-0.3.0.tar.gz
  • Upload date:
  • Size: 850.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for watchfire-0.3.0.tar.gz
Algorithm Hash digest
SHA256 3f9ed6413e1c8c085f314d5b9f9bc8d588e7665574d326431aaf4677c1e0670f
MD5 b2f296831e45f7565ff0981eb477a1e5
BLAKE2b-256 e0715e920f924e57335829d9b16706c15a5502a428dd79fb2744afd1d51fd80b

See more details on using hashes here.

Provenance

The following attestation bundles were made for watchfire-0.3.0.tar.gz:

Publisher: publish.yml on OpenAfterHours/watchfire

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

File details

Details for the file watchfire-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: watchfire-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 731.1 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for watchfire-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 f5fb14cdbe1b34f756d6ba0212d601c8172249b829a60d66bca0cd8032ef3100
MD5 840023327e5c7079b325a37d858d8b03
BLAKE2b-256 a7d1a7f48825cf5f336855cb09d822f26137944d6104cf2af5a496ef3db08fc6

See more details on using hashes here.

Provenance

The following attestation bundles were made for watchfire-0.3.0-py3-none-any.whl:

Publisher: publish.yml on OpenAfterHours/watchfire

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