Skip to main content

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:


Version: 2.0.0 Author: Netrun Systems Python: 3.11+ Framework: FastAPI 0.115+

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

netrun_errors-2.0.0.tar.gz (68.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

netrun_errors-2.0.0-py3-none-any.whl (15.4 kB view details)

Uploaded Python 3

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

Hashes for netrun_errors-2.0.0.tar.gz
Algorithm Hash digest
SHA256 6a9bc1aebfb63e9a7d9db4652c4391437598d2a71001463cf5e587641817e0c8
MD5 0ecf556f069b2487b149f48f99b8f568
BLAKE2b-256 ffe30dbc666bfa8e2bda995e91ae721a76b13fb84bb85b8a7f0f9e7ef2efa66d

See more details on using hashes here.

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

Hashes for netrun_errors-2.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 41fd56ab7f2d4025f2fdfe0d0a346a717bb689fde0c75d35a7349c007aa79011
MD5 9e7c32d23cdcb87d9bbb4fcd65a31254
BLAKE2b-256 e55ae949731b52d685c5d1ecdcf8df74bd266bd9126d4e3eaa87e97119a7ad91

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page