Dev-time conveniences for Django projects: autologin endpoint, dotfiles for LLM agents, gitignore self-check
Project description
django-dev-helpers
Dev-time conveniences for Django projects: autologin endpoint, dotfiles for LLM coding agents, agent help prompt, and gitignore self-check.
Features
- Autologin endpoint — one URL logs in a user via token, no interactive login needed
- Auth-state query toggles —
?__autologin__=tmp_off|logout|log_inon any URL to flip auth state in the browser without leaving the page - Dotfiles —
.dev_helpers_token,.dev_helpers_port,.dev_helpers_pg_*,.dev_helpers_redis_*written to project root for easycatby LLM agents - Agent help prompt — copy-pasteable curl/psql/redis-cli commands printed at startup
- Gitignore self-check — warns if dotfiles are not in
.gitignore - Browser auto-open — opens autologin URL in browser after server starts
- Production-safe — default-off, requires explicit
enabled=True, raises onDEBUG=False
Installation
pip install django-dev-helpers
# or
uv add django-dev-helpers --group dev
Quick Start
- Add to
INSTALLED_APPS:
INSTALLED_APPS = [
# ...
"django_dev_helpers",
]
- Enable in settings:
DJANGO_DEV_HELPERS = {"enabled": True}
Or via environment variable:
DJANGO_DEV_HELPERS_ENABLED=1 python manage.py runserver
That's it — autologin is wired automatically. On startup, the package will:
- Generate an autologin token
- Auto-install
AutologinMiddlewareintoMIDDLEWAREso the autologin URL works withouturls.pychanges (refuses to load ifDEBUG=False) - Write dotfiles to your project root
- Check
.gitignore - Print agent help after first request
- Open browser at the autologin URL (falls back to
/with a banner if the URL is somehow not wired)
If you'd rather wire the URL by hand (e.g. to mount it under a prefix), set
{"autologin": {"middleware_autoinstall": False}} and add to urls.py:
from django_dev_helpers.urls import autologin_urlpatterns
urlpatterns = [
*autologin_urlpatterns(),
# ... your other URLs
]
Usage
Autologin URL (token-based)
T=$(cat .dev_helpers_token)
curl -L "http://localhost:8000/__autologin__/?token=$T"
Auth-state toggles (browser-friendly)
Once AutologinMiddleware is wired (the default), every request is scanned for
a toggle query parameter. Drop it onto any URL — the middleware handles it
before URL resolution.
| URL on any view | Effect |
|---|---|
https://localhost:8000/some/page/?__autologin__=tmp_off |
Render this one request with request.user = AnonymousUser. Session unchanged — the next plain request is logged in again. Toggle param stripped from request.GET before the view sees it. |
https://localhost:8000/some/page/?__autologin__=logout |
django.contrib.auth.logout(request) — ends the session. 302 to the same path with the toggle stripped; other query parameters preserved. |
https://localhost:8000/some/page/?__autologin__=log_in (or login) |
Log in the configured user (autologin.user_lookup_field / user_lookup_value). 302 to the cleaned URL. No URL token required — the localhost host allowlist is the trust signal. |
Unknown values pass through silently (likely typos). Off-host requests pass through identically — the toggles do not announce their existence to unauthorized hosts.
Rename the parameter via autologin.query_param, or set it to "" / None
to disable the toggle layer while keeping the path-based /__autologin__/
URL working. Full details and threat model:
docs/autologin.md.
Middleware ordering
AutologinMiddleware is auto-appended at the end of settings.MIDDLEWARE
during AppConfig.ready(). That works because the toggles need
SessionMiddleware, AuthenticationMiddleware, and MessageMiddleware to
have already run by the time we look at the request — sessions for
logout/log_in, request.user set up so tmp_off can override it, and
request._messages for the path-based view's flash_message.
If you install the middleware manually (with
autologin.middleware_autoinstall=False), place it after those three:
MIDDLEWARE = [
"django.contrib.sessions.middleware.SessionMiddleware",
"django.contrib.auth.middleware.AuthenticationMiddleware",
"django.contrib.messages.middleware.MessageMiddleware",
# ... your other middleware ...
"django_dev_helpers.middleware.AutologinMiddleware",
]
Putting it before SessionMiddleware/AuthenticationMiddleware/MessageMiddleware will break logout, log_in, and flash_message respectively.
Management Commands
python manage.py dev_helpers_doctor # Full diagnostic
python manage.py dev_helpers_print_help # Print agent prompt
python manage.py dev_helpers_check_gitignore # Check .gitignore entries (read-only)
python manage.py dev_helpers_fix_gitignore # Add missing .gitignore entries (idempotent)
dev_helpers_fix_gitignore is the one you want when you see the
"missing entries from .gitignore" warning on startup — it appends the
missing dotfile names (and only those) without reordering existing
rules. Pass --dry-run to preview. See
docs/configuration.md#gitignore for
the full contract.
Documentation
- Quickstart
- Configuration reference
- Autologin endpoint
- Dotfiles + lookup chain
- Agent help / prompt
- Standalone usage
- Using with django-run-site
- Security
- Original design spec (frozen) — for design rationale
Configuration
All configuration via settings.DJANGO_DEV_HELPERS dict. See configuration docs for full reference.
DJANGO_DEV_HELPERS = {
"enabled": True,
"autologin": {
"user_lookup_field": "username",
"user_lookup_value": "admin",
"url_path": "__autologin__/",
"redirect_to": "/",
# Middleware that handles the autologin URL + auth-state toggles.
# Auto-appended to settings.MIDDLEWARE; refuses to load when DEBUG=False.
"middleware_autoinstall": True,
# Name of the query toggle (?__autologin__=tmp_off|logout|log_in).
# Set to "" or None to disable the toggle layer.
"query_param": "__autologin__",
},
"dotfiles": {
"enabled": True,
},
"agent_help": {
"auto_print": True,
},
"browser_open": {
"enabled": True,
},
"gitignore": {
"mode": "warn", # warn | auto-add | error | off
},
}
Security
This package exposes an autologin backdoor for development. It is always off by default:
enabledmust be explicitly set toTrue(via settings or env var)- Raises
ImproperlyConfiguredifDEBUG=Falseand serving HTTP - Autologin view verifies token via
hmac.compare_digest(timing-safe) - Only accepts requests from localhost/127.0.0.1 by default
- Returns 404 (not 403/401) on any failure — endpoint appears non-existent
Never install this package in production. Add it only to dev dependencies.
Requirements
- Python >= 3.11
- Django >= 4.2
Supported versions
Combinations exercised on every push by the CI matrix
(.github/workflows/ci.yml):
| Python 3.11 | Python 3.12 | Python 3.13 | |
|---|---|---|---|
| Django 4.2 | ✓ | ✓ | ✓ |
| Django 5.0 | ✓ | ✓ | ✓ |
| Django 5.1 | ✓ | ✓ | ✓ |
| Django 5.2 | ✓ | ✓ | ✓ |
| Django 6.0 | — | ✓ | ✓ |
Django 6.0 requires Python ≥ 3.12, so the (3.11, 6.0) cell is excluded
from CI.
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 django_dev_helpers-0.1.8.tar.gz.
File metadata
- Download URL: django_dev_helpers-0.1.8.tar.gz
- Upload date:
- Size: 105.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0ce5579f1ce00d2e3e99094fbee3cdc7c8d0ab4da25486fcf65a5de298e83f30
|
|
| MD5 |
2ecc60c922508b77227dcdc1c026c8be
|
|
| BLAKE2b-256 |
5868a5382e575ff7913e9582b0f50d55b1759afd4d10f13a097e7dc6993f29db
|
File details
Details for the file django_dev_helpers-0.1.8-py3-none-any.whl.
File metadata
- Download URL: django_dev_helpers-0.1.8-py3-none-any.whl
- Upload date:
- Size: 39.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5f144f5281398cacf89e7f40c609e022d2fd047abd5eca0a5ad844e9289071f3
|
|
| MD5 |
06b7a7b532412c0a91096ffba15113f2
|
|
| BLAKE2b-256 |
efe9290a4c1e9d4daf3d330f96352305e26657b71a25aa170731deeab873f58b
|