config.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. from pydantic_settings import BaseSettings, SettingsConfigDict
  2. from typing import Any, Dict, List
  3. from dotenv import load_dotenv
  4. from urllib.parse import quote
  5. import os
  6. load_dotenv()
  7. class Settings(BaseSettings):
  8. # 基础配置
  9. PROJECT_NAME: str = "Alien Cloud Python"
  10. API_V1_STR: str = "/api/v1"
  11. # 鉴权配置 (原 alien-gateway 职责)
  12. SECRET_KEY: str = os.getenv("SECRET_KEY")
  13. ALGORITHM: str = os.getenv("ALGORITHM")
  14. ACCESS_TOKEN_EXPIRE_MINUTES: int = os.getenv("ACCESS_TOKEN_EXPIRE_MINUTES") # 7天
  15. # 数据库配置
  16. DB_USER: str = os.getenv("DB_USER")
  17. DB_PASSWORD: str = os.getenv("DB_PASSWORD")
  18. DB_HOST: str = os.getenv("DB_HOST")
  19. DB_PORT: int = os.getenv("DB_PORT")
  20. DB_NAME: str = os.getenv("DB_NAME")
  21. # Redis Sentinel 高可用配置
  22. # 例: REDIS_SENTINELS=192.168.2.251:36379,192.168.2.252:36379,192.168.2.253:36379
  23. REDIS_SENTINELS: str = os.getenv("REDIS_SENTINELS", "")
  24. REDIS_MASTER_NAME: str = os.getenv("REDIS_MASTER_NAME", "mymaster")
  25. REDIS_PASSWORD: str = os.getenv("REDIS_PASSWORD", "")
  26. REDIS_DB: int = int(os.getenv("REDIS_DB", "0"))
  27. REDIS_SENTINEL_USERNAME: str = os.getenv("REDIS_SENTINEL_USERNAME", "")
  28. REDIS_SENTINEL_PASSWORD: str = os.getenv("REDIS_SENTINEL_PASSWORD", "")
  29. REDIS_SOCKET_TIMEOUT: float = float(os.getenv("REDIS_SOCKET_TIMEOUT", "0.5"))
  30. REDIS_CONNECT_TIMEOUT: float = float(os.getenv("REDIS_CONNECT_TIMEOUT", "1.0"))
  31. # 下游服务地址
  32. STORE_BASE_URL: str = os.getenv("STORE_BASE_URL") # alien_store 服务地址
  33. # 阿里云短信配置
  34. ALIYUN_SMS_SIGN_NAME_CONTRACT: str = os.getenv("ALIYUN_SMS_SIGN_NAME_CONTRACT")
  35. ALIYUN_SMS_TEMPLATE_CODE_CONTRACT: str = os.getenv("ALIYUN_SMS_TEMPLATE_CODE_CONTRACT")
  36. ALIYUN_ACCESS_KEY_ID: str = os.getenv("ALIYUN_ACCESS_KEY_ID")
  37. ALIYUN_ACCESS_KEY_SECRET: str = os.getenv("ALIYUN_ACCESS_KEY_SECRET")
  38. @property
  39. def SQLALCHEMY_DATABASE_URI(self) -> str:
  40. return f"mysql+pymysql://{self.DB_USER}:{self.DB_PASSWORD}@{self.DB_HOST}:{self.DB_PORT}/{self.DB_NAME}"
  41. @property
  42. def REDIS_SENTINEL_NODES(self) -> List[tuple[str, int]]:
  43. nodes: List[tuple[str, int]] = []
  44. for item in self.REDIS_SENTINELS.split(","):
  45. entry = item.strip()
  46. if not entry:
  47. continue
  48. if ":" not in entry:
  49. raise ValueError(f"Invalid REDIS_SENTINELS entry: {entry}")
  50. host, port = entry.split(":", 1)
  51. nodes.append((host.strip(), int(port.strip())))
  52. return nodes
  53. @property
  54. def REDIS_SENTINEL_KWARGS(self) -> Dict[str, Any]:
  55. kwargs: Dict[str, Any] = {}
  56. if self.REDIS_SENTINEL_USERNAME:
  57. kwargs["username"] = self.REDIS_SENTINEL_USERNAME
  58. if self.REDIS_SENTINEL_PASSWORD:
  59. kwargs["password"] = self.REDIS_SENTINEL_PASSWORD
  60. return kwargs
  61. @property
  62. def REDIS_SENTINEL_URL(self) -> str:
  63. # 标准 Sentinel URL:
  64. # redis+sentinel://[username[:password]@]host1:port1,host2:port2/db?sentinel_master=mymaster
  65. if self.REDIS_SENTINEL_USERNAME:
  66. user = quote(self.REDIS_SENTINEL_USERNAME, safe="")
  67. pwd = quote(self.REDIS_PASSWORD, safe="")
  68. auth = f"{user}:{pwd}@"
  69. elif self.REDIS_PASSWORD:
  70. pwd = quote(self.REDIS_PASSWORD, safe="")
  71. auth = f":{pwd}@"
  72. else:
  73. auth = ""
  74. hosts = ",".join(f"{host}:{port}" for host, port in self.REDIS_SENTINEL_NODES)
  75. master = quote(self.REDIS_MASTER_NAME, safe="")
  76. return f"redis+sentinel://{auth}{hosts}/{self.REDIS_DB}?sentinel_master={master}"
  77. @property
  78. def REDIS_CELERY_SENTINEL_URL(self) -> str:
  79. # Celery/Kombu Sentinel URL 格式:
  80. # sentinel://:redis_password@sentinel_host:port/db;...
  81. # URL 中的 auth 密码 = Redis 主从节点密码
  82. # (Kombu 通过 Sentinel 发现主节点地址后,用此密码向 Redis 主节点认证)
  83. # Sentinel 自身的密码(如有)通过 broker_transport_options["sentinel_kwargs"]["password"] 传递
  84. auth = f":{quote(self.REDIS_PASSWORD, safe='')}@" if self.REDIS_PASSWORD else ""
  85. return ";".join(
  86. f"sentinel://{auth}{host}:{port}/{self.REDIS_DB}"
  87. for host, port in self.REDIS_SENTINEL_NODES
  88. )
  89. @property
  90. def REDIS_SENTINEL_TRANSPORT_OPTIONS(self) -> Dict[str, Any]:
  91. return {
  92. "master_name": self.REDIS_MASTER_NAME,
  93. "sentinel_kwargs": self.REDIS_SENTINEL_KWARGS,
  94. "password": self.REDIS_PASSWORD,
  95. "db": self.REDIS_DB,
  96. }
  97. model_config = SettingsConfigDict(
  98. case_sensitive=True,
  99. env_file=".env",
  100. )
  101. settings = Settings()