feat: 回忆录证据血缘与内部评测可追溯,顺带对齐本地评测台与 CI

数据库与模型:新增多版迁移(章节证据快照、对话血缘、记忆事实/时间线 lineage 等),把「成稿 ↔ 对话/记忆」的溯源信息落到表结构里。
业务链路:会话与 WS、回忆录/故事流水线、记忆写入与 enrichment 等跟着接上线索与快照;新增章节证据快照与评测侧 EvalTraceService 等模块,方便组评审用的证据包。
内部评测:自动化 run 与手工 memoir 评审共用可追溯证据;rubric/ judge 相关脚本与文档有配套调整。
app-eval-web:Memoir/实验详情里能展开看证据摘要与 evidence_trace(含对话轮次 id);Vite 代理与 development.sh 注入的 API 端口与当前默认内部评测端口一致,避免改端口后页面连错服务。
工程杂项:GitHub Actions / 仓库说明有更新;各适配器与支付/配额/plan 等多处为小改动或跟随主改动的收尾;新增/扩充了?
This commit is contained in:
Kevin
2026-04-08 15:37:09 +08:00
parent 6772e1269c
commit 309a051038
109 changed files with 4125 additions and 858 deletions

View File

@@ -12,13 +12,17 @@ from datetime import datetime, timezone
from sqlalchemy import delete, func, select
from sqlalchemy.orm import Session, joinedload
from app.core.db import utc_now
from app.core.logging import get_logger
from app.features.memoir import repo as memoir_repo
from app.features.memoir.asset_resolver import strip_asset_image_refs_from_markdown
from app.features.memoir.models import ChapterStoryLink
from app.features.memoir import repo as memoir_repo
from app.features.story.image_intent_extractor import extract_primary_image_intent
from app.features.story.models import Story, StoryImageIntent, StoryVersion
from app.features.story.models import (
Story,
StoryEvidenceLink,
StoryImageIntent,
StoryVersion,
)
from app.features.story.time_hints import apply_infer_story_time_start_to_model
logger = get_logger(__name__)
@@ -108,6 +112,61 @@ def _extract_and_store_image_intent_sync(
)
def replace_story_evidence_links_sync(
session: Session,
*,
story_id: str,
chunk_ids: list[str],
fact_ids: list[str],
timeline_event_ids: list[str],
summary_ids: list[str],
) -> None:
"""以当前生成所用的检索闭包覆盖 story 证据关联artifact 当前态绑定)。"""
session.execute(
delete(StoryEvidenceLink).where(StoryEvidenceLink.story_id == story_id)
)
for cid in chunk_ids:
session.add(
StoryEvidenceLink(
id=str(uuid.uuid4()),
story_id=story_id,
evidence_type="chunk",
evidence_id=cid,
role="primary",
)
)
for fid in fact_ids:
session.add(
StoryEvidenceLink(
id=str(uuid.uuid4()),
story_id=story_id,
evidence_type="fact",
evidence_id=fid,
role="supporting",
)
)
for tid in timeline_event_ids:
session.add(
StoryEvidenceLink(
id=str(uuid.uuid4()),
story_id=story_id,
evidence_type="timeline_event",
evidence_id=tid,
role="supporting",
)
)
for sid in summary_ids:
session.add(
StoryEvidenceLink(
id=str(uuid.uuid4()),
story_id=story_id,
evidence_type="summary",
evidence_id=sid,
role="background",
)
)
def create_story_with_version_sync(
session: Session,
*,
@@ -115,6 +174,7 @@ def create_story_with_version_sync(
title: str,
canonical_markdown: str,
stage: str | None = None,
prompt_meta: dict | None = None,
) -> Story:
md = strip_asset_image_refs_from_markdown(canonical_markdown or "")
story = Story(
@@ -134,6 +194,7 @@ def create_story_with_version_sync(
markdown_snapshot=md,
actor_type="ai",
source_type="generate",
prompt_meta=prompt_meta,
)
session.add(version)
session.flush()
@@ -154,6 +215,7 @@ def append_story_version_sync(
*,
actor_type: str = "ai",
source_type: str = "generate",
prompt_meta: dict | None = None,
) -> StoryVersion:
story = session.get(Story, story_id)
if not story:
@@ -170,6 +232,7 @@ def append_story_version_sync(
actor_type=actor_type,
source_type=source_type,
parent_version_id=parent_id,
prompt_meta=prompt_meta,
)
session.add(version)
session.flush()