config.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  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. CONTRACT_BASE_URL: str = os.getenv("CONTRACT_BASE_URL") # alien_contract 服务地址
  34. # 阿里云短信配置
  35. ALIYUN_SMS_SIGN_NAME_CONTRACT: str = os.getenv("ALIYUN_SMS_SIGN_NAME_CONTRACT")
  36. ALIYUN_SMS_TEMPLATE_CODE_CONTRACT: str = os.getenv("ALIYUN_SMS_TEMPLATE_CODE_CONTRACT")
  37. ALIYUN_ACCESS_KEY_ID: str = os.getenv("ALIYUN_ACCESS_KEY_ID")
  38. ALIYUN_ACCESS_KEY_SECRET: str = os.getenv("ALIYUN_ACCESS_KEY_SECRET")
  39. @property
  40. def SQLALCHEMY_DATABASE_URI(self) -> str:
  41. return f"mysql+pymysql://{self.DB_USER}:{self.DB_PASSWORD}@{self.DB_HOST}:{self.DB_PORT}/{self.DB_NAME}"
  42. @property
  43. def REDIS_SENTINEL_NODES(self) -> List[tuple[str, int]]:
  44. nodes: List[tuple[str, int]] = []
  45. for item in self.REDIS_SENTINELS.split(","):
  46. entry = item.strip()
  47. if not entry:
  48. continue
  49. if ":" not in entry:
  50. raise ValueError(f"Invalid REDIS_SENTINELS entry: {entry}")
  51. host, port = entry.split(":", 1)
  52. nodes.append((host.strip(), int(port.strip())))
  53. return nodes
  54. @property
  55. def REDIS_SENTINEL_KWARGS(self) -> Dict[str, Any]:
  56. kwargs: Dict[str, Any] = {}
  57. if self.REDIS_SENTINEL_USERNAME:
  58. kwargs["username"] = self.REDIS_SENTINEL_USERNAME
  59. if self.REDIS_SENTINEL_PASSWORD:
  60. kwargs["password"] = self.REDIS_SENTINEL_PASSWORD
  61. return kwargs
  62. @property
  63. def REDIS_SENTINEL_URL(self) -> str:
  64. # 标准 Sentinel URL:
  65. # redis+sentinel://[username[:password]@]host1:port1,host2:port2/db?sentinel_master=mymaster
  66. if self.REDIS_SENTINEL_USERNAME:
  67. user = quote(self.REDIS_SENTINEL_USERNAME, safe="")
  68. pwd = quote(self.REDIS_PASSWORD, safe="")
  69. auth = f"{user}:{pwd}@"
  70. elif self.REDIS_PASSWORD:
  71. pwd = quote(self.REDIS_PASSWORD, safe="")
  72. auth = f":{pwd}@"
  73. else:
  74. auth = ""
  75. hosts = ",".join(f"{host}:{port}" for host, port in self.REDIS_SENTINEL_NODES)
  76. master = quote(self.REDIS_MASTER_NAME, safe="")
  77. return f"redis+sentinel://{auth}{hosts}/{self.REDIS_DB}?sentinel_master={master}"
  78. @property
  79. def REDIS_CELERY_SENTINEL_URL(self) -> str:
  80. # Celery/Kombu Sentinel URL 格式:
  81. # sentinel://:redis_password@sentinel_host:port/db;...
  82. # URL 中的 auth 密码 = Redis 主从节点密码
  83. # (Kombu 通过 Sentinel 发现主节点地址后,用此密码向 Redis 主节点认证)
  84. # Sentinel 自身的密码(如有)通过 broker_transport_options["sentinel_kwargs"]["password"] 传递
  85. auth = f":{quote(self.REDIS_PASSWORD, safe='')}@" if self.REDIS_PASSWORD else ""
  86. return ";".join(
  87. f"sentinel://{auth}{host}:{port}/{self.REDIS_DB}"
  88. for host, port in self.REDIS_SENTINEL_NODES
  89. )
  90. @property
  91. def REDIS_SENTINEL_TRANSPORT_OPTIONS(self) -> Dict[str, Any]:
  92. return {
  93. "master_name": self.REDIS_MASTER_NAME,
  94. "sentinel_kwargs": self.REDIS_SENTINEL_KWARGS,
  95. "password": self.REDIS_PASSWORD,
  96. "db": self.REDIS_DB,
  97. }
  98. model_config = SettingsConfigDict(
  99. case_sensitive=True,
  100. env_file=".env.uat",
  101. )
  102. settings = Settings()