后端(支持 api_push source 创建/调度): - schemas/source.py:SourceIn.url 改成 str(允许 api_push 的 api-push:// 占位) - admin.py create_source 简化 url 传递 - workers/__main__.py:_rebuild_jobs 跳过 api_push 源(它是被动接收,不抓取) - workers/pipeline.py:run_once 也加同条件,api_push 不进抓取循环 前端: - api/articles.ts:ArticleListItem 加 is_short_news(required)/source_ref; ArticleDetail 加 external_id;导出 IngestTokenOut;adminApi 加 list/create/revoke ingest token 三个方法 - views/Feed.vue:卡片根 class 短新闻加 short-card(淡蓝底 #f6f9fc + 左侧 3px 蓝色色条 #4f9eff);元信息栏加 📰 短讯 角标;长新闻摘要 body_zh_text 截前 200 字,短新闻不截取保留换行(white-space: pre-wrap); 短新闻不显示 AI 插图 - views/ArticleDetail.vue:tag 行加 📰 短讯 + source_ref 角标;短新闻 路径下隐藏翻译状态/重译/原文链接按钮;正文区短新闻直接渲染 body_zh_text,跳过译文/原文/AI 配图卡片;Angel + 美团双评论卡片 都保留 - views/AdminSources.vue:kind 加 api_push 选项;api_push 源 URL 字段 变只读占位、隐藏抓取间隔;列表操作列加 🔑 Token 按钮; 弹窗支持生成(raw_token 一次性显示 + 复制)/列表/撤销 文档: - docs/api-push.md:调用方契约 + 三层去重 + 限速 + lifecycle + owner 操作手册 + curl/Python 示例 + 重试策略 + 故障排查 - README.md:关键特性加 API Push;API 概览加 /api/v1/ingest 和 3 个 /admin/.../ingest-tokens 端点
67 lines
2.3 KiB
Python
67 lines
2.3 KiB
Python
"""Source schemas."""
|
|
from __future__ import annotations
|
|
|
|
from datetime import datetime
|
|
from typing import Annotated
|
|
|
|
from pydantic import BaseModel, ConfigDict, Field
|
|
|
|
from app.models.source import SourceKind
|
|
|
|
|
|
class SourceOut(BaseModel):
|
|
model_config = ConfigDict(from_attributes=True)
|
|
|
|
id: int
|
|
name: str
|
|
slug: str
|
|
kind: SourceKind
|
|
url: str
|
|
enabled: bool
|
|
region: str | None = None
|
|
language_src: str | None = None
|
|
priority: int
|
|
fetch_interval_min: int
|
|
translate_to: str
|
|
last_fetched_at: datetime | None = None
|
|
last_status: str | None = None
|
|
consecutive_failures: int = 0
|
|
# 源级屏蔽分类标签;与 llm_settings.blocklist_tags 合并后注入 classify prompt
|
|
blocklist_tags: list[str] = []
|
|
|
|
|
|
# url 字段:正常源要 HttpUrl(校验合法 URL),但 api_push 源是合成占位(类似 api-push://...)
|
|
# 用 Annotated Union 区分:rss/html_list/tg_channel → HttpUrl;api_push → str
|
|
# 但 SourceIn.kind 未知时(前端一次提交),无法静态区分。最简单的兼容:统一接受 str,
|
|
# 入库前在 admin.create_source 里按 kind 分支校验。
|
|
# 这里改成 str(最长 2048),保留手工校验的责任。
|
|
SourceUrlStr = Annotated[str, Field(min_length=1, max_length=2048)]
|
|
|
|
|
|
class SourceIn(BaseModel):
|
|
name: str = Field(min_length=1, max_length=128)
|
|
slug: str = Field(min_length=1, max_length=128, pattern=r"^[a-z0-9-]+$")
|
|
kind: SourceKind = SourceKind.RSS
|
|
# url:不再强制 HttpUrl,允许 api_push 源的合成 url(api-push://...);
|
|
# rss/html_list/tg_channel 由 admin.create_source 在入库前手工校验
|
|
url: str = Field(min_length=1, max_length=2048)
|
|
region: str | None = None
|
|
language_src: str | None = None
|
|
priority: int = Field(default=50, ge=1, le=100)
|
|
fetch_interval_min: int = Field(default=60, ge=5, le=1440)
|
|
translate_to: str = "zh"
|
|
enabled: bool = True
|
|
detail_selector: dict | None = None
|
|
headers_json: dict | None = None
|
|
blocklist_tags: list[str] = []
|
|
|
|
|
|
class SourceUpdate(BaseModel):
|
|
name: str | None = None
|
|
enabled: bool | None = None
|
|
priority: int | None = Field(default=None, ge=1, le=100)
|
|
fetch_interval_min: int | None = Field(default=None, ge=5, le=1440)
|
|
region: str | None = None
|
|
translate_to: str | None = None
|
|
blocklist_tags: list[str] | None = None
|