feat: 任务进度实时展示、接口测试、暗色主题重构及多项 bug 修复
后端 - 新增 app/task_progress.py 线程安全进度注册表 - 任务改为后台线程异步执行(_run_task_background),手动触发立即返回 task_key - 6 个任务函数(summarizer/tagger/scorer/deduplicator/brief/taxonomy)循环内上报进度 - scheduler 定时任务同步上报进度(trigger=scheduled) - 新增 GET /api/tasks/progress 与 POST /api/tasks/progress/reset 接口 - 新增 POST /api/test-connection 接口连通性测试(独立短超时客户端) - 修复 ai_client/rss_client 配置在 import 时固化的 bug(改为 property 运行时读取 settings), 导致实际任务用 .env 假 key 调 LLM 401 - 修复 ai_client 对 reasoning 模型(MiniMax-M3 等)输出 <think> 块的 JSON 解析失败 - 修复 taxonomy bootstrap:LLM 超时(改用 300s 专用 client)、MiniMax 输出审查 (精简样本仅标题 + 约束生成中性类目名)、失败误报 success(改抛异常如实标记) - 修复 models.py 双外键关系映射启动崩溃(显式 foreign_keys) - 修复 main.py SPA 路由 404、ArticleOut.published_at 序列化 500 - 移除 lifespan 同步 bootstrap 阻塞启动,改由 scheduler 后台异步执行 前端 - Deep Ink 高对比度暗色主题重构,修复 Element Plus 暗色模式对比度问题 - Tasks 页面任务进度实时展示(进度条/阶段/计数/状态/触发来源)+ 1.5s 轮询 - 接口测试面板(rssKeeper / LLM 连通性 + 延迟) - 修复 nextJobs jobId 映射 bug 部署与文档 - Dockerfile 优化(BuildKit 缓存挂载、预编译 wheel、去 gcc、阿里云镜像源) - 新增 API.md 接口文档 Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -9,6 +9,7 @@ from sqlalchemy.orm import Session
|
||||
|
||||
from config import settings
|
||||
from models import EnrichedArticle, DailyBrief
|
||||
from app.task_progress import update_progress
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
@@ -76,6 +77,7 @@ def generate_daily_brief(db: Session, date_str: str = None, force: bool = False)
|
||||
existing = db.query(DailyBrief).filter(DailyBrief.brief_date == date_str).first()
|
||||
if existing and not force:
|
||||
logger.info("日期 %s 简报已存在,跳过生成", date_str)
|
||||
update_progress("generate_daily_brief", status="running", stage="简报已存在", current=0, total=0, message="简报已存在,跳过生成")
|
||||
return {
|
||||
"date": date_str,
|
||||
"total_articles": existing.total_articles,
|
||||
@@ -86,6 +88,8 @@ def generate_daily_brief(db: Session, date_str: str = None, force: bool = False)
|
||||
day_start = datetime.strptime(date_str, "%Y-%m-%d")
|
||||
day_end = day_start + timedelta(days=1)
|
||||
|
||||
update_progress("generate_daily_brief", status="running", stage="加载文章", current=0, total=0)
|
||||
|
||||
# 取当天去重后的代表文章
|
||||
query = (
|
||||
db.query(EnrichedArticle)
|
||||
@@ -106,6 +110,7 @@ def generate_daily_brief(db: Session, date_str: str = None, force: bool = False)
|
||||
)
|
||||
|
||||
# 按分类分组并排序
|
||||
update_progress("generate_daily_brief", status="running", stage="按分类整理", current=0, total=0)
|
||||
by_category: Dict[str, List[Dict[str, Any]]] = {}
|
||||
for art in representative_articles:
|
||||
cat = art.category or "未分类"
|
||||
@@ -127,6 +132,7 @@ def generate_daily_brief(db: Session, date_str: str = None, force: bool = False)
|
||||
}
|
||||
|
||||
# 生成 Markdown 文件
|
||||
update_progress("generate_daily_brief", status="running", stage="生成 Markdown", current=0, total=0)
|
||||
output_dir = settings.brief_output_dir_path / date_str
|
||||
output_dir.mkdir(parents=True, exist_ok=True)
|
||||
markdown_path = output_dir / "daily-brief.md"
|
||||
@@ -134,6 +140,7 @@ def generate_daily_brief(db: Session, date_str: str = None, force: bool = False)
|
||||
markdown_path.write_text(markdown_content, encoding="utf-8")
|
||||
|
||||
# 更新文章 brief_date
|
||||
update_progress("generate_daily_brief", status="running", stage="保存简报", current=0, total=0)
|
||||
for art in representative_articles:
|
||||
art.brief_date = date_str
|
||||
|
||||
|
||||
Reference in New Issue
Block a user