cc-devflow 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/CLAUDE.md +83 -0
- package/.claude/agents/architecture-designer.md +443 -0
- package/.claude/agents/bug-analyzer.md +382 -0
- package/.claude/agents/checklist-agent.md +175 -0
- package/.claude/agents/clarify-analyst.md +50 -0
- package/.claude/agents/code-reviewer.md +71 -0
- package/.claude/agents/codex-analyzer.md +39 -0
- package/.claude/agents/compatibility-checker.md +580 -0
- package/.claude/agents/consistency-checker.md +532 -0
- package/.claude/agents/impact-analyzer.md +441 -0
- package/.claude/agents/planner.md +230 -0
- package/.claude/agents/prd-writer.md +320 -0
- package/.claude/agents/project-guidelines-generator.md +1329 -0
- package/.claude/agents/qa-tester.md +313 -0
- package/.claude/agents/release-manager.md +295 -0
- package/.claude/agents/security-reviewer.md +314 -0
- package/.claude/agents/style-guide-generator.md +458 -0
- package/.claude/agents/tech-architect.md +516 -0
- package/.claude/agents/ui-designer.md +485 -0
- package/.claude/commands/code-review-high.md +58 -0
- package/.claude/commands/core-architecture.md +429 -0
- package/.claude/commands/core-guidelines.md +486 -0
- package/.claude/commands/core-roadmap.md +439 -0
- package/.claude/commands/core-style.md +293 -0
- package/.claude/commands/flow-archive.md +245 -0
- package/.claude/commands/flow-checklist.md +260 -0
- package/.claude/commands/flow-clarify.md +136 -0
- package/.claude/commands/flow-constitution.md +82 -0
- package/.claude/commands/flow-dev.md +134 -0
- package/.claude/commands/flow-epic.md +150 -0
- package/.claude/commands/flow-fix.md +104 -0
- package/.claude/commands/flow-ideate.md +214 -0
- package/.claude/commands/flow-init.md +313 -0
- package/.claude/commands/flow-new.md +394 -0
- package/.claude/commands/flow-prd.md +131 -0
- package/.claude/commands/flow-qa.md +93 -0
- package/.claude/commands/flow-release.md +92 -0
- package/.claude/commands/flow-restart.md +98 -0
- package/.claude/commands/flow-status.md +64 -0
- package/.claude/commands/flow-tech.md +142 -0
- package/.claude/commands/flow-ui.md +189 -0
- package/.claude/commands/flow-update.md +111 -0
- package/.claude/commands/flow-upgrade.md +115 -0
- package/.claude/commands/flow-verify.md +96 -0
- package/.claude/commands/problem-analyzer.md +60 -0
- package/.claude/config/quality-rules.yml +161 -0
- package/.claude/docs/SPEC_KIT_CONSTITUTION_ANALYSIS.md +426 -0
- package/.claude/docs/design/consistency-conflict-detection-algorithms.md +658 -0
- package/.claude/docs/design/intent-driven-input-design.md +380 -0
- package/.claude/docs/design/prd-version-management-design.md +437 -0
- package/.claude/docs/guides/INIT_TROUBLESHOOTING.md +117 -0
- package/.claude/docs/guides/NEW_TROUBLESHOOTING.md +151 -0
- package/.claude/docs/guides/ROADMAP_TROUBLESHOOTING.md +188 -0
- package/.claude/docs/guides/TASK_COMPLETION_MARKING.md +338 -0
- package/.claude/docs/templates/ARCHITECTURE_TEMPLATE.md +633 -0
- package/.claude/docs/templates/BACKLOG_TEMPLATE.md +261 -0
- package/.claude/docs/templates/CHECKLIST_TEMPLATE.md +52 -0
- package/.claude/docs/templates/CLARIFICATION_REPORT_TEMPLATE.md +206 -0
- package/.claude/docs/templates/CODE_REVIEW_TEMPLATE.md +71 -0
- package/.claude/docs/templates/EPIC_TEMPLATE.md +805 -0
- package/.claude/docs/templates/INIT_FLOW_TEMPLATE.md +213 -0
- package/.claude/docs/templates/INTENT_CLARIFICATION_TEMPLATE.md +57 -0
- package/.claude/docs/templates/NEW_ORCHESTRATION_TEMPLATE.md +148 -0
- package/.claude/docs/templates/PRD_TEMPLATE.md +562 -0
- package/.claude/docs/templates/RESEARCH_TEMPLATE.md +276 -0
- package/.claude/docs/templates/REVIEW-HIGH.md +57 -0
- package/.claude/docs/templates/ROADMAP_DIALOGUE_TEMPLATE.md +198 -0
- package/.claude/docs/templates/ROADMAP_TEMPLATE.md +310 -0
- package/.claude/docs/templates/STYLE_TEMPLATE.md +1266 -0
- package/.claude/docs/templates/TASKS_TEMPLATE.md +523 -0
- package/.claude/docs/templates/TECH_DESIGN_TEMPLATE.md +1019 -0
- package/.claude/docs/templates/UI_PROTOTYPE_TEMPLATE.md +1436 -0
- package/.claude/guides/agent-guides/agent-coordination-guide.md +459 -0
- package/.claude/guides/project-guidelines-system.md +463 -0
- package/.claude/guides/technical-guides/datetime-handling-guide.md +563 -0
- package/.claude/guides/technical-guides/git-github-guide.md +642 -0
- package/.claude/guides/technical-guides/test-execution-guide.md +618 -0
- package/.claude/guides/workflow-guides/bug-fix-orchestrator.md +217 -0
- package/.claude/guides/workflow-guides/flow-orchestrator.md +282 -0
- package/.claude/hooks/checklist-gate.js +397 -0
- package/.claude/hooks/error-handling-reminder.sh +12 -0
- package/.claude/hooks/error-handling-reminder.ts +459 -0
- package/.claude/hooks/post-tool-use-tracker.sh +280 -0
- package/.claude/hooks/pre-tool-use-guardrail.sh +36 -0
- package/.claude/hooks/pre-tool-use-guardrail.ts +342 -0
- package/.claude/hooks/skill-activation-prompt.sh +36 -0
- package/.claude/hooks/skill-activation-prompt.ts +214 -0
- package/.claude/hooks/state/skills-used-test-guard.json +3 -0
- package/.claude/rules/devflow-conventions.md +305 -0
- package/.claude/rules/project-constitution.md +748 -0
- package/.claude/schemas/constitution.schema.json +43 -0
- package/.claude/scripts/analyze-upgrade-impact.sh +200 -0
- package/.claude/scripts/archive-requirement.sh +351 -0
- package/.claude/scripts/calculate-checklist-completion.sh +243 -0
- package/.claude/scripts/calculate-quarter.sh +206 -0
- package/.claude/scripts/check-dependencies.sh +409 -0
- package/.claude/scripts/check-prerequisites.sh +232 -0
- package/.claude/scripts/check-task-status.sh +264 -0
- package/.claude/scripts/checklist-errors.sh +131 -0
- package/.claude/scripts/common.sh +570 -0
- package/.claude/scripts/consolidate-research.sh +182 -0
- package/.claude/scripts/create-requirement.sh +426 -0
- package/.claude/scripts/export-contracts.sh +117 -0
- package/.claude/scripts/extract-data-model.sh +78 -0
- package/.claude/scripts/generate-clarification-questions.sh +377 -0
- package/.claude/scripts/generate-clarification-report.sh +463 -0
- package/.claude/scripts/generate-quickstart.sh +146 -0
- package/.claude/scripts/generate-research-tasks.sh +157 -0
- package/.claude/scripts/generate-status-report.sh +523 -0
- package/.claude/scripts/generate-tech-analysis.sh +46 -0
- package/.claude/scripts/locate-requirement-in-roadmap.sh +233 -0
- package/.claude/scripts/manage-constitution.sh +602 -0
- package/.claude/scripts/mark-task-complete.sh +198 -0
- package/.claude/scripts/populate-research-tasks.sh +259 -0
- package/.claude/scripts/recover-workflow.sh +460 -0
- package/.claude/scripts/run-clarify-scan.sh +601 -0
- package/.claude/scripts/run-high-review.sh +62 -0
- package/.claude/scripts/run-problem-analysis.sh +68 -0
- package/.claude/scripts/setup-epic.sh +173 -0
- package/.claude/scripts/sync-roadmap-progress.sh +300 -0
- package/.claude/scripts/sync-task-marks.sh +199 -0
- package/.claude/scripts/test-clarify-scan.sh +515 -0
- package/.claude/scripts/update-agent-context.sh +806 -0
- package/.claude/scripts/validate-constitution.sh +567 -0
- package/.claude/scripts/validate-hooks.sh +487 -0
- package/.claude/scripts/validate-research.sh +332 -0
- package/.claude/scripts/validate-scope-boundary.sh +493 -0
- package/.claude/scripts/verify-setup.sh +37 -0
- package/.claude/settings.json +76 -0
- package/.claude/skills/_reference-implementations/README.md +96 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/SKILL.md +302 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/architecture-overview.md +451 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/async-and-errors.md +307 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/complete-examples.md +638 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/configuration.md +275 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/database-patterns.md +224 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/middleware-guide.md +213 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/routing-and-controllers.md +756 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/sentry-and-monitoring.md +336 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/services-and-repositories.md +789 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/testing-guide.md +235 -0
- package/.claude/skills/_reference-implementations/backend-express-prisma/resources/validation-patterns.md +754 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/SKILL.md +399 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/common-patterns.md +331 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/complete-examples.md +872 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/component-patterns.md +502 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/data-fetching.md +767 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/file-organization.md +502 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/loading-and-error-states.md +501 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/performance.md +406 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/routing-guide.md +364 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/styling-guide.md +428 -0
- package/.claude/skills/_reference-implementations/frontend-react-mui/resources/typescript-standards.md +418 -0
- package/.claude/skills/cc-devflow-orchestrator/SKILL.md +229 -0
- package/.claude/skills/constitution-guardian/SKILL.md +306 -0
- package/.claude/skills/devflow-constitution-quick-ref/SKILL.md +374 -0
- package/.claude/skills/devflow-file-standards/SKILL.md +353 -0
- package/.claude/skills/devflow-tdd-enforcer/SKILL.md +192 -0
- package/.claude/skills/skill-developer/ADVANCED.md +197 -0
- package/.claude/skills/skill-developer/HOOK_MECHANISMS.md +306 -0
- package/.claude/skills/skill-developer/PATTERNS_LIBRARY.md +152 -0
- package/.claude/skills/skill-developer/SKILL.md +426 -0
- package/.claude/skills/skill-developer/SKILL_RULES_REFERENCE.md +315 -0
- package/.claude/skills/skill-developer/TRIGGER_TYPES.md +305 -0
- package/.claude/skills/skill-developer/TROUBLESHOOTING.md +514 -0
- package/.claude/skills/skill-rules.json +213 -0
- package/.claude/tests/README.md +300 -0
- package/.claude/tests/TODO.md +69 -0
- package/.claude/tests/__pycache__/test_analyze_upgrade_impact.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/__pycache__/test_consolidate_research.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/__pycache__/test_export_contracts.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/__pycache__/test_extract_data_model.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/__pycache__/test_generate_quickstart.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/__pycache__/test_generate_research_tasks.cpython-311-pytest-7.2.2.pyc +0 -0
- package/.claude/tests/constitution/run_all_constitution_tests.sh +111 -0
- package/.claude/tests/constitution/test_agent_assignment.sh +207 -0
- package/.claude/tests/constitution/test_article_coverage.sh +201 -0
- package/.claude/tests/constitution/test_template_completeness.sh +150 -0
- package/.claude/tests/constitution/test_version_consistency.sh +120 -0
- package/.claude/tests/fixtures/spec_delta_full.md +16 -0
- package/.claude/tests/fixtures/tasks_progress_sample.md +5 -0
- package/.claude/tests/run-all-tests.sh +229 -0
- package/.claude/tests/scripts/run.sh +30 -0
- package/.claude/tests/scripts/test-framework.sh +128 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh +511 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh.bak +504 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh.bak2 +505 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh.bak3 +506 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh.bak4 +507 -0
- package/.claude/tests/scripts/test_check_prerequisites.sh.bak5 +508 -0
- package/.claude/tests/scripts/test_check_task_status.sh +499 -0
- package/.claude/tests/scripts/test_common.sh +244 -0
- package/.claude/tests/scripts/test_generate_status_report.sh +71 -0
- package/.claude/tests/scripts/test_mark_task_complete.sh +441 -0
- package/.claude/tests/scripts/test_mark_task_complete.sh.backup +410 -0
- package/.claude/tests/scripts/test_recover_workflow.sh +304 -0
- package/.claude/tests/scripts/test_setup_epic.sh +437 -0
- package/.claude/tests/scripts/test_sync_task_marks.sh +196 -0
- package/.claude/tests/scripts/test_validate_constitution.sh +74 -0
- package/.claude/tests/scripts/test_validate_research.sh +462 -0
- package/.claude/tests/slugify.bats +82 -0
- package/.claude/tests/test-framework.sh +732 -0
- package/.claude/tests/test_analyze_upgrade_impact.py +34 -0
- package/.claude/tests/test_consolidate_research.py +48 -0
- package/.claude/tests/test_export_contracts.py +43 -0
- package/.claude/tests/test_extract_data_model.py +33 -0
- package/.claude/tests/test_generate_quickstart.py +50 -0
- package/.claude/tests/test_generate_research_tasks.py +52 -0
- package/.claude/tsc-cache/6e64f818-6398-49ca-8623-581a9af85c44/edited-files.log +1 -0
- package/.claude/tsc-cache/795ba6e3-b98a-423b-bab2-51aa62812569/affected-repos.txt +1 -0
- package/.claude/tsc-cache/795ba6e3-b98a-423b-bab2-51aa62812569/edited-files.log +1 -0
- package/.claude/tsc-cache/ae335694-be5a-4ba4-a1a0-b676c09a7906/affected-repos.txt +1 -0
- package/.claude/tsc-cache/ae335694-be5a-4ba4-a1a0-b676c09a7906/edited-files.log +1 -0
- package/CHANGELOG.md +507 -0
- package/LICENSE +21 -0
- package/README.md +534 -0
- package/README.zh-CN.md +530 -0
- package/bin/adapt.js +240 -0
- package/bin/cc-devflow-cli.js +185 -0
- package/bin/cc-devflow.js +78 -0
- package/config/adapters.yml +5 -0
- package/config/schema/adapters.schema.json +44 -0
- package/docs/CLAUDE.md +26 -0
- package/docs/commands/README.md +61 -0
- package/docs/commands/README.zh-CN.md +55 -0
- package/docs/commands/core-roadmap.md +106 -0
- package/docs/commands/core-roadmap.zh-CN.md +102 -0
- package/docs/commands/core-style.md +405 -0
- package/docs/commands/core-style.zh-CN.md +405 -0
- package/docs/commands/flow-init.md +134 -0
- package/docs/commands/flow-init.zh-CN.md +163 -0
- package/docs/commands/flow-new.md +274 -0
- package/docs/commands/flow-new.zh-CN.md +270 -0
- package/docs/guides/getting-started.md +204 -0
- package/docs/guides/getting-started.zh-CN.md +152 -0
- package/lib/adapters/adapter-interface.js +57 -0
- package/lib/adapters/claude-adapter.js +74 -0
- package/lib/adapters/codex-adapter.js +40 -0
- package/lib/adapters/config-validator.js +68 -0
- package/lib/adapters/logger.js +42 -0
- package/lib/adapters/registry.js +153 -0
- package/lib/compiler/CLAUDE.md +92 -0
- package/lib/compiler/__tests__/drift.test.js +215 -0
- package/lib/compiler/__tests__/errors.test.js +184 -0
- package/lib/compiler/__tests__/incremental.test.js +174 -0
- package/lib/compiler/__tests__/integration.test.js +174 -0
- package/lib/compiler/__tests__/manifest.test.js +233 -0
- package/lib/compiler/__tests__/parser.test.js +456 -0
- package/lib/compiler/__tests__/schemas.test.js +301 -0
- package/lib/compiler/__tests__/skills-registry.test.js +125 -0
- package/lib/compiler/__tests__/transformer.test.js +286 -0
- package/lib/compiler/emitters/antigravity-emitter.js +171 -0
- package/lib/compiler/emitters/base-emitter.js +73 -0
- package/lib/compiler/emitters/codex-emitter.js +52 -0
- package/lib/compiler/emitters/cursor-emitter.js +31 -0
- package/lib/compiler/emitters/index.js +50 -0
- package/lib/compiler/emitters/qwen-emitter.js +39 -0
- package/lib/compiler/errors.js +119 -0
- package/lib/compiler/index.js +256 -0
- package/lib/compiler/manifest.js +242 -0
- package/lib/compiler/parser.js +258 -0
- package/lib/compiler/platforms.js +113 -0
- package/lib/compiler/resource-copier.js +320 -0
- package/lib/compiler/rules-emitters/__tests__/antigravity-rules-emitter.test.js +191 -0
- package/lib/compiler/rules-emitters/__tests__/codex-rules-emitter.test.js +109 -0
- package/lib/compiler/rules-emitters/__tests__/cursor-rules-emitter.test.js +123 -0
- package/lib/compiler/rules-emitters/__tests__/qwen-rules-emitter.test.js +123 -0
- package/lib/compiler/rules-emitters/antigravity-rules-emitter.js +253 -0
- package/lib/compiler/rules-emitters/base-rules-emitter.js +83 -0
- package/lib/compiler/rules-emitters/codex-rules-emitter.js +116 -0
- package/lib/compiler/rules-emitters/cursor-rules-emitter.js +98 -0
- package/lib/compiler/rules-emitters/index.js +71 -0
- package/lib/compiler/rules-emitters/qwen-rules-emitter.js +70 -0
- package/lib/compiler/schemas.js +144 -0
- package/lib/compiler/skills-registry.js +225 -0
- package/lib/compiler/transformer.js +236 -0
- package/package.json +50 -0
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
|
|
3
|
+
# run_all_constitution_tests.sh - 运行所有 Constitution 测试套件
|
|
4
|
+
#
|
|
5
|
+
# 执行所有 Constitution 一致性验证测试
|
|
6
|
+
|
|
7
|
+
set -euo pipefail
|
|
8
|
+
|
|
9
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
10
|
+
|
|
11
|
+
# 颜色代码
|
|
12
|
+
RED='\033[0;31m'
|
|
13
|
+
GREEN='\033[0;32m'
|
|
14
|
+
YELLOW='\033[1;33m'
|
|
15
|
+
BLUE='\033[0;34m'
|
|
16
|
+
CYAN='\033[0;36m'
|
|
17
|
+
NC='\033[0m' # No Color
|
|
18
|
+
|
|
19
|
+
# ============================================================================
|
|
20
|
+
# 测试套件列表
|
|
21
|
+
# ============================================================================
|
|
22
|
+
|
|
23
|
+
declare -a TEST_SUITES=(
|
|
24
|
+
"test_version_consistency.sh"
|
|
25
|
+
"test_article_coverage.sh"
|
|
26
|
+
"test_template_completeness.sh"
|
|
27
|
+
"test_agent_assignment.sh"
|
|
28
|
+
)
|
|
29
|
+
|
|
30
|
+
# ============================================================================
|
|
31
|
+
# 运行测试
|
|
32
|
+
# ============================================================================
|
|
33
|
+
|
|
34
|
+
echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
|
|
35
|
+
echo -e "${CYAN} CC-DevFlow Constitution Test Suite${NC}"
|
|
36
|
+
echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
|
|
37
|
+
echo ""
|
|
38
|
+
|
|
39
|
+
total_suites=0
|
|
40
|
+
passed_suites=0
|
|
41
|
+
failed_suites=0
|
|
42
|
+
skipped_suites=0
|
|
43
|
+
|
|
44
|
+
for test_suite in "${TEST_SUITES[@]}"; do
|
|
45
|
+
test_file="$SCRIPT_DIR/$test_suite"
|
|
46
|
+
|
|
47
|
+
if [[ ! -f "$test_file" ]]; then
|
|
48
|
+
echo -e "${YELLOW}⚠${NC} Skipping $test_suite (not found)"
|
|
49
|
+
((skipped_suites++))
|
|
50
|
+
continue
|
|
51
|
+
fi
|
|
52
|
+
|
|
53
|
+
((total_suites++))
|
|
54
|
+
|
|
55
|
+
echo -e "${BLUE}▶${NC} Running $test_suite..."
|
|
56
|
+
echo ""
|
|
57
|
+
|
|
58
|
+
# 给脚本添加执行权限
|
|
59
|
+
chmod +x "$test_file"
|
|
60
|
+
|
|
61
|
+
# 运行测试
|
|
62
|
+
if bash "$test_file"; then
|
|
63
|
+
echo -e "${GREEN}✅${NC} $test_suite PASSED"
|
|
64
|
+
((passed_suites++))
|
|
65
|
+
else
|
|
66
|
+
echo -e "${RED}❌${NC} $test_suite FAILED"
|
|
67
|
+
((failed_suites++))
|
|
68
|
+
fi
|
|
69
|
+
|
|
70
|
+
echo ""
|
|
71
|
+
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
|
|
72
|
+
echo ""
|
|
73
|
+
done
|
|
74
|
+
|
|
75
|
+
# ============================================================================
|
|
76
|
+
# 总结报告
|
|
77
|
+
# ============================================================================
|
|
78
|
+
|
|
79
|
+
echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
|
|
80
|
+
echo -e "${CYAN} Test Summary${NC}"
|
|
81
|
+
echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
|
|
82
|
+
echo ""
|
|
83
|
+
echo " Total Test Suites: $total_suites"
|
|
84
|
+
echo -e " ${GREEN}Passed: $passed_suites${NC}"
|
|
85
|
+
if [[ $failed_suites -gt 0 ]]; then
|
|
86
|
+
echo -e " ${RED}Failed: $failed_suites${NC}"
|
|
87
|
+
fi
|
|
88
|
+
if [[ $skipped_suites -gt 0 ]]; then
|
|
89
|
+
echo -e " ${YELLOW}Skipped: $skipped_suites${NC}"
|
|
90
|
+
fi
|
|
91
|
+
echo ""
|
|
92
|
+
|
|
93
|
+
# 计算成功率
|
|
94
|
+
if [[ $total_suites -gt 0 ]]; then
|
|
95
|
+
success_rate=$((passed_suites * 100 / total_suites))
|
|
96
|
+
echo " Success Rate: $success_rate%"
|
|
97
|
+
echo ""
|
|
98
|
+
fi
|
|
99
|
+
|
|
100
|
+
if [[ $failed_suites -eq 0 ]]; then
|
|
101
|
+
echo -e "${GREEN}✅ All Constitution tests passed!${NC}"
|
|
102
|
+
echo ""
|
|
103
|
+
exit 0
|
|
104
|
+
else
|
|
105
|
+
echo -e "${RED}❌ Some Constitution tests failed${NC}"
|
|
106
|
+
echo ""
|
|
107
|
+
echo "Fix the failing tests and run again:"
|
|
108
|
+
echo " bash $0"
|
|
109
|
+
echo ""
|
|
110
|
+
exit 1
|
|
111
|
+
fi
|
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
|
|
3
|
+
# test_agent_assignment.sh - 测试代理 Article 分工
|
|
4
|
+
#
|
|
5
|
+
# 验证每个代理是否明确分配了负责的 Articles
|
|
6
|
+
|
|
7
|
+
set -euo pipefail
|
|
8
|
+
|
|
9
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
10
|
+
REPO_ROOT="$(cd "$SCRIPT_DIR/../../.." && pwd)"
|
|
11
|
+
|
|
12
|
+
# 加载测试框架
|
|
13
|
+
if [[ -f "$REPO_ROOT/.claude/tests/test-framework.sh" ]]; then
|
|
14
|
+
source "$REPO_ROOT/.claude/tests/test-framework.sh"
|
|
15
|
+
else
|
|
16
|
+
echo "❌ Test framework not found"
|
|
17
|
+
exit 1
|
|
18
|
+
fi
|
|
19
|
+
|
|
20
|
+
# ============================================================================
|
|
21
|
+
# 测试用例
|
|
22
|
+
# ============================================================================
|
|
23
|
+
|
|
24
|
+
test_prd_writer_enforces_article_I() {
|
|
25
|
+
# prd-writer 应该执行 Article I (Quality First)
|
|
26
|
+
local agent="$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
27
|
+
|
|
28
|
+
assert_file_exists "$agent" "prd-writer.md should exist"
|
|
29
|
+
assert_file_contains "$agent" "Article I" "prd-writer should reference Article I"
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
test_prd_writer_enforces_article_III() {
|
|
33
|
+
# prd-writer 应该执行 Article III (Security First)
|
|
34
|
+
local agent="$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
35
|
+
|
|
36
|
+
assert_file_contains "$agent" "Article III" "prd-writer should reference Article III"
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
test_prd_writer_enforces_article_X() {
|
|
40
|
+
# prd-writer 应该执行 Article X (Requirement Boundary)
|
|
41
|
+
local agent="$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
42
|
+
|
|
43
|
+
assert_file_contains "$agent" "Article X" "prd-writer should reference Article X"
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
test_planner_enforces_article_VII() {
|
|
47
|
+
# planner 应该执行 Article VII (Simplicity Gate)
|
|
48
|
+
local agent="$REPO_ROOT/.claude/agents/planner.md"
|
|
49
|
+
|
|
50
|
+
assert_file_exists "$agent" "planner.md should exist"
|
|
51
|
+
assert_file_contains "$agent" "Article VII" "planner should reference Article VII (Simplicity Gate)"
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
test_planner_enforces_article_VIII() {
|
|
55
|
+
# planner 应该执行 Article VIII (Anti-Abstraction)
|
|
56
|
+
local agent="$REPO_ROOT/.claude/agents/planner.md"
|
|
57
|
+
|
|
58
|
+
assert_file_contains "$agent" "Article VIII" "planner should reference Article VIII (Anti-Abstraction)"
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
test_planner_enforces_article_IX() {
|
|
62
|
+
# planner 应该执行 Article IX (Integration-First)
|
|
63
|
+
local agent="$REPO_ROOT/.claude/agents/planner.md"
|
|
64
|
+
|
|
65
|
+
assert_file_contains "$agent" "Article IX" "planner should reference Article IX (Integration-First)"
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
test_qa_tester_enforces_article_I() {
|
|
69
|
+
# qa-tester 应该执行 Article I (Quality First - Testing)
|
|
70
|
+
local agent="$REPO_ROOT/.claude/agents/qa-tester.md"
|
|
71
|
+
|
|
72
|
+
assert_file_exists "$agent" "qa-tester.md should exist"
|
|
73
|
+
assert_file_contains "$agent" "Article I" "qa-tester should reference Article I"
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
test_qa_tester_enforces_article_VI() {
|
|
77
|
+
# qa-tester 应该执行 Article VI (Test-First Development)
|
|
78
|
+
local agent="$REPO_ROOT/.claude/agents/qa-tester.md"
|
|
79
|
+
|
|
80
|
+
assert_file_contains "$agent" "Article VI" "qa-tester should reference Article VI (Test-First)"
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
test_security_reviewer_enforces_article_III() {
|
|
84
|
+
# security-reviewer 应该执行 Article III (Security First)
|
|
85
|
+
local agent="$REPO_ROOT/.claude/agents/security-reviewer.md"
|
|
86
|
+
|
|
87
|
+
assert_file_exists "$agent" "security-reviewer.md should exist"
|
|
88
|
+
assert_file_contains "$agent" "Article III" "security-reviewer should reference Article III"
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
test_security_reviewer_checks_hardcoded_secrets() {
|
|
92
|
+
# security-reviewer 应该明确检查 NO HARDCODED SECRETS (Article III.1)
|
|
93
|
+
local agent="$REPO_ROOT/.claude/agents/security-reviewer.md"
|
|
94
|
+
|
|
95
|
+
assert_file_contains "$agent" "NO HARDCODED SECRETS|Article III.1" "security-reviewer should check for hardcoded secrets"
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
test_all_agents_reference_constitution() {
|
|
99
|
+
local agents_dir="$REPO_ROOT/.claude/agents"
|
|
100
|
+
local agents_without_constitution=()
|
|
101
|
+
|
|
102
|
+
for agent in "$agents_dir"/*.md; do
|
|
103
|
+
if [[ -f "$agent" ]] && ! grep -q "Constitution" "$agent"; then
|
|
104
|
+
agents_without_constitution+=("$(basename "$agent")")
|
|
105
|
+
fi
|
|
106
|
+
done
|
|
107
|
+
|
|
108
|
+
# 允许一些代理不引用 Constitution (如 bug-analyzer, impact-analyzer 等工具型代理)
|
|
109
|
+
local allowed_without=(
|
|
110
|
+
"bug-analyzer.md"
|
|
111
|
+
"impact-analyzer.md"
|
|
112
|
+
"compatibility-checker.md"
|
|
113
|
+
"consistency-checker.md"
|
|
114
|
+
"release-manager.md"
|
|
115
|
+
)
|
|
116
|
+
|
|
117
|
+
# 检查数组是否为空(安全处理)
|
|
118
|
+
if [[ ${#agents_without_constitution[@]} -eq 0 ]]; then
|
|
119
|
+
# 所有代理都引用了 Constitution,测试通过
|
|
120
|
+
return 0
|
|
121
|
+
fi
|
|
122
|
+
|
|
123
|
+
for agent_name in "${agents_without_constitution[@]}"; do
|
|
124
|
+
local is_allowed=false
|
|
125
|
+
for allowed in "${allowed_without[@]}"; do
|
|
126
|
+
if [[ "$agent_name" == "$allowed" ]]; then
|
|
127
|
+
is_allowed=true
|
|
128
|
+
break
|
|
129
|
+
fi
|
|
130
|
+
done
|
|
131
|
+
|
|
132
|
+
if [[ "$is_allowed" == "false" ]]; then
|
|
133
|
+
log_error "Agent $agent_name does not reference Constitution"
|
|
134
|
+
return 1
|
|
135
|
+
fi
|
|
136
|
+
done
|
|
137
|
+
|
|
138
|
+
# 所有未引用 Constitution 的代理都在允许列表中
|
|
139
|
+
return 0
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
test_agents_reference_current_version() {
|
|
143
|
+
local constitution_version=$(grep -E '^\*\*Version\*\*:|^> \*\*Version\*\*:' "$REPO_ROOT/.claude/constitution/project-constitution.md" | head -1 | sed 's/.*v\([0-9.]*\).*/\1/')
|
|
144
|
+
|
|
145
|
+
local agents=(
|
|
146
|
+
"$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
147
|
+
"$REPO_ROOT/.claude/agents/planner.md"
|
|
148
|
+
"$REPO_ROOT/.claude/agents/qa-tester.md"
|
|
149
|
+
"$REPO_ROOT/.claude/agents/security-reviewer.md"
|
|
150
|
+
)
|
|
151
|
+
|
|
152
|
+
for agent in "${agents[@]}"; do
|
|
153
|
+
if grep -q "Constitution.*v" "$agent"; then
|
|
154
|
+
# 使用精确的版本号匹配(v数字.数字.数字)
|
|
155
|
+
local agent_version=$(grep -o "Constitution.*v[0-9]\+\.[0-9]\+\.[0-9]\+" "$agent" | head -1 | grep -o "v[0-9]\+\.[0-9]\+\.[0-9]\+" | sed 's/v//')
|
|
156
|
+
if [[ -n "$agent_version" ]]; then
|
|
157
|
+
assert_equals "$agent_version" "$constitution_version" "$(basename "$agent") should reference v$constitution_version"
|
|
158
|
+
fi
|
|
159
|
+
fi
|
|
160
|
+
done
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
test_agent_article_coverage_complete() {
|
|
164
|
+
# 验证所有 10 个 Articles 都有至少一个代理负责
|
|
165
|
+
|
|
166
|
+
local articles_with_agents=()
|
|
167
|
+
|
|
168
|
+
for article in I II III IV V VI VII VIII IX X; do
|
|
169
|
+
local has_agent=false
|
|
170
|
+
|
|
171
|
+
for agent in "$REPO_ROOT/.claude/agents"/*.md; do
|
|
172
|
+
if [[ -f "$agent" ]] && grep -q "Article $article" "$agent"; then
|
|
173
|
+
has_agent=true
|
|
174
|
+
break
|
|
175
|
+
fi
|
|
176
|
+
done
|
|
177
|
+
|
|
178
|
+
if [[ "$has_agent" == "true" ]]; then
|
|
179
|
+
articles_with_agents+=("$article")
|
|
180
|
+
else
|
|
181
|
+
log_warning "Article $article has no dedicated agent enforcement"
|
|
182
|
+
fi
|
|
183
|
+
done
|
|
184
|
+
|
|
185
|
+
# 至少要有 7 个 Articles 有代理负责 (允许一些 Articles 只在模板中执行)
|
|
186
|
+
local count=${#articles_with_agents[@]}
|
|
187
|
+
assert_gte "$count" "7" "At least 7 Articles should have agent enforcement (found $count)"
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
# ============================================================================
|
|
191
|
+
# 运行测试
|
|
192
|
+
# ============================================================================
|
|
193
|
+
|
|
194
|
+
run_tests \
|
|
195
|
+
test_prd_writer_enforces_article_I \
|
|
196
|
+
test_prd_writer_enforces_article_III \
|
|
197
|
+
test_prd_writer_enforces_article_X \
|
|
198
|
+
test_planner_enforces_article_VII \
|
|
199
|
+
test_planner_enforces_article_VIII \
|
|
200
|
+
test_planner_enforces_article_IX \
|
|
201
|
+
test_qa_tester_enforces_article_I \
|
|
202
|
+
test_qa_tester_enforces_article_VI \
|
|
203
|
+
test_security_reviewer_enforces_article_III \
|
|
204
|
+
test_security_reviewer_checks_hardcoded_secrets \
|
|
205
|
+
test_all_agents_reference_constitution \
|
|
206
|
+
test_agents_reference_current_version \
|
|
207
|
+
test_agent_article_coverage_complete
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
|
|
3
|
+
# test_article_coverage.sh - 测试 Article 覆盖率
|
|
4
|
+
#
|
|
5
|
+
# 验证每个 Article 是否都有执行点(enforcement points)
|
|
6
|
+
|
|
7
|
+
set -euo pipefail
|
|
8
|
+
|
|
9
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
10
|
+
REPO_ROOT="$(cd "$SCRIPT_DIR/../../.." && pwd)"
|
|
11
|
+
|
|
12
|
+
# 加载测试框架
|
|
13
|
+
if [[ -f "$REPO_ROOT/.claude/tests/test-framework.sh" ]]; then
|
|
14
|
+
source "$REPO_ROOT/.claude/tests/test-framework.sh"
|
|
15
|
+
else
|
|
16
|
+
echo "❌ Test framework not found"
|
|
17
|
+
exit 1
|
|
18
|
+
fi
|
|
19
|
+
|
|
20
|
+
# ============================================================================
|
|
21
|
+
# 测试用例
|
|
22
|
+
# ============================================================================
|
|
23
|
+
|
|
24
|
+
test_article_I_has_enforcement_points() {
|
|
25
|
+
# Article I: Quality First
|
|
26
|
+
# 应该在 PRD, EPIC, TASKS 模板和 prd-writer, qa-tester 代理中被引用
|
|
27
|
+
|
|
28
|
+
local count=0
|
|
29
|
+
local files=(
|
|
30
|
+
"$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
31
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
32
|
+
"$REPO_ROOT/.claude/docs/templates/TASKS_TEMPLATE.md"
|
|
33
|
+
"$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
34
|
+
"$REPO_ROOT/.claude/agents/qa-tester.md"
|
|
35
|
+
)
|
|
36
|
+
|
|
37
|
+
for file in "${files[@]}"; do
|
|
38
|
+
if [[ -f "$file" ]] && grep -q "Article I" "$file"; then
|
|
39
|
+
((count++))
|
|
40
|
+
fi
|
|
41
|
+
done
|
|
42
|
+
|
|
43
|
+
assert_gte "$count" "4" "Article I should have at least 4 enforcement points (found $count)"
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
test_article_III_has_enforcement_points() {
|
|
47
|
+
# Article III: Security First
|
|
48
|
+
# 应该在模板和 security-reviewer, prd-writer 代理中被引用
|
|
49
|
+
|
|
50
|
+
local count=0
|
|
51
|
+
local files=(
|
|
52
|
+
"$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
53
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
54
|
+
"$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
55
|
+
"$REPO_ROOT/.claude/agents/security-reviewer.md"
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
for file in "${files[@]}"; do
|
|
59
|
+
if [[ -f "$file" ]] && grep -q "Article III" "$file"; then
|
|
60
|
+
((count++))
|
|
61
|
+
fi
|
|
62
|
+
done
|
|
63
|
+
|
|
64
|
+
assert_gte "$count" "3" "Article III should have at least 3 enforcement points (found $count)"
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
test_article_VI_has_enforcement_points() {
|
|
68
|
+
# Article VI: Test-First Development
|
|
69
|
+
# 应该在 TASKS 模板和 qa-tester 代理中被引用
|
|
70
|
+
|
|
71
|
+
local count=0
|
|
72
|
+
local files=(
|
|
73
|
+
"$REPO_ROOT/.claude/docs/templates/TASKS_TEMPLATE.md"
|
|
74
|
+
"$REPO_ROOT/.claude/agents/qa-tester.md"
|
|
75
|
+
)
|
|
76
|
+
|
|
77
|
+
for file in "${files[@]}"; do
|
|
78
|
+
if [[ -f "$file" ]] && grep -q "Article VI" "$file"; then
|
|
79
|
+
((count++))
|
|
80
|
+
fi
|
|
81
|
+
done
|
|
82
|
+
|
|
83
|
+
assert_gte "$count" "2" "Article VI should have at least 2 enforcement points (found $count)"
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
test_article_VII_has_enforcement_points() {
|
|
87
|
+
# Article VII: Simplicity Gate
|
|
88
|
+
# 应该在 EPIC 模板和 planner 代理中被引用
|
|
89
|
+
|
|
90
|
+
local count=0
|
|
91
|
+
local files=(
|
|
92
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
93
|
+
"$REPO_ROOT/.claude/agents/planner.md"
|
|
94
|
+
)
|
|
95
|
+
|
|
96
|
+
for file in "${files[@]}"; do
|
|
97
|
+
if [[ -f "$file" ]] && grep -q "Article VII" "$file"; then
|
|
98
|
+
((count++))
|
|
99
|
+
fi
|
|
100
|
+
done
|
|
101
|
+
|
|
102
|
+
assert_gte "$count" "2" "Article VII should have at least 2 enforcement points (found $count)"
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
test_article_VIII_has_enforcement_points() {
|
|
106
|
+
# Article VIII: Anti-Abstraction
|
|
107
|
+
# 应该在 EPIC 模板和 planner 代理中被引用
|
|
108
|
+
|
|
109
|
+
local count=0
|
|
110
|
+
local files=(
|
|
111
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
112
|
+
"$REPO_ROOT/.claude/agents/planner.md"
|
|
113
|
+
)
|
|
114
|
+
|
|
115
|
+
for file in "${files[@]}"; do
|
|
116
|
+
if [[ -f "$file" ]] && grep -q "Article VIII" "$file"; then
|
|
117
|
+
((count++))
|
|
118
|
+
fi
|
|
119
|
+
done
|
|
120
|
+
|
|
121
|
+
assert_gte "$count" "2" "Article VIII should have at least 2 enforcement points (found $count)"
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
test_article_IX_has_enforcement_points() {
|
|
125
|
+
# Article IX: Integration-First Testing
|
|
126
|
+
# 应该在 EPIC 模板和 planner 代理中被引用
|
|
127
|
+
|
|
128
|
+
local count=0
|
|
129
|
+
local files=(
|
|
130
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
131
|
+
"$REPO_ROOT/.claude/agents/planner.md"
|
|
132
|
+
)
|
|
133
|
+
|
|
134
|
+
for file in "${files[@]}"; do
|
|
135
|
+
if [[ -f "$file" ]] && grep -q "Article IX" "$file"; then
|
|
136
|
+
((count++))
|
|
137
|
+
fi
|
|
138
|
+
done
|
|
139
|
+
|
|
140
|
+
assert_gte "$count" "2" "Article IX should have at least 2 enforcement points (found $count)"
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
test_article_X_has_enforcement_points() {
|
|
144
|
+
# Article X: Requirement Boundary
|
|
145
|
+
# 应该在 PRD 模板和 prd-writer 代理中被引用
|
|
146
|
+
|
|
147
|
+
local count=0
|
|
148
|
+
local files=(
|
|
149
|
+
"$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
150
|
+
"$REPO_ROOT/.claude/agents/prd-writer.md"
|
|
151
|
+
"$REPO_ROOT/.claude/scripts/validate-scope-boundary.sh"
|
|
152
|
+
)
|
|
153
|
+
|
|
154
|
+
for file in "${files[@]}"; do
|
|
155
|
+
if [[ -f "$file" ]] && grep -q "Article X" "$file"; then
|
|
156
|
+
((count++))
|
|
157
|
+
fi
|
|
158
|
+
done
|
|
159
|
+
|
|
160
|
+
assert_gte "$count" "2" "Article X should have at least 2 enforcement points (found $count)"
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
test_all_articles_documented_in_constitution() {
|
|
164
|
+
local constitution="$REPO_ROOT/.claude/constitution/project-constitution.md"
|
|
165
|
+
|
|
166
|
+
for article in I II III IV V VI VII VIII IX X; do
|
|
167
|
+
assert_file_contains "$constitution" "## Article $article:" "Constitution should document Article $article"
|
|
168
|
+
done
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
test_all_articles_documented_in_claude_md() {
|
|
172
|
+
local claude_md="$REPO_ROOT/CLAUDE.md"
|
|
173
|
+
|
|
174
|
+
for article in I II III IV V VI VII VIII IX X; do
|
|
175
|
+
assert_file_contains "$claude_md" "Article $article" "CLAUDE.md should mention Article $article"
|
|
176
|
+
done
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
test_impact_report_shows_all_articles() {
|
|
180
|
+
local output=$(bash "$REPO_ROOT/.claude/scripts/manage-constitution.sh" impact 2>&1)
|
|
181
|
+
|
|
182
|
+
for article in I II III IV V VI VII VIII IX X; do
|
|
183
|
+
assert_contains "$output" "Article $article:" "Impact report should include Article $article"
|
|
184
|
+
done
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
# ============================================================================
|
|
188
|
+
# 运行测试
|
|
189
|
+
# ============================================================================
|
|
190
|
+
|
|
191
|
+
run_tests \
|
|
192
|
+
test_article_I_has_enforcement_points \
|
|
193
|
+
test_article_III_has_enforcement_points \
|
|
194
|
+
test_article_VI_has_enforcement_points \
|
|
195
|
+
test_article_VII_has_enforcement_points \
|
|
196
|
+
test_article_VIII_has_enforcement_points \
|
|
197
|
+
test_article_IX_has_enforcement_points \
|
|
198
|
+
test_article_X_has_enforcement_points \
|
|
199
|
+
test_all_articles_documented_in_constitution \
|
|
200
|
+
test_all_articles_documented_in_claude_md \
|
|
201
|
+
test_impact_report_shows_all_articles
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
|
|
3
|
+
# test_template_completeness.sh - 测试模板完整性
|
|
4
|
+
#
|
|
5
|
+
# 验证所有模板是否包含必需的 Constitution Check 部分
|
|
6
|
+
|
|
7
|
+
set -euo pipefail
|
|
8
|
+
|
|
9
|
+
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
10
|
+
REPO_ROOT="$(cd "$SCRIPT_DIR/../../.." && pwd)"
|
|
11
|
+
|
|
12
|
+
# 加载测试框架
|
|
13
|
+
if [[ -f "$REPO_ROOT/.claude/tests/test-framework.sh" ]]; then
|
|
14
|
+
source "$REPO_ROOT/.claude/tests/test-framework.sh"
|
|
15
|
+
else
|
|
16
|
+
echo "❌ Test framework not found"
|
|
17
|
+
exit 1
|
|
18
|
+
fi
|
|
19
|
+
|
|
20
|
+
# ============================================================================
|
|
21
|
+
# 测试用例
|
|
22
|
+
# ============================================================================
|
|
23
|
+
|
|
24
|
+
test_prd_template_has_constitution_check() {
|
|
25
|
+
local template="$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
26
|
+
|
|
27
|
+
assert_file_exists "$template" "PRD_TEMPLATE.md should exist"
|
|
28
|
+
assert_file_contains "$template" "## Constitution Check" "PRD template should have Constitution Check section"
|
|
29
|
+
assert_file_contains "$template" ".claude/constitution/project-constitution.md" "PRD template should reference Constitution file"
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
test_epic_template_has_constitution_check() {
|
|
33
|
+
local template="$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
34
|
+
|
|
35
|
+
assert_file_exists "$template" "EPIC_TEMPLATE.md should exist"
|
|
36
|
+
assert_file_contains "$template" "## Constitution Check" "EPIC template should have Constitution Check section"
|
|
37
|
+
assert_file_contains "$template" ".claude/constitution/project-constitution.md" "EPIC template should reference Constitution file"
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
test_tasks_template_has_constitution_alignment() {
|
|
41
|
+
local template="$REPO_ROOT/.claude/docs/templates/TASKS_TEMPLATE.md"
|
|
42
|
+
|
|
43
|
+
assert_file_exists "$template" "TASKS_TEMPLATE.md should exist"
|
|
44
|
+
assert_file_contains "$template" "Constitution Alignment|Constitution Compliance" "TASKS template should have Constitution section"
|
|
45
|
+
assert_file_contains "$template" ".claude/constitution/project-constitution.md" "TASKS template should reference Constitution file"
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
test_prd_template_has_article_references() {
|
|
49
|
+
local template="$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
50
|
+
|
|
51
|
+
# PRD 应该引用 Article I, III, X
|
|
52
|
+
assert_file_contains "$template" "Article I" "PRD template should reference Article I (Quality)"
|
|
53
|
+
assert_file_contains "$template" "Article III" "PRD template should reference Article III (Security)"
|
|
54
|
+
assert_file_contains "$template" "Article X" "PRD template should reference Article X (Requirement Boundary)"
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
test_epic_template_has_phase_minus_one_gates() {
|
|
58
|
+
local template="$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
59
|
+
|
|
60
|
+
# EPIC 应该有 Phase -1 Gates
|
|
61
|
+
assert_file_contains "$template" "Phase -1.*Gate|Phase -1.*Constitutional" "EPIC template should have Phase -1 Gates"
|
|
62
|
+
assert_file_contains "$template" "Article VII" "EPIC template should reference Article VII (Simplicity Gate)"
|
|
63
|
+
assert_file_contains "$template" "Article VIII" "EPIC template should reference Article VIII (Anti-Abstraction)"
|
|
64
|
+
assert_file_contains "$template" "Article IX" "EPIC template should reference Article IX (Integration-First)"
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
test_tasks_template_has_tdd_enforcement() {
|
|
68
|
+
local template="$REPO_ROOT/.claude/docs/templates/TASKS_TEMPLATE.md"
|
|
69
|
+
|
|
70
|
+
# TASKS 应该强制执行 TDD (Article VI)
|
|
71
|
+
assert_file_contains "$template" "Article VI" "TASKS template should reference Article VI (TDD)"
|
|
72
|
+
assert_file_contains "$template" "Test.*First|TDD" "TASKS template should mention Test-First or TDD"
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
test_epic_template_has_complexity_tracking() {
|
|
76
|
+
local template="$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
77
|
+
|
|
78
|
+
# EPIC 应该有 Complexity Tracking 表格
|
|
79
|
+
assert_file_contains "$template" "Complexity Tracking" "EPIC template should have Complexity Tracking section"
|
|
80
|
+
assert_file_contains "$template" "Constitutional Violations|违规" "EPIC template should have violations table"
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
test_all_templates_reference_current_version() {
|
|
84
|
+
local constitution_version=$(grep -E '^\*\*Version\*\*:|^> \*\*Version\*\*:' "$REPO_ROOT/.claude/constitution/project-constitution.md" | head -1 | sed 's/.*v\([0-9.]*\).*/\1/')
|
|
85
|
+
|
|
86
|
+
local templates=(
|
|
87
|
+
"$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
88
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
89
|
+
"$REPO_ROOT/.claude/docs/templates/TASKS_TEMPLATE.md"
|
|
90
|
+
)
|
|
91
|
+
|
|
92
|
+
for template in "${templates[@]}"; do
|
|
93
|
+
if grep -q "Constitution.*v" "$template"; then
|
|
94
|
+
# 使用精确的版本号匹配(v数字.数字.数字)
|
|
95
|
+
local template_version=$(grep -o "Constitution.*v[0-9]\+\.[0-9]\+\.[0-9]\+" "$template" | head -1 | grep -o "v[0-9]\+\.[0-9]\+\.[0-9]\+" | sed 's/v//')
|
|
96
|
+
if [[ -n "$template_version" ]]; then
|
|
97
|
+
assert_equals "$template_version" "$constitution_version" "$(basename "$template") should reference v$constitution_version"
|
|
98
|
+
fi
|
|
99
|
+
fi
|
|
100
|
+
done
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
test_templates_have_validation_checklists() {
|
|
104
|
+
local templates=(
|
|
105
|
+
"$REPO_ROOT/.claude/docs/templates/PRD_TEMPLATE.md"
|
|
106
|
+
"$REPO_ROOT/.claude/docs/templates/EPIC_TEMPLATE.md"
|
|
107
|
+
)
|
|
108
|
+
|
|
109
|
+
for template in "${templates[@]}"; do
|
|
110
|
+
# 检查是否有 checklist 格式的 Constitution checks
|
|
111
|
+
if grep -qE "Constitution Check|Constitution Compliance" "$template"; then
|
|
112
|
+
# 检查是否有 Article 相关的 checklist(罗马数字格式: I., II., III. 等)
|
|
113
|
+
local checklist_count=$(grep -cE "^- \[ \] \*\*(I{1,3}|IV|V|VI{0,3}|IX|X)\." "$template" 2>/dev/null || echo "0")
|
|
114
|
+
# 去除可能的空白字符和换行
|
|
115
|
+
checklist_count=$(echo "$checklist_count" | tr -d ' \n\r')
|
|
116
|
+
assert_gt "$checklist_count" "0" "$(basename "$template") should have Article checklist items"
|
|
117
|
+
fi
|
|
118
|
+
done
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
test_no_orphaned_old_templates() {
|
|
122
|
+
# 检查是否有旧版模板文件(如 v2.1.1 的 TASKS_TEMPLATE_OLD.md)
|
|
123
|
+
local templates_dir="$REPO_ROOT/.claude/docs/templates"
|
|
124
|
+
local old_templates=$(find "$templates_dir" -name "*_OLD.md" -o -name "*_old.md" -o -name "*.backup.md" 2>/dev/null || echo "")
|
|
125
|
+
|
|
126
|
+
if [[ -n "$old_templates" ]]; then
|
|
127
|
+
log_warning "Found old template files: $old_templates"
|
|
128
|
+
log_warning "Consider cleaning up old templates to avoid version confusion"
|
|
129
|
+
return 1
|
|
130
|
+
fi
|
|
131
|
+
|
|
132
|
+
# 如果没有旧模板,测试通过
|
|
133
|
+
return 0
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
# ============================================================================
|
|
137
|
+
# 运行测试
|
|
138
|
+
# ============================================================================
|
|
139
|
+
|
|
140
|
+
run_tests \
|
|
141
|
+
test_prd_template_has_constitution_check \
|
|
142
|
+
test_epic_template_has_constitution_check \
|
|
143
|
+
test_tasks_template_has_constitution_alignment \
|
|
144
|
+
test_prd_template_has_article_references \
|
|
145
|
+
test_epic_template_has_phase_minus_one_gates \
|
|
146
|
+
test_tasks_template_has_tdd_enforcement \
|
|
147
|
+
test_epic_template_has_complexity_tracking \
|
|
148
|
+
test_all_templates_reference_current_version \
|
|
149
|
+
test_templates_have_validation_checklists \
|
|
150
|
+
test_no_orphaned_old_templates
|