修复问题: - 添加缺失的 httpx 依赖到 requirements.txt - 修复前端批量删除参数格式与后端不匹配(数组->对象数组) - 移除 app/api/main.py 中重复创建 app 的冗余代码 - 修复 Plugins.vue v-model 直接修改 store 状态的 Vue 警告 - 修复 README 端口/启动命令文档与实际配置不一致 - 修正 pytest.ini 过时配置 (asyncio_default_fixture_loop_scope) - 修复 WebUI index.html 语言设置为 zh-CN - 修复 .gitignore 错误忽略 tests/ 目录 后端优化: - 修复调度器默认间隔从 5 秒改为 30 分钟,避免无节制验证 - 修复 validate_all_now 在调度器停止时无法执行的 bug - 设置保存后热更新运行中调度器的验证间隔 - 将 update_score 优化为原子单事务 SQL,消除并发竞态 - 导出功能改为真正的流式分批读取(iter_batches),降低大导出内存占用 - ProxyResponse Schema 补齐 response_time_ms 字段 - 日志级别改为从配置动态读取,不再硬编码 INFO - 清理 validator_service 中的冗余 try/finally 代码 插件健壮性: - 修复 ip3366/ip89/kuaidaili/proxylist_download/speedx/yundaili/proxyscrape 的端口范围检查和 IPv6 地址解析问题(改用 rsplit + 1-65535 校验) - 修复 PluginService.list_plugins 并发竞争条件 - 修复 run_all_plugins 去重逻辑与数据库 UNIQUE 约束保持一致 - 修复 proxyscrape 异常时错误跳过 fallback 的 bug 测试: - 新增 7 个插件解析单元测试 - 新增 update_score 自动删除和 iter_batches 流式读取测试 - 全部 74 个测试通过
47 lines
1.3 KiB
Python
47 lines
1.3 KiB
Python
import logging
|
||
import os
|
||
from logging.handlers import TimedRotatingFileHandler
|
||
|
||
|
||
# 获取项目根目录并创建 logs 目录
|
||
base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
|
||
log_dir = os.path.join(base_dir, 'logs')
|
||
os.makedirs(log_dir, exist_ok=True)
|
||
|
||
log_file = os.path.join(log_dir, 'app.log')
|
||
|
||
formatter = logging.Formatter(
|
||
'[%(asctime)s] %(name)s [%(levelname)s] %(filename)s[line:%(lineno)d]: %(message)s'
|
||
)
|
||
|
||
# 按天切分的文件处理器,保留最近 7 天
|
||
file_handler = TimedRotatingFileHandler(
|
||
log_file,
|
||
when='midnight',
|
||
interval=1,
|
||
backupCount=7,
|
||
encoding='utf-8'
|
||
)
|
||
file_handler.setFormatter(formatter)
|
||
|
||
console_handler = logging.StreamHandler()
|
||
console_handler.setFormatter(formatter)
|
||
|
||
# 获取标准 logger
|
||
logger = logging.getLogger('ProxyPool')
|
||
|
||
# 尝试从配置读取日志级别,默认 INFO
|
||
try:
|
||
from app.core.config import settings
|
||
log_level = getattr(logging, settings.log_level.upper(), logging.INFO)
|
||
except Exception:
|
||
log_level = logging.INFO
|
||
logger.setLevel(log_level)
|
||
file_handler.setLevel(log_level)
|
||
console_handler.setLevel(log_level)
|
||
|
||
# 防止重复添加 handler(如模块重导入)
|
||
if not logger.handlers:
|
||
logger.addHandler(file_handler)
|
||
logger.addHandler(console_handler)
|