Pytest plugin to lint Jinja2 templates in FastAPI applications
Project description
pytest-jinja-check
A pytest plugin that lints Jinja2 templates used in FastAPI applications. Catches common issues at test time:
- Syntax validation — templates parse without errors
- Hardcoded route detection —
href="/foo"should beurl_for('foo') - Endpoint validation —
url_for('endpoint')references actually exist in your app - Context variable checking — routes pass all variables their templates need
Installation
pip install pytest-jinja-check
Or with uv:
uv add pytest-jinja-check
For endpoint validation against a live FastAPI app, install with the fastapi extra:
pip install "pytest-jinja-check[fastapi]"
# or
uv add "pytest-jinja-check[fastapi]"
Quick start
The plugin auto-registers with pytest via entry points. Add configuration to your pyproject.toml:
[tool.pytest-jinja-check]
template_dir = "templates"
python_dir = "app"
Then write tests using the provided fixtures:
# tests/test_templates.py
def test_template_syntax(template_syntax_errors):
assert not template_syntax_errors, "\n".join(str(e) for e in template_syntax_errors)
def test_no_hardcoded_routes(hardcoded_routes):
assert not hardcoded_routes, "\n".join(str(e) for e in hardcoded_routes)
def test_context_variables(missing_context_variables):
assert not missing_context_variables, "\n".join(
str(e) for e in missing_context_variables
)
def test_url_for_endpoints(validate_endpoints):
from myapp.main import app
errors = validate_endpoints(app)
assert not errors, "\n".join(str(e) for e in errors)
Run your tests as usual:
pytest
Fixtures
All analysis fixtures are session-scoped (run once per test session).
| Fixture | Returns | Description |
|---|---|---|
template_linter_config |
LinterConfig |
Resolved configuration |
template_variables |
dict[str, TemplateInfo] |
Template name -> extracted variables (with inheritance) |
route_contexts |
list[RouteContext] |
All TemplateResponse calls found via AST |
template_syntax_errors |
list[LintError] |
Templates that fail to parse |
hardcoded_routes |
list[LintError] |
Hardcoded URLs that should use url_for() |
missing_context_variables |
list[LintError] |
Routes missing required template variables |
validate_endpoints |
callable(app) -> list[LintError] |
Factory: validates url_for() refs against a live app |
Configuration
All settings in [tool.pytest-jinja-check] in your pyproject.toml:
| Key | Default | Description |
|---|---|---|
template_dir |
"templates" |
Template directory relative to project root |
python_dir |
"." |
Directory to scan for Python route files |
route_file_patterns |
["**/*.py"] |
Glob patterns for route files |
ignore_variables |
["request", "url_for", ...] |
Variables to skip (framework-provided) |
allowed_url_prefixes |
["#", "http://", ...] |
URL prefixes that aren't hardcoded routes |
You can also pass --template-lint-config <path> to pytest to specify a different directory containing pyproject.toml.
Programmatic API
The analysis functions can also be used outside of pytest:
from pytest_jinja_check import (
analyze_all_templates,
check_syntax,
check_hardcoded_routes,
check_context_variables,
validate_url_for_references,
extract_all_route_contexts,
)
from pathlib import Path
# Analyze templates
templates = analyze_all_templates(Path("templates"))
for name, info in templates.items():
print(f"{name}: needs {info.variables}")
# Find issues
syntax_errors = check_syntax(Path("templates"))
hardcoded = check_hardcoded_routes(Path("templates"))
routes = extract_all_route_contexts(Path("app"))
missing = check_context_variables(Path("templates"), routes)
How it works
- Template analysis uses
jinja2.meta.find_undeclared_variables()on parsed ASTs, recursively resolving{% extends %}to capture inherited variable requirements. - Route analysis uses Python's
astmodule to findTemplateResponse(...)calls and extract template names and context dict keys. Handles both the old positional API and newer keyword API. - Endpoint validation introspects a live FastAPI
app.routesto get registered endpoint names.
Known limitations
- Template variables from dynamic
{% extends variable %}can't be resolved. - Context dicts built outside the
TemplateResponse(...)call (e.g.,ctx = {...}; TemplateResponse("t.html", ctx)) are flagged as dynamic and skipped rather than producing false positives. - The variable union from template inheritance may over-report when a child block completely replaces a parent block that used a variable.
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 pytest_jinja_check-1.0.2.tar.gz.
File metadata
- Download URL: pytest_jinja_check-1.0.2.tar.gz
- Upload date:
- Size: 48.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c5f218b2cfe5c00c6edcf2c3929cc2f5d8e8438d96167efdc6b905f4e27751d0
|
|
| MD5 |
461b726fd460b8e7a0492ff4768728ff
|
|
| BLAKE2b-256 |
95f8682cfe9b28670e558557a6351f0f758cf094bbfe0eedbe43a424d5bcd527
|
Provenance
The following attestation bundles were made for pytest_jinja_check-1.0.2.tar.gz:
Publisher:
release.yml on Promptly-Technologies-LLC/pytest-jinja-check
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pytest_jinja_check-1.0.2.tar.gz -
Subject digest:
c5f218b2cfe5c00c6edcf2c3929cc2f5d8e8438d96167efdc6b905f4e27751d0 - Sigstore transparency entry: 1100210277
- Sigstore integration time:
-
Permalink:
Promptly-Technologies-LLC/pytest-jinja-check@e8418ad337f72ee31f8d34362ed553ebaad048a9 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/Promptly-Technologies-LLC
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@e8418ad337f72ee31f8d34362ed553ebaad048a9 -
Trigger Event:
push
-
Statement type:
File details
Details for the file pytest_jinja_check-1.0.2-py3-none-any.whl.
File metadata
- Download URL: pytest_jinja_check-1.0.2-py3-none-any.whl
- Upload date:
- Size: 14.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3ee88c7ecd9a4595cc7c79211d1423c7d40d145ff860abb096cf99da25e60efe
|
|
| MD5 |
89d8ccb506d856e174e84dcedc206560
|
|
| BLAKE2b-256 |
3ed38cf5a2d7027675ecf6fc8fe240550bef5033680751e42cf330a3e333fcf9
|
Provenance
The following attestation bundles were made for pytest_jinja_check-1.0.2-py3-none-any.whl:
Publisher:
release.yml on Promptly-Technologies-LLC/pytest-jinja-check
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pytest_jinja_check-1.0.2-py3-none-any.whl -
Subject digest:
3ee88c7ecd9a4595cc7c79211d1423c7d40d145ff860abb096cf99da25e60efe - Sigstore transparency entry: 1100210333
- Sigstore integration time:
-
Permalink:
Promptly-Technologies-LLC/pytest-jinja-check@e8418ad337f72ee31f8d34362ed553ebaad048a9 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/Promptly-Technologies-LLC
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@e8418ad337f72ee31f8d34362ed553ebaad048a9 -
Trigger Event:
push
-
Statement type: