2026-05-11 05:18:16 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
from collections.abc import AsyncIterator
|
|
|
|
|
from contextlib import asynccontextmanager
|
|
|
|
|
|
|
|
|
|
from fastapi import FastAPI
|
|
|
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
|
|
|
|
2026-05-11 03:51:24 +00:00
|
|
|
from app.api.router import api_router
|
|
|
|
|
from app.core.config import get_settings
|
|
|
|
|
from app.core.logging import get_logger, setup_logging
|
2026-05-11 05:18:16 +00:00
|
|
|
from app.core.openapi import API_DESCRIPTION, OPENAPI_TAGS
|
2026-05-11 03:51:24 +00:00
|
|
|
from app.middleware.logging import AccessLogMiddleware
|
2026-05-11 05:18:16 +00:00
|
|
|
from app.schemas.common import RootStatusRead
|
2026-05-11 03:51:24 +00:00
|
|
|
from app.services.agent_foundation import prepare_agent_foundation
|
2026-06-03 09:25:23 +08:00
|
|
|
from app.services.digital_employee_reminder_scheduler import digital_employee_reminder_scheduler
|
2026-05-11 03:51:24 +00:00
|
|
|
from app.services.employee import prepare_employee_directory
|
2026-06-03 09:25:23 +08:00
|
|
|
from app.services.employee_profile_scheduler import employee_profile_scheduler
|
|
|
|
|
from app.services.finance_dashboard_scheduler import finance_dashboard_scheduler
|
|
|
|
|
from app.services.finance_report_scheduler import finance_report_scheduler
|
2026-05-16 06:14:08 +00:00
|
|
|
from app.services.hermes_sync import sync_repository_hermes_skills
|
2026-05-17 08:38:41 +00:00
|
|
|
from app.services.knowledge import prepare_knowledge_library
|
|
|
|
|
from app.services.knowledge_index_tasks import knowledge_index_task_manager
|
|
|
|
|
from app.services.knowledge_rag import shutdown_knowledge_rag_runtime
|
|
|
|
|
from app.services.knowledge_scheduler import knowledge_index_scheduler
|
2026-05-11 05:18:16 +00:00
|
|
|
|
|
|
|
|
|
2026-06-06 17:19:07 +08:00
|
|
|
def _effective_server_workers(settings: object) -> int:
|
|
|
|
|
server_workers = getattr(settings, "server_workers", None)
|
|
|
|
|
web_concurrency = getattr(settings, "web_concurrency", None)
|
|
|
|
|
workers = web_concurrency if int(server_workers or 1) <= 1 and web_concurrency else server_workers
|
|
|
|
|
try:
|
|
|
|
|
return max(1, int(workers or 1))
|
|
|
|
|
except (TypeError, ValueError):
|
|
|
|
|
return 1
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _should_start_background_schedulers(settings: object) -> bool:
|
|
|
|
|
if not bool(getattr(settings, "background_schedulers_enabled", True)):
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
return _effective_server_workers(settings) <= 1
|
|
|
|
|
|
|
|
|
|
|
2026-05-11 05:18:16 +00:00
|
|
|
@asynccontextmanager
|
|
|
|
|
async def lifespan(_: FastAPI) -> AsyncIterator[None]:
|
|
|
|
|
settings = get_settings()
|
|
|
|
|
logger = get_logger("app.main")
|
|
|
|
|
|
|
|
|
|
prepare_employee_directory()
|
|
|
|
|
prepare_agent_foundation()
|
|
|
|
|
prepare_knowledge_library()
|
2026-05-16 06:14:08 +00:00
|
|
|
sync_repository_hermes_skills()
|
2026-06-06 17:19:07 +08:00
|
|
|
schedulers_started = _should_start_background_schedulers(settings)
|
|
|
|
|
if schedulers_started:
|
|
|
|
|
knowledge_index_scheduler.start()
|
|
|
|
|
finance_dashboard_scheduler.start()
|
|
|
|
|
employee_profile_scheduler.start()
|
|
|
|
|
digital_employee_reminder_scheduler.start()
|
|
|
|
|
finance_report_scheduler.start()
|
|
|
|
|
else:
|
|
|
|
|
logger.warning(
|
|
|
|
|
"Background schedulers skipped - workers=%s enabled=%s",
|
|
|
|
|
_effective_server_workers(settings),
|
|
|
|
|
settings.background_schedulers_enabled,
|
|
|
|
|
)
|
2026-05-11 05:18:16 +00:00
|
|
|
logger.info(
|
|
|
|
|
"Server ready - host=%s port=%s prefix=%s",
|
|
|
|
|
settings.app_host,
|
|
|
|
|
settings.app_port,
|
|
|
|
|
settings.api_v1_prefix,
|
|
|
|
|
)
|
|
|
|
|
yield
|
2026-06-06 17:19:07 +08:00
|
|
|
if schedulers_started:
|
|
|
|
|
finance_report_scheduler.shutdown()
|
|
|
|
|
digital_employee_reminder_scheduler.shutdown()
|
|
|
|
|
employee_profile_scheduler.shutdown()
|
|
|
|
|
finance_dashboard_scheduler.shutdown()
|
|
|
|
|
knowledge_index_scheduler.shutdown()
|
2026-05-17 08:38:41 +00:00
|
|
|
knowledge_index_task_manager.shutdown()
|
|
|
|
|
shutdown_knowledge_rag_runtime()
|
2026-05-11 05:18:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def create_app() -> FastAPI:
|
|
|
|
|
settings = get_settings()
|
2026-05-17 08:38:41 +00:00
|
|
|
|
|
|
|
|
setup_logging(
|
|
|
|
|
level=settings.log_level,
|
|
|
|
|
log_dir=settings.log_dir,
|
|
|
|
|
enable_file=settings.log_file_enabled,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
logger = get_logger("app.main")
|
|
|
|
|
logger.info(
|
|
|
|
|
"Starting %s (env=%s, debug=%s)", settings.app_name, settings.app_env, settings.app_debug
|
|
|
|
|
)
|
|
|
|
|
|
2026-05-11 05:18:16 +00:00
|
|
|
app = FastAPI(
|
|
|
|
|
title=settings.app_name,
|
|
|
|
|
debug=settings.app_debug,
|
|
|
|
|
version="0.1.0",
|
|
|
|
|
description=API_DESCRIPTION,
|
|
|
|
|
openapi_tags=OPENAPI_TAGS,
|
|
|
|
|
lifespan=lifespan,
|
|
|
|
|
)
|
2026-05-17 08:38:41 +00:00
|
|
|
|
|
|
|
|
app.add_middleware(AccessLogMiddleware)
|
|
|
|
|
|
|
|
|
|
if settings.cors_origins:
|
|
|
|
|
app.add_middleware(
|
|
|
|
|
CORSMiddleware,
|
|
|
|
|
allow_origins=settings.cors_origins,
|
|
|
|
|
allow_credentials=True,
|
|
|
|
|
allow_methods=["*"],
|
|
|
|
|
allow_headers=["*"],
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
app.include_router(api_router, prefix=settings.api_v1_prefix)
|
|
|
|
|
|
2026-05-11 05:18:16 +00:00
|
|
|
@app.get(
|
|
|
|
|
"/",
|
|
|
|
|
tags=["root"],
|
|
|
|
|
response_model=RootStatusRead,
|
|
|
|
|
summary="服务根检查",
|
2026-05-17 08:38:41 +00:00
|
|
|
description="用于快速确认后端服务进程已启动。",
|
2026-05-11 05:18:16 +00:00
|
|
|
)
|
|
|
|
|
def root() -> RootStatusRead:
|
|
|
|
|
return RootStatusRead(message=f"{settings.app_name} is running")
|
|
|
|
|
|
|
|
|
|
return app
|
2026-05-17 08:38:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
app = create_app()
|