For agentic workers: REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (
- [ ]) syntax for tracking.
Goal: 把旧中枢正式降级为兼容/投影层,让写前默认输入统一到 Story Contracts,让写后默认事实统一到 accepted CHAPTER_COMMIT,并让 preflight / dashboard / skills / agents 都显式反映这条主链。
Architecture: 新增统一 runtime 来源解析层,集中回答“本章现在应该信什么”。context_manager、memory_contract_adapter、extract_chapter_context、skills 与 dashboard 都只认 MASTER / VOLUME / CHAPTER / REVIEW + latest accepted CHAPTER_COMMIT;state.json / index.db / summaries / memory_scratchpad 保留,但只作为 commit 投影和查询 read-model。旧 genre-profiles.md、旧散写命令、旧 state-first 心智模型继续存在时,必须只以 fallback / compatibility 明示暴露,不能再伪装成主链。
Tech Stack: Python 3.13, pytest, Pydantic, SQLite (index.db), FastAPI dashboard, React frontend, Story System JSON artifacts under .story-system/
Spec: docs/superpowers/specs/2026-04-12-story-system-evolution-spec.md
Companion Plans: docs/superpowers/plans/2026-04-12-story-system-phase2-contract-first-runtime.md, docs/superpowers/plans/2026-04-12-story-system-phase3-chapter-commit-chain.md, docs/superpowers/plans/2026-04-12-story-system-phase4-event-log-and-override-ledger.md
本计划只覆盖 Phase 5:
CHAPTER_COMMIT 成为默认写后事实入口state / index / summary / memory 显式降级为投影/read-modelgenre-profiles.md 与旧 reference 判断链显式退化为 fallback明确不做:
StateManager / IndexManager / ScratchpadManager 的底层存储.webnovel/state.json、.webnovel/index.db、.webnovel/memory_scratchpad.json退出标准:
context_manager、memory_contract_adapter、extract_chapter_context 默认读取合同与 latest accepted commit,而不是先读旧状态再拼判断ChapterCommitService 写出的 commit 元数据足以声明其为唯一写后事实来源,并能稳定定位 MASTER / VOLUME / CHAPTER / REVIEWwebnovel-write / webnovel-query / webnovel-review / webnovel-plan 与 context-agent / data-agent 的默认指令已切到 contract-first + commit-firstgenre-profiles.md 被明示为 fallback-only;合同存在时不再参与全局系统判断.story-system 主链和 .webnovel/* 投影链的关系webnovel-writer/scripts/data_modules/story_runtime_sources.pywebnovel-writer/scripts/data_modules/story_runtime_health.pywebnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.pywebnovel-writer/scripts/data_modules/tests/test_story_runtime_health.pydocs/architecture/story-system-phase5.mdwebnovel-writer/scripts/data_modules/chapter_commit_service.pywebnovel-writer/scripts/data_modules/context_manager.pywebnovel-writer/scripts/data_modules/memory_contract_adapter.pywebnovel-writer/scripts/extract_chapter_context.pywebnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.pywebnovel-writer/scripts/data_modules/tests/test_context_manager.pywebnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.pywebnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.pywebnovel-writer/scripts/data_modules/webnovel.pywebnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.pywebnovel-writer/dashboard/app.pywebnovel-writer/dashboard/frontend/src/App.jsxwebnovel-writer/dashboard/frontend/src/api.jswebnovel-writer/skills/webnovel-write/SKILL.mdwebnovel-writer/skills/webnovel-review/SKILL.mdwebnovel-writer/skills/webnovel-query/SKILL.mdwebnovel-writer/skills/webnovel-plan/SKILL.mdwebnovel-writer/skills/webnovel-dashboard/SKILL.mdwebnovel-writer/agents/context-agent.mdwebnovel-writer/agents/data-agent.mdwebnovel-writer/scripts/data_modules/tests/test_prompt_integrity.pywebnovel-writer/references/genre-profiles.mdREADME.mddocs/architecture/overview.mddocs/guides/commands.mddocs/operations/operations.mddocs/superpowers/README.mdstory_runtime_sources.py:统一解析本章 contracts、latest commit、fallback 状态,避免各入口各自判断story_runtime_health.py:把主链状态、legacy fallback、rejected/backlog 汇总为 preflight / dashboard 共用健康报告chapter_commit_service.py:补齐 write-fact provenance,确保 commit 能明确引用 volume 合同与写后真理角色context_manager.py:把 runtime pack 切成“合同主链 + commit 摘要 + legacy fallback hints”memory_contract_adapter.py:让 load_context 和 commit_chapter 都服从 contract/commit 主链extract_chapter_context.py:把导出文本从“旧状态摘要”升级为“主链状态 + fallback 显示”webnovel.py:preflight 暴露 story runtime health,统一 CLI 对外语义dashboard/app.py + frontend/src/*:提供可视化 runtime health / commit 状态 / legacy fallback 观测skills/* + agents/*:把人机提示词中的默认工作流从旧散写链切到 commit 主链Files:
webnovel-writer/scripts/data_modules/story_runtime_sources.pywebnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.pywebnovel-writer/scripts/data_modules/chapter_commit_service.pyModify: webnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.py
[ ] Step 1: 先写 runtime 来源与 commit provenance 的失败测试
# webnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.py
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import json
from data_modules.story_runtime_sources import load_runtime_sources
def test_load_runtime_sources_prefers_latest_accepted_commit(tmp_path):
story_root = tmp_path / ".story-system"
(story_root / "chapters").mkdir(parents=True, exist_ok=True)
(story_root / "volumes").mkdir(parents=True, exist_ok=True)
(story_root / "reviews").mkdir(parents=True, exist_ok=True)
(story_root / "commits").mkdir(parents=True, exist_ok=True)
(story_root / "MASTER_SETTING.json").write_text(
json.dumps({"meta": {"contract_type": "MASTER_SETTING"}, "route": {"primary_genre": "玄幻"}}),
encoding="utf-8",
)
(story_root / "chapters" / "chapter_003.json").write_text(
json.dumps({"meta": {"contract_type": "CHAPTER_BRIEF", "chapter": 3}}),
encoding="utf-8",
)
(story_root / "volumes" / "volume_001.json").write_text(
json.dumps({"meta": {"contract_type": "VOLUME_BRIEF", "volume": 1}}),
encoding="utf-8",
)
(story_root / "reviews" / "chapter_003.review.json").write_text(
json.dumps({"meta": {"contract_type": "REVIEW_CONTRACT", "chapter": 3}}),
encoding="utf-8",
)
(story_root / "commits" / "chapter_003.commit.json").write_text(
json.dumps(
{
"meta": {"schema_version": "story-system/v1", "chapter": 3, "status": "accepted"},
"provenance": {"write_fact_role": "chapter_commit"},
"projection_status": {"state": "done", "index": "done", "summary": "done", "memory": "done"},
}
),
encoding="utf-8",
)
snapshot = load_runtime_sources(tmp_path, chapter=3)
assert snapshot.latest_accepted_commit["meta"]["status"] == "accepted"
assert snapshot.primary_write_source == "chapter_commit"
assert snapshot.fallback_sources == []
# webnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.py
def test_commit_service_includes_volume_ref_and_write_fact_provenance(tmp_path):
service = ChapterCommitService(tmp_path)
payload = service.build_commit(
chapter=3,
review_result={"blocking_count": 0},
fulfillment_result={"planned_nodes": ["发现陷阱"], "covered_nodes": ["发现陷阱"], "missed_nodes": [], "extra_nodes": []},
disambiguation_result={"pending": []},
extraction_result={"state_deltas": [], "entity_deltas": [], "accepted_events": []},
)
assert payload["contract_refs"]["volume"] == "volume_001.json"
assert payload["provenance"]["write_fact_role"] == "chapter_commit"
assert payload["provenance"]["projection_role"] == "derived_read_models"
[ ] Step 2: 跑红灯,确认新测试确实失败
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.py webnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.py -q --no-cov
Expected:
ModuleNotFoundError: No module named 'data_modules.story_runtime_sources'或 KeyError: 'volume' / 'provenance'
[ ] Step 3: 实现统一 runtime 来源解析器
# webnovel-writer/scripts/data_modules/story_runtime_sources.py
from __future__ import annotations
from dataclasses import dataclass, field
from pathlib import Path
from typing import Any
from .story_contracts import StoryContractPaths, read_json_if_exists
@dataclass
class RuntimeSourceSnapshot:
chapter: int
contracts: dict[str, dict[str, Any]]
latest_commit: dict[str, Any] | None
latest_accepted_commit: dict[str, Any] | None
fallback_sources: list[str] = field(default_factory=list)
primary_write_source: str = "chapter_commit"
def to_dict(self) -> dict[str, Any]:
return {
"chapter": self.chapter,
"contracts": self.contracts,
"latest_commit": self.latest_commit,
"latest_accepted_commit": self.latest_accepted_commit,
"fallback_sources": self.fallback_sources,
"primary_write_source": self.primary_write_source,
}
def load_runtime_sources(project_root: Path, chapter: int) -> RuntimeSourceSnapshot:
paths = StoryContractPaths.from_project_root(project_root)
contracts = {
"master": read_json_if_exists(paths.master_json) or {},
"chapter": read_json_if_exists(paths.chapter_json(chapter)) or {},
"volume": read_json_if_exists(paths.volume_json(1)) or {},
"review": read_json_if_exists(paths.review_json(chapter)) or {},
}
latest_commit = read_json_if_exists(paths.commit_json(chapter))
latest_accepted_commit = latest_commit if (latest_commit or {}).get("meta", {}).get("status") == "accepted" else None
fallback_sources: list[str] = []
for key, payload in contracts.items():
if not payload:
fallback_sources.append(f"missing_{key}_contract")
if latest_accepted_commit is None:
fallback_sources.append("missing_accepted_commit")
return RuntimeSourceSnapshot(
chapter=chapter,
contracts=contracts,
latest_commit=latest_commit,
latest_accepted_commit=latest_accepted_commit,
fallback_sources=fallback_sources,
)
[ ] Step 4: 在 commit service 中补齐 provenance 字段和 volume 引用
# webnovel-writer/scripts/data_modules/chapter_commit_service.py
from chapter_outline_loader import volume_num_for_chapter_from_state
def build_commit(
self,
chapter: int,
review_result: Dict[str, Any],
fulfillment_result: Dict[str, Any],
disambiguation_result: Dict[str, Any],
extraction_result: Dict[str, Any],
) -> Dict[str, Any]:
volume = volume_num_for_chapter_from_state(self.project_root, chapter) or 1
return {
"meta": {
"schema_version": "story-system/v1",
"chapter": chapter,
"status": status,
},
"contract_refs": {
"master": "MASTER_SETTING.json",
"volume": f"volume_{volume:03d}.json",
"chapter": f"chapter_{chapter:03d}.json",
"review": f"chapter_{chapter:03d}.review.json",
},
"provenance": {
"write_fact_role": "chapter_commit",
"projection_role": "derived_read_models",
"legacy_state_role": "projection_only",
},
"outline_snapshot": {
"planned_nodes": fulfillment_result.get("planned_nodes", []),
"covered_nodes": fulfillment_result.get("covered_nodes", []),
"missed_nodes": fulfillment_result.get("missed_nodes", []),
"extra_nodes": fulfillment_result.get("extra_nodes", []),
},
"review_result": review_result,
"fulfillment_result": fulfillment_result,
"disambiguation_result": disambiguation_result,
"accepted_events": extraction_result.get("accepted_events", []),
"state_deltas": extraction_result.get("state_deltas", []),
"entity_deltas": extraction_result.get("entity_deltas", []),
"summary_text": extraction_result.get("summary_text", ""),
"projection_status": {"state": "pending", "index": "pending", "summary": "pending", "memory": "pending"},
}
[ ] Step 5: 重新跑聚焦测试
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.py webnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.py -q --no-cov
Expected: 2 passed
[ ] Step 6: 提交本任务
git add webnovel-writer/scripts/data_modules/story_runtime_sources.py \
webnovel-writer/scripts/data_modules/chapter_commit_service.py \
webnovel-writer/scripts/data_modules/tests/test_story_runtime_sources.py \
webnovel-writer/scripts/data_modules/tests/test_chapter_commit_service.py
git commit -m "feat: add story runtime source resolver"
Files:
webnovel-writer/scripts/data_modules/context_manager.pywebnovel-writer/scripts/data_modules/memory_contract_adapter.pywebnovel-writer/scripts/extract_chapter_context.pywebnovel-writer/scripts/data_modules/tests/test_context_manager.pywebnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.pyModify: webnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.py
[ ] Step 1: 先补三组失败测试,锁定新主链语义
# webnovel-writer/scripts/data_modules/tests/test_context_manager.py
def test_context_manager_prefers_contract_route_over_legacy_genre_profile(temp_project):
refs_dir = temp_project.project_root / ".claude" / "references"
refs_dir.mkdir(parents=True, exist_ok=True)
(refs_dir / "genre-profiles.md").write_text("## 都市\n- 旧画像提示", encoding="utf-8")
(refs_dir / "reading-power-taxonomy.md").write_text("## 都市\n- 旧分类", encoding="utf-8")
state = {
"project": {"genre": "都市"},
"protagonist_state": {"name": "林默"},
"chapter_meta": {},
"disambiguation_warnings": [],
"disambiguation_pending": [],
}
temp_project.state_file.write_text(json.dumps(state, ensure_ascii=False), encoding="utf-8")
story_root = temp_project.story_system_dir
story_root.mkdir(parents=True, exist_ok=True)
(story_root / "MASTER_SETTING.json").write_text(
json.dumps(
{
"meta": {"schema_version": "story-system/v1", "contract_type": "MASTER_SETTING"},
"route": {"primary_genre": "都市异能"},
"master_constraints": {"core_tone": "先压后爆"},
},
ensure_ascii=False,
),
encoding="utf-8",
)
manager = ContextManager(temp_project)
payload = manager.build_context(3, use_snapshot=False, save_snapshot=False)
assert payload["sections"]["story_contract"]["content"]["master"]["route"]["primary_genre"] == "都市异能"
assert payload["sections"]["runtime_status"]["content"]["fallback_sources"] == ["missing_volume_contract", "missing_chapter_contract", "missing_review_contract", "missing_accepted_commit"]
# webnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.py
def test_commit_chapter_delegates_to_chapter_commit_mainline(tmp_path):
cfg = _make_project(tmp_path)
adapter = MemoryContractAdapter(cfg)
result = adapter.commit_chapter(
3,
{
"review_result": {"blocking_count": 0},
"fulfillment_result": {"planned_nodes": ["发现陷阱"], "covered_nodes": ["发现陷阱"], "missed_nodes": [], "extra_nodes": []},
"disambiguation_result": {"pending": []},
"extraction_result": {"state_deltas": [], "entity_deltas": [], "accepted_events": [], "summary_text": "本章摘要"},
},
)
assert (tmp_path / ".story-system" / "commits" / "chapter_003.commit.json").is_file()
assert result.chapter == 3
assert "commit_status=accepted" in result.warnings
# webnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.py
def test_render_text_contains_runtime_status_section(tmp_path):
from extract_chapter_context import _render_text
text = _render_text(
{
"chapter": 3,
"outline": "测试大纲",
"previous_summaries": [],
"state_summary": "旧状态摘要",
"context_contract_version": "v2",
"reader_signal": {},
"genre_profile": {},
"writing_guidance": {},
"runtime_status": {
"primary_write_source": "chapter_commit",
"fallback_sources": ["missing_accepted_commit"],
},
"latest_commit": {"meta": {"chapter": 3, "status": "rejected"}},
}
)
assert "## Runtime Status" in text
assert "- 写后事实入口: chapter_commit" in text
assert "- Legacy Fallback: missing_accepted_commit" in text
[ ] Step 2: 跑红灯,确认当前入口仍然偏旧链路
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_context_manager.py webnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.py webnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.py -q --no-cov
Expected:
KeyError: 'runtime_status'AssertionError(commit_chapter 仍未生成 .story-system/commits/chapter_003.commit.json)_render_text 中不存在 runtime status 段
[ ] Step 3: 重写 ContextManager 的 pack 组装顺序
# webnovel-writer/scripts/data_modules/context_manager.py
from .story_runtime_sources import load_runtime_sources
def _build_pack(self, chapter: int) -> Dict[str, Any]:
runtime_sources = load_runtime_sources(self.config.project_root, chapter)
state = self._load_state()
story_contract = {
"master": runtime_sources.contracts.get("master") or {},
"volume": runtime_sources.contracts.get("volume") or {},
"chapter": runtime_sources.contracts.get("chapter") or {},
"review_contract": runtime_sources.contracts.get("review") or {},
}
genre_profile = {}
if runtime_sources.fallback_sources:
genre_profile = self._load_genre_profile(state)
genre_profile["mode"] = "fallback_only"
reader_signal = self._load_reader_signal(chapter)
return {
"story_contract": story_contract,
"runtime_status": runtime_sources.to_dict(),
"latest_commit": runtime_sources.latest_accepted_commit or runtime_sources.latest_commit or {},
"genre_profile": genre_profile,
"reader_signal": reader_signal,
"preferences": self._load_json_optional(self.config.webnovel_dir / "preferences.json"),
"writing_guidance": self._build_writing_guidance(chapter, reader_signal, genre_profile),
}
[ ] Step 4: 让 MemoryContractAdapter 同时切换读链与写链
# webnovel-writer/scripts/data_modules/memory_contract_adapter.py
from .chapter_commit_service import ChapterCommitService
from .story_runtime_sources import load_runtime_sources
def load_context(self, chapter: int, budget_tokens: int = 4000) -> ContextPack:
runtime_sources = load_runtime_sources(self.config.project_root, chapter)
sections = {
"story_contracts": runtime_sources.contracts,
"runtime_status": runtime_sources.to_dict(),
"latest_commit": runtime_sources.latest_accepted_commit or runtime_sources.latest_commit or {},
}
return ContextPack(chapter=chapter, sections=sections, budget_used_tokens=0)
def commit_chapter(self, chapter: int, result: dict) -> CommitResult:
service = ChapterCommitService(self.config.project_root)
payload = service.build_commit(
chapter=chapter,
review_result=result.get("review_result", {}),
fulfillment_result=result.get("fulfillment_result", {}),
disambiguation_result=result.get("disambiguation_result", {}),
extraction_result=result.get("extraction_result", {}),
)
service.persist_commit(payload)
payload = service.apply_projections(payload) if payload["meta"]["status"] == "accepted" else payload
summary_path = str(self.config.webnovel_dir / "summaries" / f"ch{chapter:04d}.md")
return CommitResult(
chapter=chapter,
entities_added=len((payload.get("entity_deltas") or [])),
entities_updated=0,
state_changes_recorded=len((payload.get("state_deltas") or [])),
relationships_added=0,
memory_items_added=0,
summary_path=summary_path if Path(summary_path).exists() else "",
warnings=[f"commit_status={payload['meta']['status']}"],
)
[ ] Step 5: 更新 extract_chapter_context.py 的文本输出,让 legacy fallback 显式可见
# webnovel-writer/scripts/extract_chapter_context.py
def _render_text(payload: Dict[str, Any]) -> str:
lines = [f"# 第{payload.get('chapter', 0)}章上下文"]
runtime_status = payload.get("runtime_status") or {}
latest_commit = payload.get("latest_commit") or {}
lines.extend(
[
"## Runtime Status",
f"- 写后事实入口: {runtime_status.get('primary_write_source', 'unknown')}",
f"- Legacy Fallback: {', '.join(runtime_status.get('fallback_sources') or ['none'])}",
f"- Latest Commit: {(latest_commit.get('meta') or {}).get('status', 'missing')}",
]
)
return "\n".join(lines)
[ ] Step 6: 重跑聚焦测试
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_context_manager.py webnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.py webnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.py -q --no-cov
Expected: 3 passed 或对应文件内全部通过
[ ] Step 7: 提交本任务
git add webnovel-writer/scripts/data_modules/context_manager.py \
webnovel-writer/scripts/data_modules/memory_contract_adapter.py \
webnovel-writer/scripts/extract_chapter_context.py \
webnovel-writer/scripts/data_modules/tests/test_context_manager.py \
webnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.py \
webnovel-writer/scripts/data_modules/tests/test_extract_chapter_context.py
git commit -m "feat: switch context loading to contract and commit chain"
Files:
webnovel-writer/skills/webnovel-write/SKILL.mdwebnovel-writer/skills/webnovel-review/SKILL.mdwebnovel-writer/skills/webnovel-query/SKILL.mdwebnovel-writer/skills/webnovel-plan/SKILL.mdwebnovel-writer/skills/webnovel-dashboard/SKILL.mdwebnovel-writer/agents/context-agent.mdwebnovel-writer/agents/data-agent.mdModify: webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py
[ ] Step 1: 先写 prompt integrity 失败测试,锁住新主链叙述
# webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py
def test_webnovel_write_skill_uses_chapter_commit_as_step5_mainline():
text = (SKILLS_DIR / "webnovel-write" / "SKILL.md").read_text(encoding="utf-8")
assert "chapter-commit" in text
assert "accepted `CHAPTER_COMMIT`" in text
assert "state process-chapter" not in text
def test_data_agent_is_described_as_extraction_only_not_direct_write_mainline():
text = (AGENTS_DIR / "data-agent.md").read_text(encoding="utf-8")
assert "chapter-commit" in text
assert "直接写入 index.db 和 state.json" not in text
def test_webnovel_query_skill_prefers_story_system_and_memory_contract():
text = (SKILLS_DIR / "webnovel-query" / "SKILL.md").read_text(encoding="utf-8")
assert "memory-contract load-context" in text
assert ".story-system/" in text
assert 'cat "$PROJECT_ROOT/.webnovel/state.json"' not in text
[ ] Step 2: 跑红灯
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py -q --no-cov
Expected: 至少 1 个断言失败,说明提示词仍在使用旧散写心智模型
[ ] Step 3: 改写 webnovel-write,让 Step 5 以 commit 为成功判定
<!-- webnovel-writer/skills/webnovel-write/SKILL.md -->
### Step 5:构建 extraction artifacts 并提交 `CHAPTER_COMMIT`
必须产出中间文件:
- `${PROJECT_ROOT}/.webnovel/tmp/review_results.json`
- `${PROJECT_ROOT}/.webnovel/tmp/fulfillment_result.json`
- `${PROJECT_ROOT}/.webnovel/tmp/disambiguation_result.json`
- `${PROJECT_ROOT}/.webnovel/tmp/extraction_result.json`
主命令:
```bash
python -X utf8 "${SCRIPTS_DIR}/webnovel.py" --project-root "${PROJECT_ROOT}" chapter-commit \
--chapter {chapter_num} \
--review-result "${PROJECT_ROOT}/.webnovel/tmp/review_results.json" \
--fulfillment-result "${PROJECT_ROOT}/.webnovel/tmp/fulfillment_result.json" \
--disambiguation-result "${PROJECT_ROOT}/.webnovel/tmp/disambiguation_result.json" \
--extraction-result "${PROJECT_ROOT}/.webnovel/tmp/extraction_result.json"
```
成功标准:
- `.story-system/commits/chapter_{chapter_num}.commit.json` 已存在
- `meta.status == accepted`
- `projection_status` 中 `state/index/summary/memory` 均为 `done` 或明确 `skipped`
[ ] Step 4: 改写 query / review / plan / context-agent / data-agent 的默认读取与写入叙述
<!-- webnovel-writer/agents/data-agent.md -->
你负责生成 `extraction_result.json`,并为 `chapter-commit` 提供:
- `accepted_events`
- `state_deltas`
- `entity_deltas`
- `summary_text`
你不是写后真理源。
`state.json / index.db / summaries / memory_scratchpad` 的最终写入由 accepted `CHAPTER_COMMIT` 的 projection writers 完成。
<!-- webnovel-writer/skills/webnovel-query/SKILL.md -->
查询顺序固定为:
1. `.story-system/MASTER_SETTING.json`
2. `.story-system/volumes/*.json`
3. `.story-system/chapters/*.json`
4. latest accepted `.story-system/commits/chapter_XXX.commit.json`
5. `memory-contract load-context`
6. `.webnovel/state.json` / `index.db`(仅 fallback/read-model)
[ ] Step 5: 重新跑 prompt integrity
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py -q --no-cov
Expected: passed
[ ] Step 6: 提交本任务
git add webnovel-writer/skills/webnovel-write/SKILL.md \
webnovel-writer/skills/webnovel-review/SKILL.md \
webnovel-writer/skills/webnovel-query/SKILL.md \
webnovel-writer/skills/webnovel-plan/SKILL.md \
webnovel-writer/skills/webnovel-dashboard/SKILL.md \
webnovel-writer/agents/context-agent.md \
webnovel-writer/agents/data-agent.md \
webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py
git commit -m "docs: cut skills over to chapter commit mainline"
Files:
webnovel-writer/scripts/data_modules/story_runtime_health.pywebnovel-writer/scripts/data_modules/tests/test_story_runtime_health.pywebnovel-writer/scripts/data_modules/webnovel.pywebnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.pywebnovel-writer/dashboard/app.pywebnovel-writer/dashboard/frontend/src/api.jsModify: webnovel-writer/dashboard/frontend/src/App.jsx
[ ] Step 1: 先写 health helper 与 preflight 的失败测试
# webnovel-writer/scripts/data_modules/tests/test_story_runtime_health.py
from data_modules.story_runtime_health import build_story_runtime_health
def test_story_runtime_health_reports_missing_commit_as_not_ready(tmp_path):
report = build_story_runtime_health(tmp_path, chapter=3)
assert report["mainline_ready"] is False
assert "missing_accepted_commit" in report["fallback_sources"]
# webnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.py
def test_preflight_includes_story_runtime_health(monkeypatch, tmp_path, capsys):
module = _load_webnovel_module()
project_root = tmp_path / "book"
(project_root / ".webnovel").mkdir(parents=True, exist_ok=True)
(project_root / ".webnovel" / "state.json").write_text("{}", encoding="utf-8")
monkeypatch.setattr(sys, "argv", ["webnovel", "--project-root", str(project_root), "preflight", "--format", "json"])
with pytest.raises(SystemExit):
module.main()
captured = capsys.readouterr()
assert '"story_runtime"' in captured.out
assert '"mainline_ready"' in captured.out
[ ] Step 2: 跑红灯
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_story_runtime_health.py webnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.py -q --no-cov
Expected:
ModuleNotFoundError: No module named 'data_modules.story_runtime_health'preflight JSON 中不存在 story_runtime
[ ] Step 3: 实现 health helper,并接到 preflight
# webnovel-writer/scripts/data_modules/story_runtime_health.py
from __future__ import annotations
from pathlib import Path
from typing import Any
from .story_runtime_sources import load_runtime_sources
def build_story_runtime_health(project_root: Path, chapter: int | None = None) -> dict[str, Any]:
current_chapter = int(chapter or 0)
snapshot = load_runtime_sources(project_root, current_chapter) if current_chapter else None
fallback_sources = list((snapshot.fallback_sources if snapshot else ["chapter_unspecified"]))
latest_commit = (snapshot.latest_commit if snapshot else None) or {}
return {
"chapter": current_chapter,
"mainline_ready": bool(snapshot and not snapshot.fallback_sources),
"fallback_sources": fallback_sources,
"latest_commit_status": (latest_commit.get("meta") or {}).get("status", "missing"),
"primary_write_source": (snapshot.primary_write_source if snapshot else "chapter_commit"),
}
# webnovel-writer/scripts/data_modules/webnovel.py
from data_modules.story_runtime_health import build_story_runtime_health
def _build_preflight_report(explicit_project_root: Optional[str]) -> dict:
scripts_dir = _scripts_dir().resolve()
plugin_root = scripts_dir.parent
skill_root = plugin_root / "skills" / "webnovel-write"
entry_script = scripts_dir / "webnovel.py"
extract_script = scripts_dir / "extract_chapter_context.py"
checks = [
{"name": "scripts_dir", "ok": scripts_dir.is_dir(), "path": str(scripts_dir)},
{"name": "entry_script", "ok": entry_script.is_file(), "path": str(entry_script)},
{"name": "extract_context_script", "ok": extract_script.is_file(), "path": str(extract_script)},
{"name": "skill_root", "ok": skill_root.is_dir(), "path": str(skill_root)},
]
project_root = ""
project_root_error = ""
try:
resolved_root = _resolve_root(explicit_project_root)
project_root = str(resolved_root)
checks.append({"name": "project_root", "ok": True, "path": project_root})
except Exception as exc:
project_root_error = str(exc)
checks.append({"name": "project_root", "ok": False, "path": explicit_project_root or "", "error": project_root_error})
story_runtime = build_story_runtime_health(Path(project_root)) if project_root else {}
return {
"ok": all(bool(item["ok"]) for item in checks),
"project_root": project_root,
"scripts_dir": str(scripts_dir),
"skill_root": str(skill_root),
"checks": checks,
"project_root_error": project_root_error,
"story_runtime": story_runtime,
}
[ ] Step 4: 在 dashboard 暴露 story runtime health 与 latest commit 状态
# webnovel-writer/dashboard/app.py
from data_modules.story_runtime_health import build_story_runtime_health
@app.get("/api/story-runtime/health")
def story_runtime_health():
project_root = _get_project_root()
state_path = project_root / ".webnovel" / "state.json"
chapter = 0
if state_path.is_file():
state = json.loads(state_path.read_text(encoding="utf-8"))
chapter = int(((state.get("progress") or {}).get("current_chapter") or 0))
return build_story_runtime_health(project_root, chapter=chapter)
// webnovel-writer/dashboard/frontend/src/App.jsx
const [runtimeHealth, setRuntimeHealth] = useState(null)
useEffect(() => {
fetchJSON('/api/story-runtime/health').then(setRuntimeHealth).catch(() => setRuntimeHealth(null))
}, [refreshKey])
{runtimeHealth ? (
<div className="card stat-card">
<span className="stat-label">Story Runtime</span>
<span className="stat-value plain">{runtimeHealth.mainline_ready ? 'Mainline' : 'Fallback'}</span>
<span className="stat-sub">
{runtimeHealth.latest_commit_status} · {(runtimeHealth.fallback_sources || []).join(', ') || 'no fallback'}
</span>
</div>
) : null}
[ ] Step 5: 跑后端测试和前端构建
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_story_runtime_health.py webnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.py -q --no-cov
Expected: passed
Run: npm --prefix webnovel-writer/dashboard/frontend run build
Expected: Vite build success,无新增 lint/blocking error
[ ] Step 6: 提交本任务
git add webnovel-writer/scripts/data_modules/story_runtime_health.py \
webnovel-writer/scripts/data_modules/webnovel.py \
webnovel-writer/scripts/data_modules/tests/test_story_runtime_health.py \
webnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.py \
webnovel-writer/dashboard/app.py \
webnovel-writer/dashboard/frontend/src/api.js \
webnovel-writer/dashboard/frontend/src/App.jsx
git commit -m "feat: surface story runtime health in preflight and dashboard"
Files:
docs/architecture/story-system-phase5.mdwebnovel-writer/references/genre-profiles.mdREADME.mddocs/architecture/overview.mddocs/guides/commands.mddocs/operations/operations.mdModify: docs/superpowers/README.md
[ ] Step 1: 先补 architecture 文档骨架,明确主链已切换
<!-- docs/architecture/story-system-phase5.md -->
# Story System Phase 5
## 核心结论
- 写前真源:`MASTER / VOLUME / CHAPTER / REVIEW`
- 写后真源:accepted `CHAPTER_COMMIT`
- `state / index / summary / memory`:投影/read-model
- `genre-profiles.md`:fallback-only
## 默认链路
```text
story-system --persist/--emit-runtime-contracts
-> context / query / write / review 读取合同
chapter-commit --chapter N
-> accepted commit
-> projection writers
-> state / index / summaries / memory
```
[ ] Step 2: 在 genre-profiles.md 文件头显式打上 fallback-only 标记
<!-- webnovel-writer/references/genre-profiles.md -->
# genre-profiles
> **状态:Fallback Only**
> 高频题材的主判定、主调性、主禁忌已迁移到 Story Contract / CSV route seed。
> 本文件只在合同缺失、项目未升级或显式 fallback 时提供补充提示。
[ ] Step 3: 更新 README / commands / operations,把主链写成可执行手册
<!-- docs/guides/commands.md -->
## Story System 主链
1. 生成合同:
`python -X utf8 "webnovel-writer/scripts/webnovel.py" --project-root "{WORKSPACE_ROOT}" story-system "{goal}" --chapter {N} --persist --emit-runtime-contracts --format both`
2. 提交章节:
`python -X utf8 "webnovel-writer/scripts/webnovel.py" --project-root "{PROJECT_ROOT}" chapter-commit --chapter {N} --review-result ".webnovel/tmp/review_results.json" --fulfillment-result ".webnovel/tmp/fulfillment_result.json" --disambiguation-result ".webnovel/tmp/disambiguation_result.json" --extraction-result ".webnovel/tmp/extraction_result.json"`
3. 检查健康:
`python -X utf8 "webnovel-writer/scripts/webnovel.py" --project-root "{PROJECT_ROOT}" preflight --format json`
[ ] Step 4: 做一次最终回归
Run: python -m pytest webnovel-writer/scripts/data_modules/tests/test_prompt_integrity.py webnovel-writer/scripts/data_modules/tests/test_webnovel_unified_cli.py webnovel-writer/scripts/data_modules/tests/test_context_manager.py webnovel-writer/scripts/data_modules/tests/test_memory_contract_adapter.py -q --no-cov
Expected: 全部通过
[ ] Step 5: 提交本任务
git add docs/architecture/story-system-phase5.md \
webnovel-writer/references/genre-profiles.md \
README.md \
docs/architecture/overview.md \
docs/guides/commands.md \
docs/operations/operations.md \
docs/superpowers/README.md
git commit -m "docs: finalize phase5 legacy downgrade"
13.6 Phase 5:旧链路降级
7.2 运行时优先级
story_contracts -> latest accepted commit -> legacy fallback 固化到入口层7.3 写后真理源
CHAPTER_COMMIT 成为唯一写后事实入口17.1 文档更新要求
本计划统一使用以下命名,不在后续任务中换名:
RuntimeSourceSnapshotload_runtime_sources(project_root, chapter)build_story_runtime_health(project_root, chapter=None)latest_accepted_commitfallback_sourceswrite_fact_role = "chapter_commit"Plan complete and saved to docs/superpowers/plans/2026-04-13-story-system-phase5-legacy-downgrade.md. Two execution options:
1. Subagent-Driven (recommended) - I dispatch a fresh subagent per task, review between tasks, fast iteration
2. Inline Execution - Execute tasks in this session using executing-plans, batch execution with checkpoints
Which approach?