180 lines
7.5 KiB
Python
180 lines
7.5 KiB
Python
"""
|
|
配置管理模块
|
|
处理环境变量和配置参数
|
|
"""
|
|
|
|
import os
|
|
from dataclasses import dataclass
|
|
from typing import Optional
|
|
|
|
|
|
@dataclass
|
|
class Config:
|
|
"""配置类"""
|
|
# API密钥
|
|
deepseek_api_key: Optional[str] = None
|
|
openai_api_key: Optional[str] = None
|
|
gemini_api_key: Optional[str] = None
|
|
bocha_api_key: Optional[str] = None
|
|
deepseek_base_url: str = "https://api.deepseek.com"
|
|
openai_base_url: Optional[str] = None
|
|
gemini_base_url: str = "https://www.chataiapi.com/v1"
|
|
|
|
# 模型配置
|
|
default_llm_provider: str = "deepseek" # deepseek、openai 或 gemini
|
|
deepseek_model: str = "deepseek-chat"
|
|
openai_model: str = "gpt-4o-mini"
|
|
gemini_model: str = "gemini-2.5-pro"
|
|
|
|
# 搜索配置
|
|
search_timeout: int = 240
|
|
max_content_length: int = 20000
|
|
|
|
# Agent配置
|
|
max_reflections: int = 2
|
|
max_paragraphs: int = 5
|
|
|
|
# 输出配置
|
|
output_dir: str = "reports"
|
|
save_intermediate_states: bool = True
|
|
|
|
def validate(self) -> bool:
|
|
"""验证配置"""
|
|
# 检查必需的API密钥
|
|
if self.default_llm_provider == "deepseek" and not self.deepseek_api_key:
|
|
print("错误: DeepSeek API Key未设置")
|
|
return False
|
|
|
|
if self.default_llm_provider == "openai" and not self.openai_api_key:
|
|
print("错误: OpenAI API Key未设置")
|
|
return False
|
|
|
|
if self.default_llm_provider == "gemini" and not self.gemini_api_key:
|
|
print("错误: Gemini API Key未设置")
|
|
return False
|
|
|
|
if not self.bocha_api_key:
|
|
print("错误: Bocha API Key未设置")
|
|
return False
|
|
|
|
return True
|
|
|
|
@classmethod
|
|
def from_file(cls, config_file: str) -> "Config":
|
|
"""从配置文件创建配置"""
|
|
if config_file.endswith('.py'):
|
|
# Python配置文件
|
|
import importlib.util
|
|
|
|
# 动态导入配置文件
|
|
spec = importlib.util.spec_from_file_location("config", config_file)
|
|
config_module = importlib.util.module_from_spec(spec)
|
|
spec.loader.exec_module(config_module)
|
|
|
|
return cls(
|
|
deepseek_api_key=getattr(config_module, "DEEPSEEK_API_KEY", None),
|
|
openai_api_key=getattr(config_module, "OPENAI_API_KEY", None),
|
|
gemini_api_key=getattr(config_module, "GEMINI_API_KEY", None),
|
|
deepseek_base_url=getattr(config_module, "DEEPSEEK_BASE_URL", "https://api.deepseek.com"),
|
|
openai_base_url=getattr(config_module, "OPENAI_BASE_URL", None),
|
|
gemini_base_url=getattr(config_module, "GEMINI_BASE_URL", "https://www.chataiapi.com/v1"),
|
|
bocha_api_key=getattr(config_module, "BOCHA_API_KEY", None),
|
|
default_llm_provider=getattr(config_module, "DEFAULT_LLM_PROVIDER", "deepseek"),
|
|
deepseek_model=getattr(config_module, "DEEPSEEK_MODEL", "deepseek-chat"),
|
|
openai_model=getattr(config_module, "OPENAI_MODEL", "gpt-4o-mini"),
|
|
gemini_model=getattr(config_module, "GEMINI_MODEL", "gemini-2.5-pro"),
|
|
search_timeout=getattr(config_module, "SEARCH_TIMEOUT", 240),
|
|
max_content_length=getattr(config_module, "SEARCH_CONTENT_MAX_LENGTH", 20000),
|
|
max_reflections=getattr(config_module, "MAX_REFLECTIONS", 2),
|
|
max_paragraphs=getattr(config_module, "MAX_PARAGRAPHS", 5),
|
|
output_dir=getattr(config_module, "OUTPUT_DIR", "reports"),
|
|
save_intermediate_states=getattr(config_module, "SAVE_INTERMEDIATE_STATES", True)
|
|
)
|
|
else:
|
|
# .env格式配置文件
|
|
config_dict = {}
|
|
|
|
if os.path.exists(config_file):
|
|
with open(config_file, 'r', encoding='utf-8') as f:
|
|
for line in f:
|
|
line = line.strip()
|
|
if line and not line.startswith('#') and '=' in line:
|
|
key, value = line.split('=', 1)
|
|
config_dict[key.strip()] = value.strip()
|
|
|
|
return cls(
|
|
deepseek_api_key=config_dict.get("DEEPSEEK_API_KEY"),
|
|
openai_api_key=config_dict.get("OPENAI_API_KEY"),
|
|
gemini_api_key=config_dict.get("GEMINI_API_KEY"),
|
|
deepseek_base_url=config_dict.get("DEEPSEEK_BASE_URL", "https://api.deepseek.com"),
|
|
openai_base_url=config_dict.get("OPENAI_BASE_URL"),
|
|
gemini_base_url=config_dict.get("GEMINI_BASE_URL", "https://www.chataiapi.com/v1"),
|
|
bocha_api_key=config_dict.get("BOCHA_API_KEY"),
|
|
default_llm_provider=config_dict.get("DEFAULT_LLM_PROVIDER", "deepseek"),
|
|
deepseek_model=config_dict.get("DEEPSEEK_MODEL", "deepseek-chat"),
|
|
openai_model=config_dict.get("OPENAI_MODEL", "gpt-4o-mini"),
|
|
gemini_model=config_dict.get("GEMINI_MODEL", "gemini-2.5-pro"),
|
|
search_timeout=int(config_dict.get("SEARCH_TIMEOUT", "240")),
|
|
max_content_length=int(config_dict.get("SEARCH_CONTENT_MAX_LENGTH", "20000")),
|
|
max_reflections=int(config_dict.get("MAX_REFLECTIONS", "2")),
|
|
max_paragraphs=int(config_dict.get("MAX_PARAGRAPHS", "5")),
|
|
output_dir=config_dict.get("OUTPUT_DIR", "reports"),
|
|
save_intermediate_states=config_dict.get("SAVE_INTERMEDIATE_STATES", "true").lower() == "true"
|
|
)
|
|
|
|
|
|
def load_config(config_file: Optional[str] = None) -> Config:
|
|
"""
|
|
加载配置
|
|
|
|
Args:
|
|
config_file: 配置文件路径,如果不指定则使用默认路径
|
|
|
|
Returns:
|
|
配置对象
|
|
"""
|
|
# 确定配置文件路径
|
|
if config_file:
|
|
if not os.path.exists(config_file):
|
|
raise FileNotFoundError(f"配置文件不存在: {config_file}")
|
|
file_to_load = config_file
|
|
else:
|
|
# 尝试加载常见的配置文件
|
|
for config_path in ["config.py", "config.env", ".env"]:
|
|
if os.path.exists(config_path):
|
|
file_to_load = config_path
|
|
print(f"已找到配置文件: {config_path}")
|
|
break
|
|
else:
|
|
raise FileNotFoundError("未找到配置文件,请创建 config.py 文件")
|
|
|
|
# 创建配置对象
|
|
config = Config.from_file(file_to_load)
|
|
|
|
# 验证配置
|
|
if not config.validate():
|
|
raise ValueError("配置验证失败,请检查配置文件中的API密钥")
|
|
|
|
return config
|
|
|
|
|
|
def print_config(config: Config):
|
|
"""打印配置信息(隐藏敏感信息)"""
|
|
print("\n=== 当前配置 ===")
|
|
print(f"LLM提供商: {config.default_llm_provider}")
|
|
print(f"DeepSeek模型: {config.deepseek_model}")
|
|
print(f"OpenAI模型: {config.openai_model}")
|
|
|
|
print(f"搜索超时: {config.search_timeout}秒")
|
|
print(f"最大内容长度: {config.max_content_length}")
|
|
print(f"最大反思次数: {config.max_reflections}")
|
|
print(f"最大段落数: {config.max_paragraphs}")
|
|
print(f"输出目录: {config.output_dir}")
|
|
print(f"保存中间状态: {config.save_intermediate_states}")
|
|
|
|
# 显示API密钥状态(不显示实际密钥)
|
|
print(f"DeepSeek API Key: {'已设置' if config.deepseek_api_key else '未设置'}")
|
|
print(f"OpenAI API Key: {'已设置' if config.openai_api_key else '未设置'}")
|
|
print(f"Bocha API Key: {'已设置' if config.bocha_api_key else '未设置'}")
|
|
print("==================\n")
|