2026-02-05 13:56:39 +08:00
|
|
|
|
"""
|
|
|
|
|
|
配置管理
|
|
|
|
|
|
"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
import os
|
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
from typing import Optional
|
|
|
|
|
|
from dataclasses import dataclass
|
|
|
|
|
|
from pydantic import BaseModel
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
|
|
class DatabaseConfig:
|
2026-02-05 13:56:39 +08:00
|
|
|
|
"""数据库配置"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
url: str = "sqlite:///./data/alert_platform.db"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
2026-02-09 17:47:35 +08:00
|
|
|
|
class COSConfig:
|
2026-03-02 13:33:28 +08:00
|
|
|
|
"""腾讯云 COS 存储配置(通过 CVM 角色认证,无需密钥)"""
|
2026-02-09 17:47:35 +08:00
|
|
|
|
region: str = "ap-beijing"
|
|
|
|
|
|
bucket: str = "" # 格式: bucketname-appid
|
|
|
|
|
|
upload_prefix: str = "alerts" # 对象 Key 前缀
|
|
|
|
|
|
presign_expire: int = 1800 # 预签名URL有效期(秒),默认30分钟
|
2026-03-02 13:33:28 +08:00
|
|
|
|
enabled: bool = False # 是否启用 COS
|
2026-02-02 09:40:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
|
|
class AppConfig:
|
2026-02-05 13:56:39 +08:00
|
|
|
|
"""应用配置"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
host: str = "0.0.0.0"
|
|
|
|
|
|
port: int = 8000
|
|
|
|
|
|
debug: bool = True
|
2026-02-06 16:39:53 +08:00
|
|
|
|
dev_mode: bool = True # 开发模式:跳过认证验证,返回超级管理员权限
|
2026-02-02 09:40:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
|
|
class AIModelConfig:
|
2026-02-05 13:56:39 +08:00
|
|
|
|
"""AI 模型配置"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
endpoint: str = ""
|
|
|
|
|
|
api_key: str = ""
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-02-10 15:22:01 +08:00
|
|
|
|
@dataclass
|
|
|
|
|
|
class RedisConfig:
|
|
|
|
|
|
"""Redis 配置"""
|
|
|
|
|
|
host: str = "localhost"
|
|
|
|
|
|
port: int = 6379
|
|
|
|
|
|
password: str = ""
|
|
|
|
|
|
db: int = 0
|
|
|
|
|
|
max_connections: int = 50
|
|
|
|
|
|
decode_responses: bool = True
|
2026-02-05 13:56:39 +08:00
|
|
|
|
enabled: bool = True
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-02-24 13:59:13 +08:00
|
|
|
|
@dataclass
|
|
|
|
|
|
class CameraNameConfig:
|
|
|
|
|
|
"""摄像头名称格式化配置"""
|
|
|
|
|
|
# WVP API基础URL
|
|
|
|
|
|
wvp_api_base: str = "http://localhost:18080"
|
|
|
|
|
|
|
|
|
|
|
|
# 显示格式模板(支持变量:{camera_code}, {name}, {stream})
|
|
|
|
|
|
# 可选格式:
|
2026-02-24 14:08:36 +08:00
|
|
|
|
# - "{name}" - 仅名称(推荐,告警列表使用)
|
|
|
|
|
|
# - "{camera_code} {name}/{stream}" - cam_xxx 名称/流id(完整格式)
|
2026-02-24 13:59:13 +08:00
|
|
|
|
# - "{name}/{stream}" - 名称/流id
|
|
|
|
|
|
# - "{camera_code}" - 仅code
|
2026-02-24 14:08:36 +08:00
|
|
|
|
display_format: str = "{name}"
|
2026-02-24 13:59:13 +08:00
|
|
|
|
|
|
|
|
|
|
# 名称字段优先级(从高到低)
|
|
|
|
|
|
# 从StreamProxy对象中提取名称时的字段优先级
|
|
|
|
|
|
# 注意:gb_name 可能包含 "/" 后缀,会自动去除
|
|
|
|
|
|
name_field_priority: list = None
|
|
|
|
|
|
|
|
|
|
|
|
# 查询超时(秒)
|
2026-03-02 13:33:28 +08:00
|
|
|
|
query_timeout: int = 15
|
2026-02-24 13:59:13 +08:00
|
|
|
|
|
|
|
|
|
|
def __post_init__(self):
|
|
|
|
|
|
if self.name_field_priority is None:
|
|
|
|
|
|
# 默认优先级:gb_name > app > stream
|
|
|
|
|
|
self.name_field_priority = ["gbName", "app", "stream"]
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-02-02 09:40:02 +08:00
|
|
|
|
class Settings(BaseModel):
|
2026-02-05 13:56:39 +08:00
|
|
|
|
"""全局配置"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
database: DatabaseConfig = DatabaseConfig()
|
2026-02-09 17:47:35 +08:00
|
|
|
|
cos: COSConfig = COSConfig()
|
2026-02-02 09:40:02 +08:00
|
|
|
|
app: AppConfig = AppConfig()
|
|
|
|
|
|
ai_model: AIModelConfig = AIModelConfig()
|
2026-02-10 15:22:01 +08:00
|
|
|
|
redis: RedisConfig = RedisConfig()
|
2026-02-24 13:59:13 +08:00
|
|
|
|
camera_name: CameraNameConfig = CameraNameConfig()
|
2026-02-02 09:40:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def load_settings() -> Settings:
|
2026-02-05 13:56:39 +08:00
|
|
|
|
"""从环境变量加载配置"""
|
2026-02-02 09:40:02 +08:00
|
|
|
|
from dotenv import load_dotenv
|
|
|
|
|
|
load_dotenv()
|
|
|
|
|
|
|
|
|
|
|
|
return Settings(
|
|
|
|
|
|
database=DatabaseConfig(
|
|
|
|
|
|
url=os.getenv("DATABASE_URL", "sqlite:///./data/alert_platform.db"),
|
|
|
|
|
|
),
|
2026-02-09 17:47:35 +08:00
|
|
|
|
cos=COSConfig(
|
|
|
|
|
|
region=os.getenv("COS_REGION", "ap-beijing"),
|
|
|
|
|
|
bucket=os.getenv("COS_BUCKET", ""),
|
|
|
|
|
|
upload_prefix=os.getenv("COS_UPLOAD_PREFIX", "alerts"),
|
|
|
|
|
|
presign_expire=int(os.getenv("COS_PRESIGN_EXPIRE", "1800")),
|
|
|
|
|
|
enabled=os.getenv("COS_ENABLED", "false").lower() == "true",
|
2026-02-02 09:40:02 +08:00
|
|
|
|
),
|
|
|
|
|
|
app=AppConfig(
|
|
|
|
|
|
host=os.getenv("APP_HOST", "0.0.0.0"),
|
|
|
|
|
|
port=int(os.getenv("APP_PORT", "8000")),
|
|
|
|
|
|
debug=os.getenv("DEBUG", "true").lower() == "true",
|
2026-02-06 16:39:53 +08:00
|
|
|
|
dev_mode=os.getenv("DEV_MODE", "true").lower() == "true",
|
2026-02-02 09:40:02 +08:00
|
|
|
|
),
|
|
|
|
|
|
ai_model=AIModelConfig(
|
|
|
|
|
|
endpoint=os.getenv("AI_MODEL_ENDPOINT", ""),
|
|
|
|
|
|
api_key=os.getenv("AI_MODEL_API_KEY", ""),
|
|
|
|
|
|
),
|
2026-02-10 15:22:01 +08:00
|
|
|
|
redis=RedisConfig(
|
|
|
|
|
|
host=os.getenv("REDIS_HOST", "localhost"),
|
|
|
|
|
|
port=int(os.getenv("REDIS_PORT", "6379")),
|
|
|
|
|
|
password=os.getenv("REDIS_PASSWORD", ""),
|
|
|
|
|
|
db=int(os.getenv("REDIS_DB", "0")),
|
|
|
|
|
|
max_connections=int(os.getenv("REDIS_MAX_CONNECTIONS", "50")),
|
|
|
|
|
|
enabled=os.getenv("REDIS_ENABLED", "true").lower() == "true",
|
2026-02-05 13:56:39 +08:00
|
|
|
|
),
|
2026-02-24 13:59:13 +08:00
|
|
|
|
camera_name=CameraNameConfig(
|
|
|
|
|
|
wvp_api_base=os.getenv("WVP_API_BASE", "http://localhost:18080"),
|
2026-02-24 14:08:36 +08:00
|
|
|
|
display_format=os.getenv("CAMERA_NAME_FORMAT", "{name}"),
|
2026-03-02 13:33:28 +08:00
|
|
|
|
query_timeout=int(os.getenv("CAMERA_QUERY_TIMEOUT", "15")),
|
2026-02-24 13:59:13 +08:00
|
|
|
|
),
|
2026-02-02 09:40:02 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
settings = load_settings()
|