mirror of
https://github.com/JefferyHcool/BiliNote.git
synced 2026-06-12 19:20:00 +08:00
fix(backend): 部署韧性——模型自愈/就绪门禁/全局代理/启动诊断
- whisper: model.bin 截断/损坏时删目录重下重试一次,修「Unable to
open file model.bin」死循环;mlx 同样按 config.json 判完整性
- /generate_note 加就绪门禁:本地转写引擎模型没下好直接拦截,返回
reason=transcriber_model_not_ready,不让任务静默卡在首次下载
- 全局代理:新增 ProxyConfigManager(JSON 配置 + HTTP_PROXY env 兜底)
+ build_openai_client,统一注入代理到 LLM/Groq 客户端;yt-dlp 与
youtube-transcript-api 也走代理
- build_openai_client 校验 api_key 非空,空 key 给「xxx 的 API Key
未配置」而不是天书般的 Illegal header value b'Bearer '
- universal_gpt: 模型拒绝自定义 temperature(o1/o3/gpt-5 系列)时
就地去掉参数重试,不消耗重试预算
- connect_test 改用真实 chat completion 而非 /v1/models 探测
- main.py: lifespan 拆 [startup 1/5..5/5] 分段日志 + 异常清晰定位
- /sys_health 重构为结构化返回 {backend,ffmpeg,db,whisper_model}
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -39,24 +39,50 @@ if not os.path.exists(out_dir):
|
||||
|
||||
@asynccontextmanager
|
||||
async def lifespan(app: FastAPI):
|
||||
register_handler()
|
||||
init_db()
|
||||
# 转写器不再在启动时强制初始化,而是在首次生成笔记时按需创建
|
||||
# 如果配置了不可用的类型(如 mlx-whisper 未安装),会在使用时报错而非静默回退
|
||||
_cfg = TranscriberConfigManager().get_config()
|
||||
logger.info(f"当前转写器配置: type={_cfg['transcriber_type']}, model_size={_cfg['whisper_model_size']}")
|
||||
seed_default_providers()
|
||||
# 启动序列拆成 5 步、每步独立日志 + 异常时打明确的 [startup N/5 FAILED] 标记。
|
||||
# 目的:用户 docker logs 一眼能看出后端死在哪一步,避免「容器一直重启但看不出原因」。
|
||||
try:
|
||||
logger.info("[startup 1/5] register_handler() — 注册事件处理器")
|
||||
register_handler()
|
||||
|
||||
logger.info("[startup 2/5] init_db() — 初始化 SQLite 数据库")
|
||||
init_db()
|
||||
|
||||
logger.info("[startup 3/5] TranscriberConfigManager — 读取转写器配置")
|
||||
# 转写器不再在启动时强制初始化,而是在首次生成笔记时按需创建。
|
||||
# 如果配置了不可用的类型(如 mlx-whisper 未安装),会在使用时报错而非静默回退。
|
||||
_cfg = TranscriberConfigManager().get_config()
|
||||
logger.info(
|
||||
f" 当前转写器: type={_cfg['transcriber_type']}, "
|
||||
f"model_size={_cfg['whisper_model_size']}"
|
||||
)
|
||||
|
||||
logger.info("[startup 4/5] seed_default_providers() — 初始化默认 LLM 供应商")
|
||||
seed_default_providers()
|
||||
|
||||
logger.info("[startup 5/5] 启动完成,等待请求")
|
||||
except Exception:
|
||||
logger.exception("[startup FAILED] 后端启动期异常,详见堆栈;容器会退出并由 restart 策略决定是否重试")
|
||||
raise
|
||||
|
||||
yield
|
||||
|
||||
app = create_app(lifespan=lifespan)
|
||||
|
||||
# 允许的源:本地 web 端 + Tauri 桌面端 + 浏览器扩展(chrome/edge/firefox)
|
||||
# 用 regex 是因为 chrome-extension://<id> 的 id 在每次开发版加载时不固定
|
||||
# Tauri 2 不同平台 webview origin 不一样,必须全列:
|
||||
# - macOS: tauri://localhost (自定义协议)
|
||||
# - Windows: https://tauri.localhost (Edge WebView2)
|
||||
# - Linux: http://tauri.localhost (WebKitGTK)
|
||||
# 漏掉哪个都会导致桌面端 fetch 返回 200 但 browser 因为 CORS 拒绝读响应,
|
||||
# 表现为前端「连不上后端」但后端日志一片 200 OK。
|
||||
CORS_ORIGIN_REGEX = (
|
||||
r"^chrome-extension://[a-z]+$"
|
||||
r"|^moz-extension://.+$"
|
||||
r"|^http://(localhost|127\.0\.0\.1)(:\d+)?$"
|
||||
r"|^http://tauri\.localhost$"
|
||||
r"|^tauri://localhost$"
|
||||
r"|^https?://tauri\.localhost$"
|
||||
)
|
||||
|
||||
app.add_middleware(
|
||||
|
||||
Reference in New Issue
Block a user