FastAPI middleware for request ID tracking, correlation IDs, and extensible request context with logging integration
Project description
fastapi-request-context
FastAPI middleware for request ID tracking, correlation IDs, and extensible request context with first-class logging integration.
Features
- Automatic request ID generation - Every request gets a unique ID
- Correlation ID support - Accept from header or generate for distributed tracing
- Response header injection - Automatically add
X-Request-IdandX-Correlation-Idto responses - Pluggable context backends - Use
contextvars(default) orcontext-logging - Custom context fields - Extend with your own fields via StrEnum
- Logging integration - JSON and human-readable formatters with automatic context injection
- Validation utilities - Check that routes and dependencies are async
- Zero configuration - Works out of the box with sensible defaults
- Type-safe - Full type hints and mypy strict mode
Installation
# Basic installation
pip install fastapi-request-context
# With context-logging support
pip install fastapi-request-context[context-logging]
# With JSON formatter support
pip install fastapi-request-context[json-formatter]
# All optional dependencies
pip install fastapi-request-context[all]
Using uv:
uv add fastapi-request-context
Quick Start
from fastapi import FastAPI
from fastapi_request_context import RequestContextMiddleware
# Create your app
app = FastAPI()
# Keep reference to raw app if needed (e.g., for TaskIQ, testing)
raw_app = app
# Wrap with middleware
app = RequestContextMiddleware(app)
Every request now has:
- Unique
request_id(always generated) correlation_id(fromX-Correlation-Idheader or generated)- Both added to response headers
- Context available in all log records (including access logs!)
Usage
Access Context Values
from fastapi_request_context import get_context, get_full_context, StandardContextField
@app.get("/")
async def root():
request_id = get_context(StandardContextField.REQUEST_ID)
correlation_id = get_context(StandardContextField.CORRELATION_ID)
# Or get everything
all_context = get_full_context()
return {"request_id": request_id}
Custom Context Fields
from enum import StrEnum
from fastapi_request_context import set_context, get_context
class MyContextField(StrEnum):
USER_ID = "user_id"
ORG_ID = "org_id"
async def get_current_user(token: str):
user_id = decode_token(token)
set_context(MyContextField.USER_ID, user_id)
return user_id
@app.get("/me")
async def me(user_id: int = Depends(get_current_user)):
# Context is available throughout the request
return {"user_id": get_context(MyContextField.USER_ID)}
Configuration
from fastapi_request_context import RequestContextMiddleware, RequestContextConfig
from uuid import uuid4
config = RequestContextConfig(
# Custom ID generators
request_id_generator=lambda: str(uuid4()),
correlation_id_generator=lambda: str(uuid4()),
# Custom header names
request_id_header="X-My-Request-Id",
correlation_id_header="X-My-Correlation-Id",
# Disable response headers
add_response_headers=False,
# Use context-logging adapter
context_adapter="context_logging",
# Only process HTTP (not WebSocket)
scope_types={"http"},
)
app = RequestContextMiddleware(app, config=config)
Logging Integration
JSON Formatter (Production)
import logging
from fastapi_request_context.formatters import JsonContextFormatter
handler = logging.StreamHandler()
handler.setFormatter(JsonContextFormatter())
logging.basicConfig(handlers=[handler], level=logging.INFO)
# Logs automatically include context:
# {"message": "Processing", "level": "INFO", "request_id": "...", "user_id": 123}
Simple Formatter (Human-Readable)
from fastapi_request_context import StandardContextField
from fastapi_request_context.formatters import SimpleContextFormatter
handler = logging.StreamHandler()
handler.setFormatter(SimpleContextFormatter(
fmt="%(asctime)s %(levelname)s %(context)s %(message)s",
shorten_fields={StandardContextField.REQUEST_ID}, # Show first 8 chars
hidden_fields={StandardContextField.CORRELATION_ID}, # Hide completely
))
logging.basicConfig(handlers=[handler], level=logging.INFO)
# Output: 2025-01-15 10:30:00 INFO [request_id=3fa85f64 user_id=123] Processing
Access Logs Integration
Context is automatically available in all log records, including Uvicorn access logs when using context-logging adapter:
from fastapi_request_context import RequestContextMiddleware, RequestContextConfig
config = RequestContextConfig(context_adapter="context_logging")
app = RequestContextMiddleware(app, config=config)
# Now access logs will include request_id and correlation_id!
# Example: INFO 127.0.0.1:8000 - "GET / HTTP/1.1" 200 [request_id=abc123]
Custom Context Adapter
from fastapi_request_context.adapters import ContextAdapter
class RedisAdapter(ContextAdapter):
def set_value(self, key: str, value: Any) -> None:
redis.hset(self._request_key, key, value)
def get_value(self, key: str) -> Any:
return redis.hget(self._request_key, key)
def get_all(self) -> dict[str, Any]:
return redis.hgetall(self._request_key)
def enter_context(self, initial_values: dict[str, Any]) -> None:
self._request_key = f"request:{uuid4()}"
redis.hmset(self._request_key, initial_values)
def exit_context(self) -> None:
redis.delete(self._request_key)
config = RequestContextConfig(context_adapter=RedisAdapter())
app = RequestContextMiddleware(app, config=config)
Validation Utilities
Ensure all routes and dependencies are async (required for proper context propagation):
from fastapi_request_context.validation import check_routes_and_dependencies_are_async
@app.on_event("startup")
async def validate():
warnings = check_routes_and_dependencies_are_async(app)
# Logs warnings for any sync routes/dependencies
# Or raise an error
check_routes_and_dependencies_are_async(app, raise_on_sync=True)
API Reference
Middleware
RequestContextMiddleware(app, config=None)- Main middleware class
Configuration
RequestContextConfig- Configuration dataclass with all options
Context Functions
set_context(key, value)- Set a context valueget_context(key)- Get a context value (returns None if not set)get_full_context()- Get all context values as a dict
Fields
StandardContextField- Built-in fields (REQUEST_ID, CORRELATION_ID)
Adapters
ContextAdapter- Protocol for custom adaptersContextVarsAdapter- Default adapter using Python's contextvarsContextLoggingAdapter- Adapter using context-logging library (enables access log integration)
Formatters
JsonContextFormatter- JSON formatter for structured loggingSimpleContextFormatter- Human-readable formatter with inline context
Validation
is_async(func)- Check if a function is asynccheck_dependencies_are_async(deps)- Check dependenciescheck_routes_and_dependencies_are_async(app)- Check entire app
Why This Library?
vs. Manual Implementation
| Feature | Manual | This Library |
|---|---|---|
| Request ID generation | DIY | ✅ Built-in |
| Correlation ID | DIY | ✅ Built-in |
| Response headers | DIY | ✅ Automatic |
| Context storage | DIY | ✅ Pluggable |
| Logging integration | DIY | ✅ Included |
| Type safety | Maybe | ✅ Full |
| Tests | Maybe | ✅ 100% coverage |
vs. Other Libraries
- Zero dependencies beyond FastAPI (optional extras available)
- Pluggable adapters - not locked to one context library
- Validation utilities - catch sync code issues early
- Production-ready formatters - JSON and local dev support
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
# Clone the repo
git clone https://github.com/ADR-007/fastapi-request-context.git
cd fastapi-request-context
# Install dependencies
uv sync --all-extras
# Run tests
make test
# Run linting
make lint
# Fix linting issues
make lint-fix
License
MIT License - see LICENSE for details.
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
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 fastapi_request_context-1.0.0.tar.gz.
File metadata
- Download URL: fastapi_request_context-1.0.0.tar.gz
- Upload date:
- Size: 75.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
88479d5bcf28aef4c6b61b04514fa497c214c475cda8154bbf8a598119eee331
|
|
| MD5 |
a9465cb19f50f2bff3a9890b46975f04
|
|
| BLAKE2b-256 |
055f4f1a35b8b130b385bf1b5e54b62b32ac9330cfa8cee1e8d321f9fcdea0eb
|
Provenance
The following attestation bundles were made for fastapi_request_context-1.0.0.tar.gz:
Publisher:
release.yaml on ADR-007/fastapi-request-context
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastapi_request_context-1.0.0.tar.gz -
Subject digest:
88479d5bcf28aef4c6b61b04514fa497c214c475cda8154bbf8a598119eee331 - Sigstore transparency entry: 731166699
- Sigstore integration time:
-
Permalink:
ADR-007/fastapi-request-context@e3901c948439205a4a83daa542a7a736b8c0752e -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ADR-007
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@e3901c948439205a4a83daa542a7a736b8c0752e -
Trigger Event:
workflow_run
-
Statement type:
File details
Details for the file fastapi_request_context-1.0.0-py3-none-any.whl.
File metadata
- Download URL: fastapi_request_context-1.0.0-py3-none-any.whl
- Upload date:
- Size: 19.5 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 |
ee9fbb5bc3cc7a4eea5dc97abb9cd8aa361f5fa0f780d71eacd29f83a99e2969
|
|
| MD5 |
775cedf4391c9933cc8a5c88f8d16ac4
|
|
| BLAKE2b-256 |
21739cc22b6ffe3b26eb25e8e2528d6d5c42d06a55ca4770b0f029126c1d78dd
|
Provenance
The following attestation bundles were made for fastapi_request_context-1.0.0-py3-none-any.whl:
Publisher:
release.yaml on ADR-007/fastapi-request-context
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastapi_request_context-1.0.0-py3-none-any.whl -
Subject digest:
ee9fbb5bc3cc7a4eea5dc97abb9cd8aa361f5fa0f780d71eacd29f83a99e2969 - Sigstore transparency entry: 731166702
- Sigstore integration time:
-
Permalink:
ADR-007/fastapi-request-context@e3901c948439205a4a83daa542a7a736b8c0752e -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ADR-007
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@e3901c948439205a4a83daa542a7a736b8c0752e -
Trigger Event:
workflow_run
-
Statement type: