Reusable FastAPI base exceptions, ApiMessage, and a unified exception handler
Project description
af-fastapi-exceptions
Reusable exception handling for FastAPI apps: a base ServiceError hierarchy you can raise
(and extend), plus a middleware + validation handler so that every error — raised before,
during, or after the route — reaches the client as the same JSON body.
Bring your own response model. The library never defines or imposes a response schema —
you pass your own model down, and the handlers use only the slice they need (construct it with
message + errors, then call .model_dump()). Your app keeps one model for both success and
error responses; nothing is coupled across the boundary.
Installation
pip install af-fastapi-exceptions
# or with Poetry:
poetry add af-fastapi-exceptions
Raising errors
Raise a ServiceError (or a subclass) anywhere; the middleware turns it into your response
model with the right status code.
from allfly.fastapi.exceptions import EntityNotFoundError, ForbiddenError
def get_user(user_id: str):
user = repo.find(user_id)
if not user:
raise EntityNotFoundError(f"No user {user_id}") # -> 404
if not user.active:
raise ForbiddenError() # -> 403
return user
Built-in classes: UnauthorizedRequestError (401), ForbiddenError (403),
BadRequestError (400), EntityNotFoundError (404), ResourceConflictError (409),
UnprocessableRequestError (422), UnsupportedFeatureError (501),
DownstreamServiceError (502).
ServiceWarning is a ServiceError subclass for expected/recoverable conditions — logged
at warning level instead of error. ForbiddenError and UnsupportedFeatureError are
warnings.
Extend them
from allfly.fastapi.exceptions import UnprocessableRequestError
class FopError(UnprocessableRequestError):
def __init__(self, message="Payment method was rejected."):
super().__init__(message)
Your response model
Provide any model whose instances have a .model_dump() and that can be constructed with
message= and errors= (a pydantic model with those two fields — plus whatever else you want,
e.g. timestamp, metadata — is the common case). The library only ever sets message and
errors; the rest come from your model's defaults. This is the ApiErrorBody protocol:
class ApiErrorBody(Protocol):
def __init__(self, *, message: str, errors: list[str]) -> None: ...
def model_dump(self, *, mode: str = "json", exclude_none: bool = True) -> dict: ...
Wiring it into your app
from fastapi import FastAPI
from fastapi.exceptions import RequestValidationError
from allfly.fastapi.exceptions import (
ExceptionMiddleware,
DefaultExceptionHandlerSettings,
build_validation_handler,
build_error_responses,
)
from myapp.models import ApiMessage # <-- YOUR model
app = FastAPI(responses=build_error_responses(ApiMessage)) # OpenAPI error schemas
# Catches ServiceError (-> its status) and any unexpected Exception (-> 500).
app.add_middleware(
ExceptionMiddleware,
error_model=ApiMessage,
settings=DefaultExceptionHandlerSettings(production=is_production()),
)
# RequestValidationError is raised during request parsing, before the middleware runs,
# so register it as an exception handler too — same body, built from your model.
app.add_exception_handler(RequestValidationError, build_validation_handler(ApiMessage))
Settings (why not read your env directly?)
The middleware only needs to know one thing: are we in production? (In production the 500 handler hides the raw exception string; otherwise it includes it to aid debugging.)
Rather than force a settings system on you, it takes any object matching the
ExceptionHandlerSettings protocol — a single production: bool. Use the provided
DefaultExceptionHandlerSettings, or pass your own object exposing production.
Logging
Handlers log via loguru — ServiceWarning at warning,
real errors at error/exception. If your app doesn't configure loguru, the messages are
simply not emitted at higher levels by default.
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 af_fastapi_exceptions-0.0.2.tar.gz.
File metadata
- Download URL: af_fastapi_exceptions-0.0.2.tar.gz
- Upload date:
- Size: 4.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0eaa9bf4e6d94c7c9a5911254c688ffcb0faa16df5fe859d4ab7edc66bf5301c
|
|
| MD5 |
145aa6ba34950276b59738b260401da6
|
|
| BLAKE2b-256 |
8a8192d5314f385b99455b853608571daddab5842677be2e60fc71fc1a8cea96
|
Provenance
The following attestation bundles were made for af_fastapi_exceptions-0.0.2.tar.gz:
Publisher:
af-fastapi-exceptions-publish.yml on travelallfly/allfly-py-libs
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
af_fastapi_exceptions-0.0.2.tar.gz -
Subject digest:
0eaa9bf4e6d94c7c9a5911254c688ffcb0faa16df5fe859d4ab7edc66bf5301c - Sigstore transparency entry: 2038057138
- Sigstore integration time:
-
Permalink:
travelallfly/allfly-py-libs@97ec8f6cdae25c7c71eb69939af4950a615160e3 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/travelallfly
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
af-fastapi-exceptions-publish.yml@97ec8f6cdae25c7c71eb69939af4950a615160e3 -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file af_fastapi_exceptions-0.0.2-py3-none-any.whl.
File metadata
- Download URL: af_fastapi_exceptions-0.0.2-py3-none-any.whl
- Upload date:
- Size: 6.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 |
279ac7eb996300d7c02466cb4e5065d07097c22cde16165c4324bee9190d48bf
|
|
| MD5 |
bf247bab3308956ba3addcbbffd0e5a8
|
|
| BLAKE2b-256 |
6d7b382b8d65132e5306d93c95f747ddd0decb5340408b7992c66b5f46b4324d
|
Provenance
The following attestation bundles were made for af_fastapi_exceptions-0.0.2-py3-none-any.whl:
Publisher:
af-fastapi-exceptions-publish.yml on travelallfly/allfly-py-libs
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
af_fastapi_exceptions-0.0.2-py3-none-any.whl -
Subject digest:
279ac7eb996300d7c02466cb4e5065d07097c22cde16165c4324bee9190d48bf - Sigstore transparency entry: 2038057641
- Sigstore integration time:
-
Permalink:
travelallfly/allfly-py-libs@97ec8f6cdae25c7c71eb69939af4950a615160e3 -
Branch / Tag:
refs/heads/master - Owner: https://github.com/travelallfly
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
af-fastapi-exceptions-publish.yml@97ec8f6cdae25c7c71eb69939af4950a615160e3 -
Trigger Event:
workflow_dispatch
-
Statement type: