A convenient package for building both client and server implementations for JWTs or Json Web Tokens. It allows you to build either side by only including the module you need and includes an optional custom user model.
Project description
Django-EasyJWT
A Django package for implementing remote JWT authentication in microservice architectures. It provides a centralized authentication service with multiple client services authenticating against it.
Table of Contents
- Why Django-EasyJWT?
- Features
- Installation
- Architecture Overview
- Quick Start
- Configuration Reference
- API Endpoints
- Testing the API
- Advanced Usage
- Security
- Running Tests
- Changelog
- Acknowledgements
- License
Why Django-EasyJWT?
When managing multiple services with the same users, a centralized authentication service eliminates password confusion and provides a single source of truth. Django-EasyJWT was built for scenarios where:
- Multiple services share the same user base
- Different services require different access levels
- You need custom user data and permissions passed through authentication
- You want to keep your auth service lean and behind a private network
Features
- JWT Authentication: Access/refresh token pairs and sliding tokens
- Remote Auth: Client services authenticate against a central auth service
- Stateless Auth: Optional token-only authentication with no DB lookup (
JWTStatelessUserAuthentication) - Session Support: Authenticated users can access Django admin
- Custom User Model: Optional email-based user model included
- Token Blacklisting: Optional token revocation via outstanding/blacklist tracking
- Password Security: Django password validators enforced on create-user and password-change
- User Registration: Built-in
create-user/endpoint with email normalization - Password Change: Built-in
password-change/endpoint with credential verification - Sliding Tokens: Optional single-token JWT with embedded refresh window
- Extensible: Custom serializers for additional user data
- Configurable: HTTP timeouts, SSL verification, and configurable endpoint paths
- Production Warnings: Automatic logging when SSL or insecure HTTP is configured
Installation
uv add django-easyjwt
Or with pip:
pip install django-easyjwt
Architecture Overview
┌─────────────────┐ ┌─────────────────┐
│ Client-Service │ ──────► │ Auth-Service │
│ (Port 8001) │ │ (Port 8000) │
├─────────────────┤ ├─────────────────┤
│ easyjwt_client │ │ easyjwt_auth │
│ easyjwt_user │ │ easyjwt_user │
└─────────────────┘ └─────────────────┘
│ │
▼ ▼
Local DB Auth DB
(User copies) (User source)
The package contains three sub-packages:
| Package | Used In | Purpose |
|---|---|---|
easyjwt_auth |
Auth-Service | JWT token generation and validation |
easyjwt_client |
Client-Service | Remote authentication against auth-service |
easyjwt_user |
Both (optional) | Custom user model with email as username |
Quick Start
Create an Auth-Service
uv venv
source .venv/bin/activate
uv pip install django djangorestframework django_easyjwt
uv run django-admin startproject config
mv config auth-service
cd auth-service
Add to config/settings.py:
from datetime import timedelta
INSTALLED_APPS = [
# ...
'rest_framework',
'easyjwt_auth',
'easyjwt_user',
]
AUTH_USER_MODEL = "easyjwt_user.User"
REST_FRAMEWORK = {
"DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.IsAuthenticated",),
"DEFAULT_AUTHENTICATION_CLASSES": (
"rest_framework.authentication.SessionAuthentication",
"easyjwt_auth.authentication.JWTAuthentication",
),
}
EASY_JWT = {
"ACCESS_TOKEN_LIFETIME": timedelta(minutes=5),
"REFRESH_TOKEN_LIFETIME": timedelta(days=1),
"ROTATE_REFRESH_TOKENS": False,
"BLACKLIST_AFTER_ROTATION": False,
"UPDATE_LAST_LOGIN": False,
"ALGORITHM": "HS256",
"SIGNING_KEY": "d577273ff885c3f84dadb8578bb40000", # Set properly for production!
"VERIFYING_KEY": None,
"AUDIENCE": None,
"ISSUER": None,
"JWK_URL": None,
"LEEWAY": 0,
"USER_ID_FIELD": "id",
"USER_ID_CLAIM": "user_id",
"USER_AUTHENTICATION_RULE": "easyjwt_auth.authentication.default_user_authentication_rule",
"AUTH_TOKEN_CLASSES": ("easyjwt_auth.tokens.AccessToken",),
"TOKEN_TYPE_CLAIM": "token_type",
"TOKEN_USER_CLASS": "easyjwt_auth.models.TokenUser",
"JTI_CLAIM": "jti",
"CHECK_REVOKE_TOKEN": False,
"SLIDING_TOKEN_REFRESH_EXP_CLAIM": "refresh_exp",
"SLIDING_TOKEN_LIFETIME": timedelta(minutes=5),
"SLIDING_TOKEN_REFRESH_LIFETIME": timedelta(days=1),
}
Add to config/urls.py:
from django.urls import path, include
urlpatterns = [
# ...
path('auth/', include('easyjwt_auth.urls')),
path('auth/', include('easyjwt_user.urls')),
]
Run migrations and create a superuser:
uv run python manage.py makemigrations
uv run python manage.py migrate
uv run python manage.py createsuperuser
uv run python manage.py runserver 0.0.0.0:8000
Create a Client-Service
cd ..
uv venv
source .venv/bin/activate
uv pip install django django_rest_framework django_easyjwt
uv run django-admin startproject config
mv config client-service
cd client-service
Add to config/settings.py:
INSTALLED_APPS = [
# ...
'rest_framework',
'easyjwt_client',
'easyjwt_user',
]
AUTH_USER_MODEL = "easyjwt_user.User"
REST_FRAMEWORK = {
"DEFAULT_PERMISSION_CLASSES": ("rest_framework.permissions.IsAuthenticated",),
"DEFAULT_AUTHENTICATION_CLASSES": (
"easyjwt_client.authentication.EasyJWTAuthentication",
"rest_framework.authentication.SessionAuthentication",
),
}
AUTHENTICATION_BACKENDS = [
'django.contrib.auth.backends.ModelBackend',
'easyjwt_client.authentication.RemoteAuthBackend',
]
EASY_JWT = {
"AUTH_HEADER_TYPES": ("Bearer",),
"AUTH_HEADER_NAME": "Authorization",
"REMOTE_AUTH_SERVICE_URL": "http://127.0.0.1:8000",
"REMOTE_AUTH_SERVICE_TOKEN_PATH": "/auth/token/",
"REMOTE_AUTH_SERVICE_REFRESH_PATH": "/auth/token/refresh/",
"REMOTE_AUTH_SERVICE_VERIFY_PATH": "/auth/token/verify/",
"REMOTE_AUTH_SERVICE_USER_PATH": "/auth/user/",
"REMOTE_AUTH_SERVICE_PASSWORD_CHANGE_PATH": "/auth/password-change/",
"REMOTE_AUTH_REQUEST_TIMEOUT": 30,
"REMOTE_AUTH_SSL_VERIFY": True,
"USER_ID_FIELD": "id",
"USER_ID_CLAIM": "user_id",
}
Add to config/urls.py:
from django.urls import path, include
from test_app.views import TestView
urlpatterns = [
path('admin/', admin.site.urls),
path('auth/', include("easyjwt_client.urls")),
path('api/test/', TestView.as_view()),
]
Create test_app/views.py:
from rest_framework import generics
from rest_framework.response import Response
class TestView(generics.GenericAPIView):
def get(self, request):
return Response("success", status=200)
Run migrations:
uv run python manage.py makemigrations
uv run python manage.py migrate
uv run python manage.py runserver 0.0.0.0:8001
Standing up the Services
Run both services in separate terminals:
# Terminal 1 - Auth-Service
cd auth-service
source .venv/bin/activate
uv run python manage.py runserver 0.0.0.0:8000
# Terminal 2 - Client-Service
cd client-service
source .venv/bin/activate
uv run python manage.py runserver 0.0.0.0:8001
Configuration Reference
Auth-Service Settings
| Setting | Type | Default | Description |
|---|---|---|---|
ACCESS_TOKEN_LIFETIME |
timedelta | timedelta(minutes=5) |
Access token validity duration |
REFRESH_TOKEN_LIFETIME |
timedelta | timedelta(days=1) |
Refresh token validity duration |
ROTATE_REFRESH_TOKENS |
bool | False |
Issue a new refresh token on each refresh |
BLACKLIST_AFTER_ROTATION |
bool | False |
Blacklist old refresh token after rotation |
UPDATE_LAST_LOGIN |
bool | False |
Update user's last_login on token issuance |
ALGORITHM |
str | "HS256" |
JWT signing algorithm |
SIGNING_KEY |
str | settings.SECRET_KEY |
Secret key for signing tokens |
VERIFYING_KEY |
str | "" |
Public key for asymmetric algorithms |
AUDIENCE |
str | None |
Expected JWT audience claim |
ISSUER |
str | None |
Expected JWT issuer claim |
LEEWAY |
int/float | 0 |
Leeway in seconds for token expiration checks |
AUTH_HEADER_TYPES |
tuple | ("Bearer",) |
Valid Authorization header types |
AUTH_HEADER_NAME |
str | "HTTP_AUTHORIZATION" |
Header name for extracting JWT |
USER_ID_FIELD |
str | "id" |
User model field for ID |
USER_ID_CLAIM |
str | "user_id" |
JWT claim name for user ID |
TOKEN_USER_CLASS |
str | "easyjwt_auth.models.TokenUser" |
Stateless user class for token-only auth |
CHECK_REVOKE_TOKEN |
bool | False |
Invalidate tokens on password change |
REVOKE_TOKEN_CLAIM |
str | "hash_password" |
Claim name for password hash revocation check |
SLIDING_TOKEN_LIFETIME |
timedelta | timedelta(minutes=5) |
Sliding token validity |
SLIDING_TOKEN_REFRESH_LIFETIME |
timedelta | timedelta(days=1) |
Sliding token refresh window |
Client-Service Settings
| Setting | Type | Default | Description |
|---|---|---|---|
REMOTE_AUTH_SERVICE_URL |
str | "http://127.0.0.1:8000" |
Base URL of auth-service |
REMOTE_AUTH_SERVICE_TOKEN_PATH |
str | "/auth/token/" |
Token endpoint path |
REMOTE_AUTH_SERVICE_REFRESH_PATH |
str | "/auth/token/refresh/" |
Refresh endpoint path |
REMOTE_AUTH_SERVICE_VERIFY_PATH |
str | "/auth/token/verify/" |
Verify endpoint path |
REMOTE_AUTH_SERVICE_USER_PATH |
str | "/auth/user/" |
User endpoint path |
REMOTE_AUTH_SERVICE_PASSWORD_CHANGE_PATH |
str | "/auth/password-change/" |
Password change endpoint path |
REMOTE_AUTH_REQUEST_TIMEOUT |
int | 30 |
HTTP request timeout in seconds |
REMOTE_AUTH_SSL_VERIFY |
bool | True |
Verify SSL certificates |
AUTH_HEADER_TYPES |
tuple | ("Bearer",) |
Valid auth header types |
USER_MODEL_SERIALIZER |
str | "easyjwt_user.serializers.TokenUserSerializer" |
Serializer for deserializing remote user data |
API Endpoints
Auth-Service (easyjwt_auth.urls)
| Endpoint | Method | Auth | Description |
|---|---|---|---|
/auth/token/ |
POST | None | Obtain access and refresh tokens |
/auth/token/refresh/ |
POST | None | Refresh an expired access token |
/auth/token/verify/ |
POST | None | Verify if a token is valid |
/auth/create-user/ |
POST | None | Register a new user account |
/auth/password-change/ |
POST | None | Change password (verifies credentials) |
/auth/login/ |
GET | None | Django LoginView for session auth |
Client-Service (easyjwt_client.urls)
| Endpoint | Method | Auth | Description |
|---|---|---|---|
/auth/token/ |
POST | None | Obtain tokens via remote auth-service |
/auth/token/refresh/ |
POST | None | Refresh token via remote |
/auth/token/verify/ |
POST | None | Verify token via remote |
/auth/password-change/ |
POST | Session | Change password (validates match) |
/auth/password-change/done/ |
GET | Session | Password change confirmation page |
/auth/password-reset/ |
GET | None | Redirects to auth-service reset page |
/auth/login/ |
GET | None | Django LoginView for session auth |
User (easyjwt_user.urls)
| Endpoint | Method | Auth | Description |
|---|---|---|---|
/auth/user/ |
GET | JWT | Get current user's profile data |
Testing the API
Obtain Token Pair
curl -X POST \
-H "Content-Type: application/json" \
-d '{"email": "user@test.com", "password": "user-pass"}' \
http://127.0.0.1:8001/auth/token/
Response:
{
"refresh": "...",
"access": "..."
}
Make Authenticated Request
export ACCESS_TOKEN=<your_access_token>
curl -H "Authorization: Bearer ${ACCESS_TOKEN}" \
http://127.0.0.1:8001/api/test/
Response: "success"
Refresh Token
curl -X POST \
-H "Content-Type: application/json" \
-d '{"refresh": "${REFRESH_TOKEN}"}' \
http://127.0.0.1:8001/auth/token/refresh/
Verify Token
curl -X POST \
-H "Content-Type: application/json" \
-d '{"token": "${ACCESS_TOKEN}"}' \
http://127.0.0.1:8001/auth/token/verify/
Create User (Auth-Service)
curl -X POST \
-H "Content-Type: application/json" \
-d '{"email": "newuser@test.com", "password": "S3cure!P@ssw0rd"}' \
http://127.0.0.1:8000/auth/create-user/
Response:
{
"id": 1,
"email": "newuser@test.com",
"first_name": "",
"last_name": "",
"phone": null
}
Change Password (Auth-Service)
curl -X POST \
-H "Content-Type: application/json" \
-d '{"email": "user@test.com", "password": "old-pass", "new_password": "N3wP@ss!2026"}' \
http://127.0.0.1:8000/auth/password-change/
Get User Profile
curl -H "Authorization: Bearer ${ACCESS_TOKEN}" \
http://127.0.0.1:8000/auth/user/
Response:
{
"id": 1,
"email": "user@test.com",
"first_name": "Test",
"last_name": "User"
}
Advanced Usage
Extra Data
You can pass additional user data from the auth-service to client-services using custom serializers.
Auth-Service Configuration
models.py:
class AccessGroup(models.Model):
user = models.OneToOneField(User, related_name="accessgroup", on_delete=models.CASCADE)
user_type = models.TextField()
serializers.py:
class AccessGroupSerializer(serializers.ModelSerializer):
class Meta:
model = AccessGroup
fields = ("user_type",)
class TokenUserSerializer(serializers.ModelSerializer):
accessgroup = AccessGroupSerializer()
class Meta:
model = User
fields = ("id", "email", "first_name", "last_name", "accessgroup")
Add to EASY_JWT settings:
"USER_MODEL_SERIALIZER": "userdata.serializers.TokenUserSerializer",
Client-Service Configuration
Use the same model and serializer, but override create() and update() methods:
class TokenUserSerializer(serializers.ModelSerializer):
accessgroup = AccessGroupSerializer()
class Meta:
model = User
fields = ("id", "email", "first_name", "last_name", "accessgroup")
def create(self, validated_data):
accessgroup = validated_data.pop("accessgroup")
user, _ = User.objects.get_or_create(
email=validated_data.pop("email"),
defaults=validated_data
)
AccessGroup.objects.update_or_create(user=user, defaults=accessgroup)
return user
def update(self, instance, validated_data):
accessgroup = validated_data.pop("accessgroup")
for attr, value in validated_data.items():
setattr(instance, attr, value)
instance.save()
AccessGroup.objects.update_or_create(user=instance, defaults=accessgroup)
return instance
Permissions
Use custom permission classes to control access based on user data:
from rest_framework import permissions
class AccessGroupPermission(permissions.BasePermission):
message = "You do not have permission to this service"
def has_permission(self, request, view):
return (
not request.user.is_anonymous
and hasattr(request.user, 'accessgroup')
and view.access_level.startswith(request.user.accessgroup.user_type)
)
Security
- Password Validation: Django's
AUTH_PASSWORD_VALIDATORSare enforced on both thecreate-user/andpassword-change/endpoints - Email Normalization: User creation uses
UserManager.create_user()which normalizes and lowercases email addresses - Credential Verification: Password change requires the current password — old credentials are verified before accepting a new password
- Error Sanitization: Remote auth error responses do not leak upstream server details (raw response bodies, tracebacks) to clients
- SSL/HTTP Warnings: The client app logs a warning at startup when
DEBUG=Falseand eitherSSL_VERIFY=FalseorREMOTE_AUTH_SERVICE_URLuseshttp:// - Token Revocation: Optional
CHECK_REVOKE_TOKENinvalidates all outstanding JWTs when a user changes their password - Token Blacklisting: Optional
easyjwt_auth.token_blacklistapp provides server-side token revocation with outstanding token tracking
Running Tests
uv pip install -e ".[dev]"
uv run pytest
Or with coverage:
uv run pytest --cov=easyjwt_auth --cov=easyjwt_client --cov=easyjwt_user
Changelog
See CHANGELOG.md for full version history.
v1.0.6+
- Fixed
TOKEN_USER_CLASSdefault pointing to wrong module path CreateUserSerializernow usescreate_user()for email normalization- Added
REMOTE_AUTH_SERVICE_PASSWORD_CHANGE_PATHsetting (was hardcoded) TokenViewBaseserializer classes resolved at runtime, not import time- Added catch-all
requests.RequestExceptionhandler for unhandled HTTP errors - Merged CI publish workflow into version workflow (fixes
[skip ci]blocking PyPI)
v1.0.5+
- Security hardening — 21 review issues fixed across security, bugs, design, and test coverage
- Removed dead
jwt_secretfield (migration 0002) - Unified settings access via
api_settingsacross all modules RemoteAuthBackendinheritsBaseBackend(correct Django signature)- Added comprehensive test coverage (61% → 88%, 155 tests)
- Password confirmation check, response sanitization, admin password validation
- SSL/HTTP production warning in client
apps.py
v1.0.0
- Replaced MD5 with SHA-256 for password hashing
- Renamed
ModelBackendtoRemoteAuthBackend - Added configurable HTTP request timeouts
- Made SSL verification configurable
- Consolidated HTTP error handling
- Added
__all__exports to public modules
Acknowledgements
This package is heavily based on djangorestframework-simplejwt and influenced by SimpleJWT.
An example implementation is available at django-easyjwt-example.
License
MIT License - see LICENCE file.
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_easyjwt-1.0.9.tar.gz.
File metadata
- Download URL: django_easyjwt-1.0.9.tar.gz
- Upload date:
- Size: 34.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
975e73bf598cab9716331216568ae9965403861c651a213ce2f018a6cda0486f
|
|
| MD5 |
3d67883adc2f6ae4032acd9f85b9f230
|
|
| BLAKE2b-256 |
0ef8a2f629a8219ff98ac9ec1f1ce8461cf9aeb994d633a90d2ed337711e5c8a
|
Provenance
The following attestation bundles were made for django_easyjwt-1.0.9.tar.gz:
Publisher:
release-version.yml on garrethcain/django-easyjwt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
django_easyjwt-1.0.9.tar.gz -
Subject digest:
975e73bf598cab9716331216568ae9965403861c651a213ce2f018a6cda0486f - Sigstore transparency entry: 1843564656
- Sigstore integration time:
-
Permalink:
garrethcain/django-easyjwt@853e3761a4a327b4f3506e113f1d7ada363db1b8 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/garrethcain
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release-version.yml@853e3761a4a327b4f3506e113f1d7ada363db1b8 -
Trigger Event:
push
-
Statement type:
File details
Details for the file django_easyjwt-1.0.9-py3-none-any.whl.
File metadata
- Download URL: django_easyjwt-1.0.9-py3-none-any.whl
- Upload date:
- Size: 44.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b787eb5fe31666006b41b42081fa4095416f9905d912b28594e9eaa2f6349c9e
|
|
| MD5 |
ee2a83c8b31d3829936481c970284b09
|
|
| BLAKE2b-256 |
597ff744c319042353c20e0b1caeb6afaef710e58ee5c92c31fb9453efe7c111
|
Provenance
The following attestation bundles were made for django_easyjwt-1.0.9-py3-none-any.whl:
Publisher:
release-version.yml on garrethcain/django-easyjwt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
django_easyjwt-1.0.9-py3-none-any.whl -
Subject digest:
b787eb5fe31666006b41b42081fa4095416f9905d912b28594e9eaa2f6349c9e - Sigstore transparency entry: 1843564718
- Sigstore integration time:
-
Permalink:
garrethcain/django-easyjwt@853e3761a4a327b4f3506e113f1d7ada363db1b8 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/garrethcain
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release-version.yml@853e3761a4a327b4f3506e113f1d7ada363db1b8 -
Trigger Event:
push
-
Statement type: