Unified error handling for Netrun Systems FastAPI applications
Project description
netrun-errors
Unified error handling for Netrun Systems FastAPI applications.
Version 2.0.0 - Namespace Package Migration
IMPORTANT: Starting with version 2.0.0, netrun-errors uses namespace packaging. The import path has changed from netrun_errors to netrun.errors.
Migration Guide
# OLD (deprecated, still works but shows warning):
from netrun_errors import NetrunException, InvalidCredentialsError
from netrun_errors import install_exception_handlers, install_error_logging_middleware
# NEW (recommended):
from netrun.errors import NetrunException, InvalidCredentialsError
from netrun.errors import install_exception_handlers, install_error_logging_middleware
Backwards Compatibility: The old netrun_errors import path continues to work in version 2.x with a deprecation warning. It will be removed in version 3.0.0.
Why this change? Namespace packaging aligns with Python standards and enables better organization across the Netrun ecosystem, allowing multiple related packages under the netrun.* namespace.
Features
- Structured Error Responses: Consistent JSON error format across all services
- Automatic Correlation IDs: Request tracking with unique identifiers
- Machine-Readable Error Codes: Frontend-friendly error classification
- Global Exception Handlers: FastAPI integration with comprehensive error handling
- Request/Response Logging: Middleware with performance tracking
- Type Safety: Full typing support with mypy strict mode
- Comprehensive Testing: 90%+ test coverage
Installation
pip install netrun-errors
Quick Start
from fastapi import FastAPI
from netrun.errors import (
install_exception_handlers,
install_error_logging_middleware,
InvalidCredentialsError,
ResourceNotFoundError,
)
# Create FastAPI app
app = FastAPI()
# Install error handlers and middleware
install_exception_handlers(app)
install_error_logging_middleware(app)
# Use exceptions in your routes
@app.get("/users/{user_id}")
async def get_user(user_id: str):
if user_id == "999":
raise ResourceNotFoundError(
resource_type="User",
resource_id=user_id
)
return {"user_id": user_id}
@app.post("/auth/login")
async def login(email: str, password: str):
if not validate_credentials(email, password):
raise InvalidCredentialsError()
return {"token": "..."}
Error Response Format
All errors return structured JSON responses:
{
"error": {
"code": "AUTH_INVALID_CREDENTIALS",
"message": "Invalid email or password",
"details": {},
"correlation_id": "req-20251125-143210-a8f3c9",
"timestamp": "2025-11-25T14:32:10.523Z",
"path": "/api/v1/auth/login"
}
}
Available Exceptions
Authentication (401 Unauthorized)
- InvalidCredentialsError: Invalid email/password combination
- TokenExpiredError: Authentication token has expired
- TokenInvalidError: Malformed or invalid token
- TokenRevokedError: Token has been explicitly revoked
- AuthenticationRequiredError: Endpoint requires authentication
from netrun.errors import InvalidCredentialsError, TokenExpiredError
# Example usage
raise InvalidCredentialsError()
raise TokenExpiredError(message="Your session has expired")
Authorization (403 Forbidden)
- InsufficientPermissionsError: User lacks required permissions
- TenantAccessDeniedError: Cross-tenant access attempt
from netrun.errors import InsufficientPermissionsError, TenantAccessDeniedError
# Example usage
raise InsufficientPermissionsError(
message="You need admin role to perform this action",
details={"required_role": "admin", "user_role": "member"}
)
raise TenantAccessDeniedError(
details={"user_tenant_id": "123", "resource_tenant_id": "456"}
)
Resource (404 Not Found, 409 Conflict)
- ResourceNotFoundError: Requested resource does not exist
- ResourceConflictError: Operation conflicts with existing state
from netrun.errors import ResourceNotFoundError, ResourceConflictError
# Example usage
raise ResourceNotFoundError(
resource_type="Product",
resource_id="SKU-12345"
)
raise ResourceConflictError(
message="Email address already registered",
details={"email": "user@example.com"}
)
Service (503 Service Unavailable)
- ServiceUnavailableError: Service or dependency unavailable
- TemporalUnavailableError: Workflow engine unavailable
from netrun.errors import ServiceUnavailableError, TemporalUnavailableError
# Example usage
raise ServiceUnavailableError(
message="Database connection failed",
details={"retry_after": 30}
)
raise TemporalUnavailableError(
message="Unable to start workflow",
details={"workflow_id": "user-onboarding-123"}
)
Correlation IDs
Every exception automatically generates a unique correlation ID for request tracking:
from netrun.errors import ResourceNotFoundError
try:
raise ResourceNotFoundError(resource_type="User", resource_id="123")
except ResourceNotFoundError as exc:
print(exc.correlation_id) # req-20251125-143210-a8f3c9
Correlation IDs are also:
- Injected into response headers as
X-Correlation-ID - Available in request state for logging:
request.state.correlation_id - Included in all log messages for distributed tracing
Custom Details
Add context-specific information to error responses:
from netrun.errors import InvalidCredentialsError
raise InvalidCredentialsError(
message="Invalid credentials",
details={
"attempt_number": 3,
"max_attempts": 5,
"lockout_duration": 300,
}
)
Global Exception Handlers
The install_exception_handlers() function registers handlers for:
- NetrunException: All custom Netrun exceptions
- RequestValidationError: FastAPI validation errors
- HTTPException: Generic HTTP exceptions
- Exception: Unhandled exceptions (500 errors)
All handlers return consistent JSON format with correlation IDs.
Error Logging Middleware
The install_error_logging_middleware() function provides:
- Request/response logging: Structured logs with correlation IDs
- Performance tracking: Duration measurement for all requests
- Automatic correlation ID injection: Available in
request.state.correlation_id - Exception logging: Full traceback logging for errors
from fastapi import FastAPI, Request
from netrun.errors import install_error_logging_middleware
import logging
app = FastAPI()
install_error_logging_middleware(app)
# Configure logging
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s - %(name)s - %(levelname)s - %(message)s'
)
@app.get("/example")
async def example(request: Request):
# Access correlation ID
correlation_id = request.state.correlation_id
logger.info(f"Processing request: {correlation_id}")
return {"status": "ok"}
Development
Setup
# Clone repository
git clone https://github.com/netrun-systems/netrun-errors.git
cd netrun-errors
# Install development dependencies
pip install -e ".[dev]"
Testing
# Run tests with coverage
pytest
# Run tests with verbose output
pytest -v
# Run specific test file
pytest tests/test_exceptions.py
# Generate HTML coverage report
pytest --cov-report=html
Code Quality
# Format code with black
black netrun/errors tests
# Lint with ruff
ruff check netrun/errors tests
# Type check with mypy
mypy netrun/errors
License
MIT License - see LICENSE file for details
Contributing
Contributions welcome! Please submit pull requests to the main repository.
Support
For issues, questions, or feature requests, please contact:
- Email: dev@netrunsystems.com
- Website: https://www.netrunsystems.com
Version: 2.0.0 Author: Netrun Systems Python: 3.11+ Framework: FastAPI 0.115+
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 netrun_errors-2.0.0.tar.gz.
File metadata
- Download URL: netrun_errors-2.0.0.tar.gz
- Upload date:
- Size: 68.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6a9bc1aebfb63e9a7d9db4652c4391437598d2a71001463cf5e587641817e0c8
|
|
| MD5 |
0ecf556f069b2487b149f48f99b8f568
|
|
| BLAKE2b-256 |
ffe30dbc666bfa8e2bda995e91ae721a76b13fb84bb85b8a7f0f9e7ef2efa66d
|
File details
Details for the file netrun_errors-2.0.0-py3-none-any.whl.
File metadata
- Download URL: netrun_errors-2.0.0-py3-none-any.whl
- Upload date:
- Size: 15.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
41fd56ab7f2d4025f2fdfe0d0a346a717bb689fde0c75d35a7349c007aa79011
|
|
| MD5 |
9e7c32d23cdcb87d9bbb4fcd65a31254
|
|
| BLAKE2b-256 |
e55ae949731b52d685c5d1ecdcf8df74bd266bd9126d4e3eaa87e97119a7ad91
|