feat: 配置写死与 baked 模块,Alembic 建表,百度仅 BAIDU_*

- 新增 app/baked/algorithm|pipeline,非部署参数不再走 env;Settings 保留 DB/HTTP/RTSP/海康/百度/MinIO/Demo
- 移除 init_db_schema 与 reload 配置;main 仅 check_database;start*.sh 在 uvicorn 前执行 alembic upgrade head
- 依赖 psycopg[binary] 供 Alembic 同步 URL;alembic/env 注释与预发清单更新
- 撕段门控消费管线、各视频/语音/归档调用改为 baked
- 百度环境变量仅 BAIDU_APP_ID、BAIDU_API_KEY、BAIDU_SECRET_KEY 与 BAIDU_* 超时/ASR;人脸脚本与 baidu_speech 文案同步
- 全量单测与 .env.example 更新;.gitignore 忽略 refs/(本地权重/视频不入库)

Made-with: Cursor
This commit is contained in:
Kevin
2026-04-24 15:33:22 +08:00
parent b651364877
commit 8a4bad99d3
47 changed files with 1333 additions and 648 deletions

5
app/baked/__init__.py Normal file
View File

@@ -0,0 +1,5 @@
"""编译进代码的默认参数(非部署差异);见 `algorithm` 与 `pipeline` 子模块。"""
from app.baked import algorithm, pipeline
__all__ = ["algorithm", "pipeline"]

63
app/baked/algorithm.py Normal file
View File

@@ -0,0 +1,63 @@
"""YOLO 耗材/手部与撕段四模型:路径与超参写死,与业务线部署解耦。"""
from __future__ import annotations
from pathlib import Path
_PACKAGE_DIR = Path(__file__).resolve().parent.parent
def default_consumable_classifier_weights_path() -> str:
return str(_PACKAGE_DIR / "resources" / "consumable_classifier.pt")
def default_consumable_classifier_labels_yaml_path() -> str:
return str(_PACKAGE_DIR / "resources" / "consumable_classifier_labels.yaml")
def default_camera_rtsp_urls_sample_path() -> str:
return str(_PACKAGE_DIR / "resources" / "camera_rtsp_urls.sample.json")
# --- 耗材分类YOLO-cls---
CONSUMABLE_CLASSIFIER_WEIGHTS: str = default_consumable_classifier_weights_path()
CONSUMABLE_CLASSIFIER_IMGSZ: int = 224
CONSUMABLE_CLASSIFIER_DEVICE: str = ""
CONSUMABLE_CLASSIFIER_TOPK: int = 5
CONSUMABLE_MIN_CLS_CONFIDENCE: float = 0.5
CONSUMABLE_CLASSIFIER_LABELS_YAML_PATH: str = default_consumable_classifier_labels_yaml_path()
CONSUMABLE_VISION_WINDOW_SEC: float = 15.0
# --- 手部检测;全空路径则整帧分类 ---
HAND_DETECTION_WEIGHTS: str = ""
HAND_DETECTION_IMGSZ: int = 640
HAND_DETECTION_CONF: float = 0.25
HAND_DETECTION_PAD_RATIO: float = 0.30
HAND_DETECTION_MIN_CROP_PX: int = 64
HAND_DETECTION_DEVICE: str = ""
# --- 撕段四模型(与 haocai_consumption demo 同构);无权重文件时勿开启 TEAR_SEGMENT_ENABLED ---
TEAR_SEGMENT_ENABLED: bool = False
TEAR_SEGMENT_PRIMARY_CAMERA_ID: str = ""
TEAR_SEGMENT_HAND_DET_WEIGHTS: str = ""
TEAR_SEGMENT_TEAR_WEIGHTS: str = ""
TEAR_SEGMENT_GOODBAD_WEIGHTS: str = ""
TEAR_SEGMENT_HAOCAI_WEIGHTS: str = ""
TEAR_SEGMENT_LABELS_YAML_PATH: str = ""
TEAR_SEGMENT_ASSUMED_FPS: float = 25.0
TEAR_SEGMENT_DET_CONF: float = 0.25
TEAR_SEGMENT_PAD_RATIO: float = 0.30
TEAR_SEGMENT_TEAR_CONF: float = 0.35
TEAR_SEGMENT_TEAR_SMOOTH: int = 5
TEAR_SEGMENT_GAP_RATIO: float = 1.5
TEAR_SEGMENT_MIN_TEAR_SEC: float = 1.2
TEAR_SEGMENT_MIN_GAP_SEC: float = 1.0
TEAR_SEGMENT_DET_IMGSZ: int = 640
TEAR_SEGMENT_TEAR_IMGSZ: int = 224
TEAR_SEGMENT_GOODBAD_IMGSZ: int = 224
TEAR_SEGMENT_HAOCAI_IMGSZ: int = 224
TEAR_SEGMENT_TEAR_DEVICE: str = ""
TEAR_SEGMENT_GOODBAD_DEVICE: str = ""
TEAR_SEGMENT_HAOCAI_DEVICE: str = ""
TEAR_SEGMENT_LOG_TXT: bool = False
TEAR_SEGMENT_LOG_TXT_PATH: str = "logs/tear_segment_{surgery_id}.txt"

39
app/baked/pipeline.py Normal file
View File

@@ -0,0 +1,39 @@
"""视频流、待确认、归档与可观测性默认可调参数(非 env"""
# --- 手术录制 API 重试 ---
SURGERY_RECORDING_MAX_ATTEMPTS: int = 3
SURGERY_RECORDING_RETRY_DELAY_SECONDS: float = 1.0
# --- RTSP 连接与抽帧、推理门控(不含 URLURL 在 Settings---
VIDEO_OPEN_TIMEOUT_SEC: float = 15.0
VIDEO_READ_FAILURE_RECONNECT_THRESHOLD: int = 15
VIDEO_RECONNECT_BACKOFF_SECONDS: float = 1.0
VIDEO_INFERENCE_INTERVAL_SEC: float = 2.0
VIDEO_INFERENCE_CONFIDENCE_THRESHOLD: float = 0.35
VIDEO_AUTO_CONFIRM_CONFIDENCE: float = 0.9
VIDEO_VOICE_CONFIRM_MIN_CONFIDENCE: float = 0.35
VOICE_CONFIRMATION_ENABLED: bool = True
VIDEO_VOICE_CONFIRM_DOCTOR_ID: str = "voice"
VIDEO_DETAIL_COOLDOWN_SEC: float = 15.0
VIDEO_JPEG_QUALITY: int = 85
VIDEO_RESULT_DOCTOR_ID: str = "vision"
VIDEO_LOG_INFERENCE_RESULTS: bool = False
# --- 停录后归档重试 + durable fallback ---
ARCHIVE_PERSIST_RETRY_INTERVAL_SECONDS: float = 30.0
ARCHIVE_PERSIST_MAX_RETRIES: int = 12
ARCHIVE_PERSIST_BACKOFF_CAP_SECONDS: float = 900.0
ARCHIVE_PERSIST_DURABLE_FALLBACK_DIR: str = "logs/pending_archive"
ARCHIVE_PERSIST_DURABLE_FALLBACK_ENABLED: bool = True
# --- 消耗/语音 文本日志与终端展示 ---
CONSUMPTION_TSV_LOG_ENABLED: bool = True
CONSUMPTION_TSV_LOG_PATH: str = "logs/consumption_{surgery_id}.txt"
CONSUMPTION_LOG_MARKDOWN_TERMINAL: bool = True
CONSUMPTION_LOG_TIMEZONE: str = ""
VOICE_FILE_LOG_ENABLED: bool = True
VOICE_FILE_LOG_PATH: str = "logs/voice_{surgery_id}.txt"
VOICE_UPLOAD_MAX_BYTES: int = 10 * 1024 * 1024
VOICE_CONFIRM_MAX_FAILED_PARSE_ROUNDS: int = 2