hxf/backend/th_agenter/core/app.py

177 lines
5.4 KiB
Python

"""FastAPI application factory."""
import logging
from contextlib import asynccontextmanager
from fastapi import FastAPI
from fastapi.middleware.cors import CORSMiddleware
from fastapi.middleware.trustedhost import TrustedHostMiddleware
from fastapi.responses import JSONResponse
from fastapi.exceptions import RequestValidationError
from starlette.exceptions import HTTPException as StarletteHTTPException
from .config import Settings
from .logging import setup_logging
from .middleware import UserContextMiddleware
from ..api.routes import router
from ..db.database import init_db
from ..api.endpoints import table_metadata
@asynccontextmanager
async def lifespan(app: FastAPI):
"""Application lifespan manager."""
# Startup
logging.info("Starting up TH-Agenter application...")
await init_db()
logging.info("Database initialized")
yield
# Shutdown
logging.info("Shutting down TH-Agenter application...")
def create_app(settings: Settings = None) -> FastAPI:
"""Create and configure FastAPI application."""
if settings is None:
from .config import get_settings
settings = get_settings()
# Setup logging
setup_logging(settings.logging)
# Create FastAPI app
app = FastAPI(
title=settings.app_name,
version=settings.app_version,
description="A modern chat agent application with Vue frontend and FastAPI backend",
debug=settings.debug,
lifespan=lifespan,
)
# Add middleware
setup_middleware(app, settings)
# Add exception handlers
setup_exception_handlers(app)
# Include routers
app.include_router(router, prefix="/api")
app.include_router(table_metadata.router)
# 在现有导入中添加
from ..api.endpoints import database_config
# 在路由注册部分添加
app.include_router(database_config.router)
# Health check endpoint
@app.get("/health")
async def health_check():
return {"status": "healthy", "version": settings.app_version}
# Root endpoint
@app.get("/")
async def root():
return {"message": "Chat Agent API is running"}
# Test endpoint
@app.get("/test")
async def test_endpoint():
return {"message": "API is working"}
return app
def setup_middleware(app: FastAPI, settings: Settings) -> None:
"""Setup application middleware."""
# User context middleware (should be first to set context for all requests)
app.add_middleware(UserContextMiddleware)
# CORS middleware
app.add_middleware(
CORSMiddleware,
allow_origins=settings.cors.allowed_origins,
allow_credentials=True,
allow_methods=settings.cors.allowed_methods,
allow_headers=settings.cors.allowed_headers,
)
# Trusted host middleware (for production)
if settings.environment == "production":
app.add_middleware(
TrustedHostMiddleware,
allowed_hosts=["*"] # Configure this properly in production
)
def setup_exception_handlers(app: FastAPI) -> None:
"""Setup global exception handlers."""
@app.exception_handler(StarletteHTTPException)
async def http_exception_handler(request, exc):
return JSONResponse(
status_code=exc.status_code,
content={
"error": {
"type": "http_error",
"message": exc.detail,
"status_code": exc.status_code
}
}
)
def make_json_serializable(obj):
"""递归地将对象转换为JSON可序列化的格式"""
if obj is None or isinstance(obj, (str, int, float, bool)):
return obj
elif isinstance(obj, bytes):
return obj.decode('utf-8')
elif isinstance(obj, (ValueError, Exception)):
return str(obj)
elif isinstance(obj, dict):
return {k: make_json_serializable(v) for k, v in obj.items()}
elif isinstance(obj, (list, tuple)):
return [make_json_serializable(item) for item in obj]
else:
# For any other object, convert to string
return str(obj)
@app.exception_handler(RequestValidationError)
async def validation_exception_handler(request, exc):
# Convert any non-serializable objects to strings in error details
try:
errors = make_json_serializable(exc.errors())
except Exception as e:
# Fallback: if even our conversion fails, use a simple error message
errors = [{"type": "serialization_error", "msg": f"Error processing validation details: {str(e)}"}]
return JSONResponse(
status_code=422,
content={
"error": {
"type": "validation_error",
"message": "Request validation failed",
"details": errors
}
}
)
@app.exception_handler(Exception)
async def general_exception_handler(request, exc):
logging.error(f"Unhandled exception: {exc}", exc_info=True)
return JSONResponse(
status_code=500,
content={
"error": {
"type": "internal_error",
"message": "Internal server error"
}
}
)
# Create the app instance
app = create_app()