d7f712e3c1
Co-authored-by: multica-agent <github@multica.ai>
180 lines
5.9 KiB
Python
180 lines
5.9 KiB
Python
from functools import lru_cache
|
|
import re
|
|
from typing import Literal
|
|
from urllib.parse import quote_plus
|
|
|
|
from pydantic import field_validator
|
|
from pydantic_settings import BaseSettings, SettingsConfigDict
|
|
|
|
|
|
class Settings(BaseSettings):
|
|
api_name: str = "fquiz-api"
|
|
api_version: str = "0.1.0"
|
|
api_host: str = "0.0.0.0"
|
|
api_port: int = 8000
|
|
api_cors_origins: str = "http://localhost:3000,http://127.0.0.1:3000"
|
|
api_cors_origin_regex: str | None = None
|
|
|
|
database_url: str | None = None
|
|
db_host: str = "127.0.0.1"
|
|
db_port: int = 5434
|
|
db_name: str = "postgres"
|
|
db_schema: str = "public"
|
|
db_username: str = "fquiz"
|
|
db_password: str = "fquiz"
|
|
user_username_column: Literal["username", "user_name"] = "username"
|
|
user_password_column: Literal["password", "password_hash"] = "password_hash"
|
|
user_status_column: Literal["status", "state"] = "status"
|
|
file_vfs_root: str = "./data/vfs"
|
|
minio_enabled: bool = True
|
|
minio_endpoint: str = "http://minio:9000"
|
|
minio_access_key: str = "minioadmin"
|
|
minio_secret_key: str = "minioadmin"
|
|
minio_bucket: str = "fquiz-files"
|
|
minio_region: str = "us-east-1"
|
|
|
|
jwt_secret_key: str = "change-this-in-production"
|
|
jwt_algorithm: str = "HS256"
|
|
access_token_expire_minutes: int = 480
|
|
refresh_token_expire_days: int = 30
|
|
|
|
refresh_cookie_name: str = "refresh_token"
|
|
refresh_cookie_secure: bool = False
|
|
refresh_cookie_samesite: Literal["lax", "strict", "none"] = "lax"
|
|
|
|
celery_broker_url: str | None = None
|
|
celery_result_backend: str | None = None
|
|
celery_timezone: str = "Asia/Shanghai"
|
|
scheduler_expire_interval_seconds: int = 60
|
|
flower_api_base_url: str = "http://flower:5555"
|
|
flower_api_timeout_seconds: int = 10
|
|
flower_basic_auth: str = ""
|
|
worker_registry_ttl_seconds: int = 90
|
|
|
|
wine_binary_path: str = "wine"
|
|
wine_allowed_root: str = "./data/wine"
|
|
wine_default_timeout_seconds: int = 300
|
|
wine_max_timeout_seconds: int = 1800
|
|
|
|
atp_engine_mode: Literal["wine", "native"] = "wine"
|
|
atp_engine_executable: str = "atp/tpbig.exe"
|
|
atp_storage_root: str = "./data/wine/atp-models"
|
|
atp_engine_workdir: str = "runs"
|
|
atp_engine_default_timeout_seconds: int = 600
|
|
atp_engine_max_timeout_seconds: int = 3600
|
|
atp_legacy_root: str = "./data/wine/ATP"
|
|
atp_tpbig_executable: str = "ATP/tpbig.exe"
|
|
atp_rjtzl_executable: str = "ATP/rjtzl.exe"
|
|
atp_template_root: str = "./data/wine/ATP/templates"
|
|
atp_run_root: str = "./data/wine/atp-runs"
|
|
atp_egm_subdir: str = "EGM"
|
|
|
|
initial_admin_email: str | None = None
|
|
initial_admin_user_id: str = "admin"
|
|
initial_admin_username: str = "管理员"
|
|
initial_admin_password: str | None = None
|
|
|
|
model_config = SettingsConfigDict(
|
|
env_file=".env",
|
|
env_file_encoding="utf-8",
|
|
case_sensitive=False,
|
|
extra="ignore",
|
|
)
|
|
|
|
@field_validator(
|
|
"access_token_expire_minutes",
|
|
"refresh_token_expire_days",
|
|
"db_port",
|
|
"scheduler_expire_interval_seconds",
|
|
"flower_api_timeout_seconds",
|
|
"worker_registry_ttl_seconds",
|
|
"wine_default_timeout_seconds",
|
|
"wine_max_timeout_seconds",
|
|
"atp_engine_default_timeout_seconds",
|
|
"atp_engine_max_timeout_seconds",
|
|
)
|
|
@classmethod
|
|
def validate_positive_numbers(cls, value: int) -> int:
|
|
if value <= 0:
|
|
msg = "Value must be greater than 0"
|
|
raise ValueError(msg)
|
|
return value
|
|
|
|
@property
|
|
def cors_origins(self) -> list[str]:
|
|
origins: list[str] = []
|
|
for origin in self.api_cors_origins.split(","):
|
|
normalized = origin.strip()
|
|
if not normalized:
|
|
continue
|
|
if normalized == "*" or "*" in normalized:
|
|
continue
|
|
origins.append(normalized)
|
|
return origins
|
|
|
|
@property
|
|
def cors_origin_regex(self) -> str | None:
|
|
regex_parts: list[str] = []
|
|
for origin in self.api_cors_origins.split(","):
|
|
normalized = origin.strip()
|
|
if not normalized:
|
|
continue
|
|
if normalized == "*":
|
|
regex_parts.append(".*")
|
|
continue
|
|
if "*" in normalized:
|
|
wildcard_regex = re.escape(normalized).replace(r"\*", ".*")
|
|
regex_parts.append(f"^{wildcard_regex}$")
|
|
|
|
if self.api_cors_origin_regex:
|
|
normalized = self.api_cors_origin_regex.strip()
|
|
if normalized:
|
|
regex_parts.append(normalized)
|
|
|
|
if not regex_parts:
|
|
return None
|
|
return "|".join(f"(?:{part})" for part in regex_parts)
|
|
|
|
@property
|
|
def resolved_database_url(self) -> str:
|
|
explicit_database_url = (self.database_url or "").strip()
|
|
if explicit_database_url:
|
|
return explicit_database_url
|
|
|
|
username = quote_plus(self.db_username.strip())
|
|
password = quote_plus(self.db_password.strip())
|
|
host = self.db_host.strip()
|
|
database = self.db_name.strip()
|
|
return f"postgresql+psycopg://{username}:{password}@{host}:{self.db_port}/{database}"
|
|
|
|
@property
|
|
def resolved_db_schema(self) -> str | None:
|
|
schema = self.db_schema.strip()
|
|
if not schema:
|
|
return None
|
|
return schema
|
|
|
|
@property
|
|
def resolved_celery_broker_url(self) -> str:
|
|
return (self.celery_broker_url or "redis://redis:6379/0").strip()
|
|
|
|
@property
|
|
def resolved_celery_result_backend(self) -> str:
|
|
return (self.celery_result_backend or "redis://redis:6379/1").strip()
|
|
|
|
@property
|
|
def resolved_flower_api_base_url(self) -> str:
|
|
value = (self.flower_api_base_url or "").strip().rstrip("/")
|
|
if not value:
|
|
return "http://flower:5555"
|
|
return value
|
|
|
|
@property
|
|
def resolved_flower_basic_auth(self) -> str:
|
|
return (self.flower_basic_auth or "").strip()
|
|
|
|
|
|
@lru_cache
|
|
def get_settings() -> Settings:
|
|
return Settings()
|