288 lines
12 KiB
Python
288 lines
12 KiB
Python
import subprocess
|
|
import sys
|
|
import tempfile
|
|
import unittest
|
|
from pathlib import Path
|
|
|
|
ROOT = Path(__file__).resolve().parents[1]
|
|
SCRIPT = ROOT / "scripts" / "playbook.py"
|
|
DEFAULT_DEPLOY_ROOT = "docs/standards/playbook"
|
|
|
|
|
|
def run_cli(*args):
|
|
return subprocess.run(
|
|
[sys.executable, str(SCRIPT), *args],
|
|
capture_output=True,
|
|
text=True,
|
|
)
|
|
|
|
|
|
def run_script(script_path: Path, *args, cwd: Path | None = None):
|
|
return subprocess.run(
|
|
[sys.executable, str(script_path), *args],
|
|
capture_output=True,
|
|
text=True,
|
|
cwd=str(cwd) if cwd else None,
|
|
)
|
|
|
|
|
|
class SyncTemplatesPlaceholdersTests(unittest.TestCase):
|
|
def test_templates_no_longer_expose_main_language_placeholder(self):
|
|
example_text = (ROOT / "playbook.toml.example").read_text(encoding="utf-8")
|
|
self.assertNotIn("main_language", example_text)
|
|
|
|
templates_readme = (ROOT / "templates" / "README.md").read_text(
|
|
encoding="utf-8"
|
|
)
|
|
self.assertNotIn("{{MAIN_LANGUAGE}}", templates_readme)
|
|
self.assertNotIn("{{LANGUAGE_1}}", templates_readme)
|
|
self.assertNotIn("docs/workflows/", templates_readme)
|
|
self.assertNotIn("templates/workflows/", templates_readme)
|
|
self.assertIn("docs/superpowers/", templates_readme)
|
|
self.assertIn("docs/prompts/README.md", templates_readme)
|
|
self.assertIn("完整主链只在 `AGENT_RULES.template.md` 定义", templates_readme)
|
|
|
|
agents_template = (ROOT / "templates" / "AGENTS.template.md").read_text(
|
|
encoding="utf-8"
|
|
)
|
|
self.assertNotIn("{{MAIN_LANGUAGE}}", agents_template)
|
|
self.assertIn("AGENT_RULES.md", agents_template)
|
|
self.assertIn("docs/prompts/README.md", agents_template)
|
|
|
|
tech_context_template = (
|
|
ROOT / "templates" / "memory-bank" / "tech-context.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertNotIn("{{MAIN_LANGUAGE}}", tech_context_template)
|
|
self.assertNotIn("{{LANGUAGE_1}}", tech_context_template)
|
|
self.assertNotIn("**主要语言**", tech_context_template)
|
|
self.assertIn("## 不可假设项", tech_context_template)
|
|
|
|
project_brief_template = (
|
|
ROOT / "templates" / "memory-bank" / "project-brief.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("## 成功定义", project_brief_template)
|
|
|
|
update_memory_template = (
|
|
ROOT / "templates" / "prompts" / "coding" / "update-memory.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("workflow-state", update_memory_template)
|
|
self.assertIn("plan-status", update_memory_template)
|
|
self.assertIn("## Updated Files", update_memory_template)
|
|
self.assertIn("## New Context", update_memory_template)
|
|
self.assertIn("## Outstanding Risks", update_memory_template)
|
|
self.assertIn("AGENT_RULES.local.md", update_memory_template)
|
|
|
|
close_task_template = (
|
|
ROOT / "templates" / "prompts" / "coding" / "close-task.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("main_loop.py finish", close_task_template)
|
|
self.assertIn("workflow-state.phase", close_task_template)
|
|
self.assertIn("## Completed", close_task_template)
|
|
self.assertIn("## Not Completed", close_task_template)
|
|
self.assertIn("## Verification", close_task_template)
|
|
self.assertIn("## Risks", close_task_template)
|
|
self.assertIn("## Next Steps", close_task_template)
|
|
|
|
verify_change_template = (
|
|
ROOT / "templates" / "prompts" / "coding" / "verify-change.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("workflow-state.phase", verify_change_template)
|
|
self.assertIn("plan-status", verify_change_template)
|
|
self.assertIn("## Validated", verify_change_template)
|
|
self.assertIn("## Evidence", verify_change_template)
|
|
self.assertIn("## Not Validated", verify_change_template)
|
|
self.assertIn("## Risks", verify_change_template)
|
|
|
|
clarify_template = (
|
|
ROOT / "templates" / "prompts" / "coding" / "clarify.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertNotIn("Vibe-coding", clarify_template)
|
|
self.assertIn("## Current Understanding", clarify_template)
|
|
self.assertIn("## Open Question", clarify_template)
|
|
self.assertIn("## Recommended Default", clarify_template)
|
|
|
|
code_review_template = (
|
|
ROOT / "templates" / "prompts" / "coding" / "code-review.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("## Findings", code_review_template)
|
|
self.assertIn("## Open Questions", code_review_template)
|
|
self.assertIn("## Residual Risk", code_review_template)
|
|
|
|
prompts_readme = (
|
|
ROOT / "templates" / "prompts" / "README.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("AGENT_RULES.md", prompts_readme)
|
|
self.assertIn("docs/superpowers/", prompts_readme)
|
|
self.assertIn("不是流程权威", prompts_readme)
|
|
self.assertNotIn("playbook.py -record-spec", prompts_readme)
|
|
self.assertNotIn("playbook.py -record-plan", prompts_readme)
|
|
self.assertNotIn("using-superpowers", prompts_readme)
|
|
self.assertNotIn("main_loop.py claim", prompts_readme)
|
|
self.assertNotIn("docs/workflows/", prompts_readme)
|
|
self.assertIn("设计与计划产物", prompts_readme)
|
|
|
|
agent_behavior_template = (
|
|
ROOT / "templates" / "prompts" / "system" / "agent-behavior.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("AGENT_RULES.md", agent_behavior_template)
|
|
self.assertIn("AGENT_RULES.local.md", agent_behavior_template)
|
|
self.assertIn("docs/superpowers/specs/", agent_behavior_template)
|
|
self.assertIn("docs/superpowers/plans/", agent_behavior_template)
|
|
self.assertNotIn("using-superpowers", agent_behavior_template)
|
|
self.assertNotIn("main_loop.py claim", agent_behavior_template)
|
|
self.assertNotIn("playbook.py -record-spec", agent_behavior_template)
|
|
self.assertNotIn("playbook.py -record-plan", agent_behavior_template)
|
|
self.assertIn("项目上下文与执行状态写入 `memory-bank/`", agent_behavior_template)
|
|
|
|
rules_template = (
|
|
ROOT / "templates" / "AGENT_RULES.template.md"
|
|
).read_text(encoding="utf-8")
|
|
self.assertIn("唯一流程约束中心", rules_template)
|
|
self.assertIn("唯一设计与计划产物中心", rules_template)
|
|
self.assertIn("memory-bank/progress.md", rules_template)
|
|
|
|
def test_sync_templates_replaces_playbook_scripts_without_main_language_support(self):
|
|
with tempfile.TemporaryDirectory() as tmp_dir:
|
|
config_body = f"""
|
|
[playbook]
|
|
project_root = \"{tmp_dir}\"
|
|
deploy_root = "{DEFAULT_DEPLOY_ROOT}"
|
|
|
|
[sync_rules]
|
|
|
|
[sync_memory_bank]
|
|
|
|
[sync_standards]
|
|
langs = [\"cpp\", \"tsl\"]
|
|
"""
|
|
config_path = Path(tmp_dir) / "playbook.toml"
|
|
config_path.write_text(config_body, encoding="utf-8")
|
|
|
|
result = run_cli("-config", str(config_path))
|
|
self.assertEqual(result.returncode, 0, msg=result.stderr)
|
|
|
|
agents_md = Path(tmp_dir) / "AGENTS.md"
|
|
text = agents_md.read_text(encoding="utf-8")
|
|
self.assertIn(".agents/cpp/index.md", text)
|
|
self.assertNotIn("{{MAIN_LANGUAGE}}", text)
|
|
|
|
tech_context = Path(tmp_dir) / "memory-bank" / "tech-context.md"
|
|
tech_context_text = tech_context.read_text(encoding="utf-8")
|
|
self.assertNotIn("{{LANGUAGE_1}}", tech_context_text)
|
|
self.assertNotIn("{{MAIN_LANGUAGE}}", tech_context_text)
|
|
self.assertNotIn("**主要语言**", tech_context_text)
|
|
|
|
rules_md = Path(tmp_dir) / "AGENT_RULES.md"
|
|
rules_text = rules_md.read_text(encoding="utf-8")
|
|
self.assertIn(
|
|
"docs/standards/playbook/scripts/main_loop.py claim",
|
|
rules_text,
|
|
)
|
|
self.assertIn("docs/superpowers/plans", rules_text)
|
|
self.assertNotIn("plan_progress.py", rules_text)
|
|
self.assertIn("记录 `phase=planning` 与 `spec=<path>`", rules_text)
|
|
self.assertIn(
|
|
"记录 `plan=<path>`、`executor=executing-plans`、",
|
|
rules_text,
|
|
)
|
|
self.assertIn("未领取 Plan 前,不得直接进入 `$executing-plans`", rules_text)
|
|
self.assertIn("默认执行使用 `$executing-plans`", rules_text)
|
|
self.assertIn("不是默认执行器", rules_text)
|
|
self.assertNotIn("{{PLAYBOOK_SCRIPTS}}", rules_text)
|
|
self.assertFalse(rules_text.endswith("\n\n"))
|
|
|
|
def test_sync_standards_rewrites_typescript_docs_prefix_for_vendored_playbook(self):
|
|
with tempfile.TemporaryDirectory() as tmp_dir:
|
|
root = Path(tmp_dir)
|
|
vendor_config = root / "vendor.toml"
|
|
vendor_config.write_text(
|
|
f"""
|
|
[playbook]
|
|
project_root = "{tmp_dir}"
|
|
deploy_root = "{DEFAULT_DEPLOY_ROOT}"
|
|
|
|
[vendor]
|
|
langs = ["typescript"]
|
|
""",
|
|
encoding="utf-8",
|
|
)
|
|
|
|
vendor_result = run_cli("-config", str(vendor_config))
|
|
self.assertEqual(vendor_result.returncode, 0, msg=vendor_result.stderr)
|
|
|
|
sync_config = root / "sync.toml"
|
|
sync_config.write_text(
|
|
f"""
|
|
[playbook]
|
|
project_root = "{tmp_dir}"
|
|
deploy_root = "{DEFAULT_DEPLOY_ROOT}"
|
|
|
|
[sync_standards]
|
|
langs = ["typescript"]
|
|
""",
|
|
encoding="utf-8",
|
|
)
|
|
|
|
vendored_script = (
|
|
root / "docs" / "standards" / "playbook" / "scripts" / "playbook.py"
|
|
)
|
|
sync_result = run_script(
|
|
vendored_script, "-config", str(sync_config), cwd=root
|
|
)
|
|
self.assertEqual(sync_result.returncode, 0, msg=sync_result.stderr)
|
|
|
|
agents_index = root / ".agents" / "typescript" / "index.md"
|
|
text = agents_index.read_text(encoding="utf-8")
|
|
self.assertIn("`docs/standards/playbook/docs/typescript/", text)
|
|
self.assertNotIn("`docs/typescript/", text)
|
|
|
|
def test_sync_memory_bank_includes_active_context_and_human_readable_progress(self):
|
|
with tempfile.TemporaryDirectory() as tmp_dir:
|
|
config_body = f"""
|
|
[playbook]
|
|
project_root = "{tmp_dir}"
|
|
deploy_root = "{DEFAULT_DEPLOY_ROOT}"
|
|
|
|
[sync_rules]
|
|
|
|
[sync_memory_bank]
|
|
project_name = "MyProject"
|
|
"""
|
|
config_path = Path(tmp_dir) / "playbook.toml"
|
|
config_path.write_text(config_body, encoding="utf-8")
|
|
|
|
result = run_cli("-config", str(config_path))
|
|
self.assertEqual(result.returncode, 0, msg=result.stderr)
|
|
|
|
active_context = Path(tmp_dir) / "memory-bank" / "active-context.md"
|
|
self.assertTrue(active_context.is_file())
|
|
|
|
progress = Path(tmp_dir) / "memory-bank" / "progress.md"
|
|
progress_text = progress.read_text(encoding="utf-8")
|
|
self.assertIn("## Current Focus", progress_text)
|
|
self.assertIn("## 状态块示例", progress_text)
|
|
self.assertIn("phase: planning", progress_text)
|
|
self.assertIn("executor: executing-plans", progress_text)
|
|
self.assertIn("<!-- workflow-state:start -->", progress_text)
|
|
self.assertIn("<!-- workflow-state:end -->", progress_text)
|
|
self.assertIn("## Plan Status", progress_text)
|
|
self.assertIn("<!-- plan-status:start -->", progress_text)
|
|
self.assertIn("<!-- plan-status:end -->", progress_text)
|
|
|
|
system_patterns = Path(tmp_dir) / "memory-bank" / "system-patterns.md"
|
|
system_patterns_text = system_patterns.read_text(encoding="utf-8")
|
|
self.assertIn("# 系统模式与约束", system_patterns_text)
|
|
self.assertIn("## 核心不变量", system_patterns_text)
|
|
|
|
agents_md = Path(tmp_dir) / "AGENTS.md"
|
|
agents_text = agents_md.read_text(encoding="utf-8")
|
|
self.assertIn("memory-bank/active-context.md", agents_text)
|
|
|
|
rules_md = Path(tmp_dir) / "AGENT_RULES.md"
|
|
rules_text = rules_md.read_text(encoding="utf-8")
|
|
self.assertIn("memory-bank/active-context.md", rules_text)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|