A command-line architectural health checker for Django projects.
It scans source code statically and flags structural issues before they become entrenched technical debt:
- Fat models
- God apps
- Circular imports
- Missing service layer boundaries
- Celery tasks without retry
- Direct SQL usage
- N+1 query risks
- N+1 serializer risks
- Migration safety risks
Analyzing: /home/user/myproject
── Fat Models ──────────────────────────────
[CRITICAL] core/models.py → UserProfile (34 methods)
Found 1 fat model(s).
── God Apps ────────────────────────────────
[WARNING] core/ owns 41% of total project code (860 / 2,102 lines)
Found 1 god app(s).
── Circular Imports ────────────────────────
No circular imports found.
── Missing Service Layer ────────────────────
[WARNING] orders/views.py → create_order() makes direct ORM calls
Found 1 missing service layer issue(s).
── Celery Tasks Without Retry ──────────────
[CRITICAL] payments/tasks.py → send_invoice_email() — high-stakes task, no retry configured
Found 1 Celery task(s) without retry.
Django projects often drift toward the same architectural problems:
- Models absorb business logic until they become hard to reason about
- One app quietly turns into the center of the codebase
- View functions talk directly to the ORM and blur boundaries
- Celery tasks lose work because retries were never configured
- Circular imports pile up as module boundaries erode
These issues are easy to normalize and hard to see in code review. django-arch-check makes them visible early, in local development and in CI.
pip install django-arch-checkRequirements:
- Python 3.11+
- No Django runtime setup required
The tool is static-only: it reads source files, parses ASTs, and never imports your project code.
# Analyze a project and print findings to the terminal
django-arch-check analyze /path/to/project
# Generate a self-contained HTML report
django-arch-check analyze --format html /path/to/project
# Emit machine-readable JSON for scripts and dashboards
django-arch-check analyze --format json /path/to/project > results.json
# Emit SARIF for GitHub code scanning, VS Code, and CI dashboards
django-arch-check analyze --format sarif /path/to/project > results.sarif
# Tune thresholds
django-arch-check analyze \
--fat-model-threshold 20 \
--god-app-threshold 40 \
/path/to/projectdjango-arch-check ships a ready-made .pre-commit-hooks.yaml, so teams can add it to an existing pre-commit setup with a single hook entry:
repos:
- repo: https://github.com/RJ-Gamer/django-arch-check
rev: v0.8.0
hooks:
- id: django-arch-checkThe bundled hook runs django-arch-check analyze . from the repository root and disables filename passing, which makes it work correctly for a whole-project architecture scan.
You can still pass your own CLI options from .pre-commit-config.yaml:
repos:
- repo: https://github.com/RJ-Gamer/django-arch-check
rev: v0.8.0
hooks:
- id: django-arch-check
args: [--ignore-path, legacy/]Use --ignore to skip one or more detectors entirely.
django-arch-check analyze --ignore fat_models /path/to/project
django-arch-check analyze --ignore fat_models --ignore god_apps /path/to/projectValid detector names:
fat_modelsgod_appscircular_importsmissing_service_layercelery_tasksdirect_sqln_plus_onemigration_safetyn1_serializer_risk
If an invalid detector name is passed, the CLI exits with a clear error:
Error: Unknown detector 'fat_modelz'. Valid detectors are: fat_models, god_apps, circular_imports, missing_service_layer, celery_tasks, direct_sql, n_plus_one, migration_safety, n1_serializer_risk
In HTML reports, skipped detectors are shown as:
⊘ Skipped (--ignore flag)
Use --ignore-path to skip files whose relative path contains a given substring.
django-arch-check analyze --ignore-path legacy/ /path/to/project
django-arch-check analyze --ignore-path legacy/ --ignore-path archive/ /path/to/projectThis is applied across all detectors. If a file path contains the ignored string, that file is not analyzed.
Examples:
--ignore-path legacy/skips files under paths likelegacy/models.py--ignore-path archive/skips files under paths likeapps/orders/archive/tasks.py
Path ignores are substring-based, not glob-based.
Current django-arch-check analyze --help output:
Usage: main analyze [OPTIONS] PROJECT_PATH
Analyze a Django project at PROJECT_PATH for architectural issues.
Options:
--fat-model-threshold N Flag models with >= N non-dunder methods. [default:
15]
--god-app-threshold PCT Flag apps owning >= PCT% of total project LOC.
[default: 30]
--ignore DETECTOR Ignore a detector by name. Repeatable.
--ignore-path PATH Skip files whose path contains PATH. Repeatable.
--format [text|html|json|sarif]
Output format: text/html/json/sarif. HTML writes
arch-report.html; the others use stdout. [default:
text]
--help Show this message and exit.
| Code | Meaning |
|---|---|
0 |
No critical findings |
1 |
At least one critical finding, or a CLI usage error |
This makes the tool suitable for CI gating.
Flags Django model classes with too many non-dunder methods.
- Warning:
method_count >= threshold - Critical:
method_count >= threshold * 2 - Default threshold:
15 - Default critical boundary:
30
Notes:
- Counts
defandasync def - Ignores dunder methods like
__str__ - Scans Python files across the project
Flags Django apps that own too much of the total project LOC.
- Warning:
percentage >= threshold - Critical:
percentage >= threshold + 20 - Default threshold:
30% - Default critical boundary:
50%
Notes:
- LOC excludes blank lines and standalone comment lines
- Requires at least 2 Django apps before it reports findings
- Uses
models.pyorapps.pyto identify app directories
Flags cycles in the intra-project import graph.
- Critical: any detected cycle
Notes:
- Only top-level imports are analyzed
- Function-level imports are intentionally ignored
- Reports both short and multi-node cycles
Flags views that contain too many direct ORM calls and likely need service-layer extraction.
- Warning: 2 or more direct
Model.objects.*calls in a single view function or method - Critical: 4 or more direct
Model.objects.*calls in a single view function or method
Notes:
- Scans
views.pyfiles only - Supports function-based views and class-based view methods
- Uses ORM call count, not raw line count
Flags Celery tasks that lack retry configuration.
- Critical: task name contains
payment,email,invoice, ornotification, and has no retry config - Warning: any other task with no retry config
Retry config is considered present when the decorator includes any of:
max_retriesautoretry_forretry_backoff
Notes:
- Detects both
@shared_taskand@app.task - Skips migration files
Flags raw SQL patterns that bypass Django's ORM.
Detected patterns:
cursor.execute(connection.cursor().raw(.extra(select=
Severity:
- Warning only
Notes:
- Migration files are excluded
Flags likely N+1 query patterns inside loops and list comprehensions.
- Warning: ORM call inside a loop or list comprehension, with no
select_relatedorprefetch_relatedfound in the same function scope
Notes:
- Scans
views.pyandserializers.py - Looks for
X.objects.method(...)patterns inside loops - Heuristic by design; false positives and false negatives are possible
Flags migration operations that carry deployment or data-safety risk.
- Warning:
RemoveField— field removal is irreversible - Warning:
RenameField— breaks code referencing the old name during rolling deploys - Warning:
AddFieldwith a NOT NULL column and nodefault— fails on non-empty tables - Warning:
RunPythonwithoutatomic = Falseon the Migration class — long-running data migrations hold locks - Warning:
RunSQL— raw SQL bypasses Django's ORM safety layer
Every finding includes an advisory message explaining the risk and a safer alternative approach. To suppress a known-safe finding, add # django-arch-check: ignore on the operation line:
(
migrations.RemoveField( # django-arch-check: ignore
model_name="order",
name="legacy_status",
),
)Notes:
- Only scans files inside
migrations/directories - Skips
__init__.py - Does not block or fail — advises only
Flags DRF serializer and viewset patterns that commonly trigger per-object ORM access.
- Error: ORM call inside a
SerializerMethodFieldgetter - Error: nested serializer field with no matching
prefetch_related(...)orselect_related(...)in the paired viewset - Error: serializer
source=targeting a model@propertythat performs ORM work - Warning: bare
queryset = Model.objects.all()on a viewset paired with a relational serializer
Notes:
- Uses static AST analysis only; it does not import Django or DRF
- Includes code snippets in HTML, JSON, and SARIF output
- Treats detector-level
errorfindings as critical in aggregate report badges and score summaries
django-arch-check analyze --format html /path/to/projectThe generated arch-report.html is self-contained and works offline.
It includes:
- A weighted, size-aware health score from
0to100 - A letter grade from
AtoFwith a plain-language label - Summary counts for critical and warning findings
- A per-detector score breakdown table
- One section per detector
- Accordion-style code snippets for findings that include source context
- Sticky severity filters for critical-only and warning-only views
- Skipped detector notes when
--ignoreis used - Dark and light theme toggle with
localStoragepersistence
When you generate HTML from the CLI, it also prints the score, grade, and label in the terminal before showing the saved report path.
The score is weighted by detector risk and normalized by project size, so a critical circular import hurts more than a warning-level code smell, and the same finding counts more in a 5-file project than in a 500-file one.
Formula:
weighted_score = sum of detector/severity weights
normalized_density = weighted_score / ln(file_count + 1)
density_penalty = min(65, round(normalized_density * 8))
absolute_penalty = min(15, round(weighted_score * 0.08))
score = max(0, 100 - density_penalty - absolute_penalty)
Default detector weights:
circular_importscritical =10celery_taskscritical =8, warning =3migration_safetywarning =6missing_service_layercritical =4, warning =2n_plus_onewarning =3direct_sqlwarning =2god_appscritical =3, warning =1.5fat_modelscritical =2, warning =1n1_serializer_riskerror =3, warning =1.5
Grades:
A(90-100) = ExcellentB(75-89) = GoodC(60-74) = Needs WorkD(40-59) = PoorF(0-39) = Critical
For automation, use machine-readable output modes:
django-arch-check analyze --format json ./ > results.json
django-arch-check analyze --format sarif ./ > results.sarifJSON output is designed for scripting, custom dashboards, and third-party integrations.
It includes:
- Tool metadata and version
- Project path and generation timestamp
- Health score and severity summary
- Per-detector findings, skip state, and normalized messages
- Code snippet payloads for detectors that provide source context
The JSON summary uses the same project-aware health score calculation as the HTML report.
SARIF output follows SARIF 2.1.0, the standard format consumed by:
- GitHub Advanced Security / code scanning
- VS Code's Problems panel
- CI systems and security dashboards that ingest SARIF
Each result includes the detector rule id, severity level, message, and source location when one is available.
Both JSON and SARIF are emitted as ASCII-safe JSON so shell redirection works reliably on Windows cp1252 terminals.
- name: Check Django architecture
run: |
pip install django-arch-check
django-arch-check analyze ./If you want to ignore legacy areas while still gating the rest of the codebase:
- name: Check Django architecture
run: |
pip install django-arch-check
django-arch-check analyze --ignore-path legacy/ ./To generate an HTML report artifact:
- name: Django architecture report
run: |
pip install django-arch-check
django-arch-check analyze --format html ./
- name: Upload report
uses: actions/upload-artifact@v4
with:
name: arch-report
path: arch-report.html- name: Generate SARIF report
run: |
pip install django-arch-check
django-arch-check analyze --format sarif ./ > results.sarif
- name: Upload SARIF
uses: github/codeql-action/upload-sarif@v3
with:
sarif_file: results.sarifdjango-arch-check analyzes source code statically.
It:
- Walks the project tree
- Skips common non-source directories like
.venv,node_modules, and caches - Parses Python files with the standard-library
astmodule - Runs each detector independently through a central analyzer
It does not:
- Import your Django project
- Require configured settings
- Hit the database
- Execute application code
That makes it safe to run in CI, pre-commit hooks, and partially broken repos.
- Circular import detection only covers top-level imports
--ignore-pathuses substring matching, not glob syntax- Missing service layer detection only scans files literally named
views.py - N+1 detection is heuristic and only reasons within a single function scope
- Serializer N+1 detection is heuristic and relies on class-name and field-name matching across serializers, models, and viewsets
- God app analysis requires at least 2 detectable Django apps
git clone https://github.com/RJ-Gamer/django-arch-check.git
cd django-arch-check
python -m venv .venv
source .venv/bin/activate # Windows: .venv\Scripts\activate
pip install -e ".[dev]"
pytest -q --basetemp .pytest-tmp
django-arch-check analyze /path/to/projectdjango_arch_check/
├── __init__.py
├── cli.py
├── analyzer.py
├── report.py
└── detectors/
├── __init__.py
├── fat_models.py
├── god_apps.py
├── circular_imports.py
├── missing_service_layer.py
├── celery_tasks.py
├── direct_sql.py
├── migration_safety.py
├── n1_serializer_risk.py
└── n_plus_one.py
tests/
├── conftest.py
├── test_analyzer.py
├── test_fat_models.py
├── test_god_apps.py
├── test_circular_imports.py
├── test_missing_service_layer.py
├── test_celery_tasks.py
├── test_direct_sql.py
├── test_migration_safety.py
├── test_n1_serializer_risk.py
├── test_n_plus_one.py
├── test_report.py
└── test_cli.py
- Create
django_arch_check/detectors/my_detector.py - Add a
detect(...)function that returns finding dataclasses - Add the detector to
analyzer.pyandAnalysisResult - Add text output in
cli.py - Add HTML rendering support in
report.py - Add tests
Issues and pull requests are welcome.
If you are proposing a larger detector or behavior change, opening an issue first is appreciated.
The current release version is 0.8.0.
MIT. See LICENSE.
