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.
Files changed (277) hide show
  1. package/.claude/CLAUDE.md +83 -0
  2. package/.claude/agents/architecture-designer.md +443 -0
  3. package/.claude/agents/bug-analyzer.md +382 -0
  4. package/.claude/agents/checklist-agent.md +175 -0
  5. package/.claude/agents/clarify-analyst.md +50 -0
  6. package/.claude/agents/code-reviewer.md +71 -0
  7. package/.claude/agents/codex-analyzer.md +39 -0
  8. package/.claude/agents/compatibility-checker.md +580 -0
  9. package/.claude/agents/consistency-checker.md +532 -0
  10. package/.claude/agents/impact-analyzer.md +441 -0
  11. package/.claude/agents/planner.md +230 -0
  12. package/.claude/agents/prd-writer.md +320 -0
  13. package/.claude/agents/project-guidelines-generator.md +1329 -0
  14. package/.claude/agents/qa-tester.md +313 -0
  15. package/.claude/agents/release-manager.md +295 -0
  16. package/.claude/agents/security-reviewer.md +314 -0
  17. package/.claude/agents/style-guide-generator.md +458 -0
  18. package/.claude/agents/tech-architect.md +516 -0
  19. package/.claude/agents/ui-designer.md +485 -0
  20. package/.claude/commands/code-review-high.md +58 -0
  21. package/.claude/commands/core-architecture.md +429 -0
  22. package/.claude/commands/core-guidelines.md +486 -0
  23. package/.claude/commands/core-roadmap.md +439 -0
  24. package/.claude/commands/core-style.md +293 -0
  25. package/.claude/commands/flow-archive.md +245 -0
  26. package/.claude/commands/flow-checklist.md +260 -0
  27. package/.claude/commands/flow-clarify.md +136 -0
  28. package/.claude/commands/flow-constitution.md +82 -0
  29. package/.claude/commands/flow-dev.md +134 -0
  30. package/.claude/commands/flow-epic.md +150 -0
  31. package/.claude/commands/flow-fix.md +104 -0
  32. package/.claude/commands/flow-ideate.md +214 -0
  33. package/.claude/commands/flow-init.md +313 -0
  34. package/.claude/commands/flow-new.md +394 -0
  35. package/.claude/commands/flow-prd.md +131 -0
  36. package/.claude/commands/flow-qa.md +93 -0
  37. package/.claude/commands/flow-release.md +92 -0
  38. package/.claude/commands/flow-restart.md +98 -0
  39. package/.claude/commands/flow-status.md +64 -0
  40. package/.claude/commands/flow-tech.md +142 -0
  41. package/.claude/commands/flow-ui.md +189 -0
  42. package/.claude/commands/flow-update.md +111 -0
  43. package/.claude/commands/flow-upgrade.md +115 -0
  44. package/.claude/commands/flow-verify.md +96 -0
  45. package/.claude/commands/problem-analyzer.md +60 -0
  46. package/.claude/config/quality-rules.yml +161 -0
  47. package/.claude/docs/SPEC_KIT_CONSTITUTION_ANALYSIS.md +426 -0
  48. package/.claude/docs/design/consistency-conflict-detection-algorithms.md +658 -0
  49. package/.claude/docs/design/intent-driven-input-design.md +380 -0
  50. package/.claude/docs/design/prd-version-management-design.md +437 -0
  51. package/.claude/docs/guides/INIT_TROUBLESHOOTING.md +117 -0
  52. package/.claude/docs/guides/NEW_TROUBLESHOOTING.md +151 -0
  53. package/.claude/docs/guides/ROADMAP_TROUBLESHOOTING.md +188 -0
  54. package/.claude/docs/guides/TASK_COMPLETION_MARKING.md +338 -0
  55. package/.claude/docs/templates/ARCHITECTURE_TEMPLATE.md +633 -0
  56. package/.claude/docs/templates/BACKLOG_TEMPLATE.md +261 -0
  57. package/.claude/docs/templates/CHECKLIST_TEMPLATE.md +52 -0
  58. package/.claude/docs/templates/CLARIFICATION_REPORT_TEMPLATE.md +206 -0
  59. package/.claude/docs/templates/CODE_REVIEW_TEMPLATE.md +71 -0
  60. package/.claude/docs/templates/EPIC_TEMPLATE.md +805 -0
  61. package/.claude/docs/templates/INIT_FLOW_TEMPLATE.md +213 -0
  62. package/.claude/docs/templates/INTENT_CLARIFICATION_TEMPLATE.md +57 -0
  63. package/.claude/docs/templates/NEW_ORCHESTRATION_TEMPLATE.md +148 -0
  64. package/.claude/docs/templates/PRD_TEMPLATE.md +562 -0
  65. package/.claude/docs/templates/RESEARCH_TEMPLATE.md +276 -0
  66. package/.claude/docs/templates/REVIEW-HIGH.md +57 -0
  67. package/.claude/docs/templates/ROADMAP_DIALOGUE_TEMPLATE.md +198 -0
  68. package/.claude/docs/templates/ROADMAP_TEMPLATE.md +310 -0
  69. package/.claude/docs/templates/STYLE_TEMPLATE.md +1266 -0
  70. package/.claude/docs/templates/TASKS_TEMPLATE.md +523 -0
  71. package/.claude/docs/templates/TECH_DESIGN_TEMPLATE.md +1019 -0
  72. package/.claude/docs/templates/UI_PROTOTYPE_TEMPLATE.md +1436 -0
  73. package/.claude/guides/agent-guides/agent-coordination-guide.md +459 -0
  74. package/.claude/guides/project-guidelines-system.md +463 -0
  75. package/.claude/guides/technical-guides/datetime-handling-guide.md +563 -0
  76. package/.claude/guides/technical-guides/git-github-guide.md +642 -0
  77. package/.claude/guides/technical-guides/test-execution-guide.md +618 -0
  78. package/.claude/guides/workflow-guides/bug-fix-orchestrator.md +217 -0
  79. package/.claude/guides/workflow-guides/flow-orchestrator.md +282 -0
  80. package/.claude/hooks/checklist-gate.js +397 -0
  81. package/.claude/hooks/error-handling-reminder.sh +12 -0
  82. package/.claude/hooks/error-handling-reminder.ts +459 -0
  83. package/.claude/hooks/post-tool-use-tracker.sh +280 -0
  84. package/.claude/hooks/pre-tool-use-guardrail.sh +36 -0
  85. package/.claude/hooks/pre-tool-use-guardrail.ts +342 -0
  86. package/.claude/hooks/skill-activation-prompt.sh +36 -0
  87. package/.claude/hooks/skill-activation-prompt.ts +214 -0
  88. package/.claude/hooks/state/skills-used-test-guard.json +3 -0
  89. package/.claude/rules/devflow-conventions.md +305 -0
  90. package/.claude/rules/project-constitution.md +748 -0
  91. package/.claude/schemas/constitution.schema.json +43 -0
  92. package/.claude/scripts/analyze-upgrade-impact.sh +200 -0
  93. package/.claude/scripts/archive-requirement.sh +351 -0
  94. package/.claude/scripts/calculate-checklist-completion.sh +243 -0
  95. package/.claude/scripts/calculate-quarter.sh +206 -0
  96. package/.claude/scripts/check-dependencies.sh +409 -0
  97. package/.claude/scripts/check-prerequisites.sh +232 -0
  98. package/.claude/scripts/check-task-status.sh +264 -0
  99. package/.claude/scripts/checklist-errors.sh +131 -0
  100. package/.claude/scripts/common.sh +570 -0
  101. package/.claude/scripts/consolidate-research.sh +182 -0
  102. package/.claude/scripts/create-requirement.sh +426 -0
  103. package/.claude/scripts/export-contracts.sh +117 -0
  104. package/.claude/scripts/extract-data-model.sh +78 -0
  105. package/.claude/scripts/generate-clarification-questions.sh +377 -0
  106. package/.claude/scripts/generate-clarification-report.sh +463 -0
  107. package/.claude/scripts/generate-quickstart.sh +146 -0
  108. package/.claude/scripts/generate-research-tasks.sh +157 -0
  109. package/.claude/scripts/generate-status-report.sh +523 -0
  110. package/.claude/scripts/generate-tech-analysis.sh +46 -0
  111. package/.claude/scripts/locate-requirement-in-roadmap.sh +233 -0
  112. package/.claude/scripts/manage-constitution.sh +602 -0
  113. package/.claude/scripts/mark-task-complete.sh +198 -0
  114. package/.claude/scripts/populate-research-tasks.sh +259 -0
  115. package/.claude/scripts/recover-workflow.sh +460 -0
  116. package/.claude/scripts/run-clarify-scan.sh +601 -0
  117. package/.claude/scripts/run-high-review.sh +62 -0
  118. package/.claude/scripts/run-problem-analysis.sh +68 -0
  119. package/.claude/scripts/setup-epic.sh +173 -0
  120. package/.claude/scripts/sync-roadmap-progress.sh +300 -0
  121. package/.claude/scripts/sync-task-marks.sh +199 -0
  122. package/.claude/scripts/test-clarify-scan.sh +515 -0
  123. package/.claude/scripts/update-agent-context.sh +806 -0
  124. package/.claude/scripts/validate-constitution.sh +567 -0
  125. package/.claude/scripts/validate-hooks.sh +487 -0
  126. package/.claude/scripts/validate-research.sh +332 -0
  127. package/.claude/scripts/validate-scope-boundary.sh +493 -0
  128. package/.claude/scripts/verify-setup.sh +37 -0
  129. package/.claude/settings.json +76 -0
  130. package/.claude/skills/_reference-implementations/README.md +96 -0
  131. package/.claude/skills/_reference-implementations/backend-express-prisma/SKILL.md +302 -0
  132. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/architecture-overview.md +451 -0
  133. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/async-and-errors.md +307 -0
  134. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/complete-examples.md +638 -0
  135. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/configuration.md +275 -0
  136. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/database-patterns.md +224 -0
  137. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/middleware-guide.md +213 -0
  138. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/routing-and-controllers.md +756 -0
  139. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/sentry-and-monitoring.md +336 -0
  140. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/services-and-repositories.md +789 -0
  141. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/testing-guide.md +235 -0
  142. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/validation-patterns.md +754 -0
  143. package/.claude/skills/_reference-implementations/frontend-react-mui/SKILL.md +399 -0
  144. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/common-patterns.md +331 -0
  145. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/complete-examples.md +872 -0
  146. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/component-patterns.md +502 -0
  147. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/data-fetching.md +767 -0
  148. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/file-organization.md +502 -0
  149. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/loading-and-error-states.md +501 -0
  150. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/performance.md +406 -0
  151. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/routing-guide.md +364 -0
  152. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/styling-guide.md +428 -0
  153. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/typescript-standards.md +418 -0
  154. package/.claude/skills/cc-devflow-orchestrator/SKILL.md +229 -0
  155. package/.claude/skills/constitution-guardian/SKILL.md +306 -0
  156. package/.claude/skills/devflow-constitution-quick-ref/SKILL.md +374 -0
  157. package/.claude/skills/devflow-file-standards/SKILL.md +353 -0
  158. package/.claude/skills/devflow-tdd-enforcer/SKILL.md +192 -0
  159. package/.claude/skills/skill-developer/ADVANCED.md +197 -0
  160. package/.claude/skills/skill-developer/HOOK_MECHANISMS.md +306 -0
  161. package/.claude/skills/skill-developer/PATTERNS_LIBRARY.md +152 -0
  162. package/.claude/skills/skill-developer/SKILL.md +426 -0
  163. package/.claude/skills/skill-developer/SKILL_RULES_REFERENCE.md +315 -0
  164. package/.claude/skills/skill-developer/TRIGGER_TYPES.md +305 -0
  165. package/.claude/skills/skill-developer/TROUBLESHOOTING.md +514 -0
  166. package/.claude/skills/skill-rules.json +213 -0
  167. package/.claude/tests/README.md +300 -0
  168. package/.claude/tests/TODO.md +69 -0
  169. package/.claude/tests/__pycache__/test_analyze_upgrade_impact.cpython-311-pytest-7.2.2.pyc +0 -0
  170. package/.claude/tests/__pycache__/test_consolidate_research.cpython-311-pytest-7.2.2.pyc +0 -0
  171. package/.claude/tests/__pycache__/test_export_contracts.cpython-311-pytest-7.2.2.pyc +0 -0
  172. package/.claude/tests/__pycache__/test_extract_data_model.cpython-311-pytest-7.2.2.pyc +0 -0
  173. package/.claude/tests/__pycache__/test_generate_quickstart.cpython-311-pytest-7.2.2.pyc +0 -0
  174. package/.claude/tests/__pycache__/test_generate_research_tasks.cpython-311-pytest-7.2.2.pyc +0 -0
  175. package/.claude/tests/constitution/run_all_constitution_tests.sh +111 -0
  176. package/.claude/tests/constitution/test_agent_assignment.sh +207 -0
  177. package/.claude/tests/constitution/test_article_coverage.sh +201 -0
  178. package/.claude/tests/constitution/test_template_completeness.sh +150 -0
  179. package/.claude/tests/constitution/test_version_consistency.sh +120 -0
  180. package/.claude/tests/fixtures/spec_delta_full.md +16 -0
  181. package/.claude/tests/fixtures/tasks_progress_sample.md +5 -0
  182. package/.claude/tests/run-all-tests.sh +229 -0
  183. package/.claude/tests/scripts/run.sh +30 -0
  184. package/.claude/tests/scripts/test-framework.sh +128 -0
  185. package/.claude/tests/scripts/test_check_prerequisites.sh +511 -0
  186. package/.claude/tests/scripts/test_check_prerequisites.sh.bak +504 -0
  187. package/.claude/tests/scripts/test_check_prerequisites.sh.bak2 +505 -0
  188. package/.claude/tests/scripts/test_check_prerequisites.sh.bak3 +506 -0
  189. package/.claude/tests/scripts/test_check_prerequisites.sh.bak4 +507 -0
  190. package/.claude/tests/scripts/test_check_prerequisites.sh.bak5 +508 -0
  191. package/.claude/tests/scripts/test_check_task_status.sh +499 -0
  192. package/.claude/tests/scripts/test_common.sh +244 -0
  193. package/.claude/tests/scripts/test_generate_status_report.sh +71 -0
  194. package/.claude/tests/scripts/test_mark_task_complete.sh +441 -0
  195. package/.claude/tests/scripts/test_mark_task_complete.sh.backup +410 -0
  196. package/.claude/tests/scripts/test_recover_workflow.sh +304 -0
  197. package/.claude/tests/scripts/test_setup_epic.sh +437 -0
  198. package/.claude/tests/scripts/test_sync_task_marks.sh +196 -0
  199. package/.claude/tests/scripts/test_validate_constitution.sh +74 -0
  200. package/.claude/tests/scripts/test_validate_research.sh +462 -0
  201. package/.claude/tests/slugify.bats +82 -0
  202. package/.claude/tests/test-framework.sh +732 -0
  203. package/.claude/tests/test_analyze_upgrade_impact.py +34 -0
  204. package/.claude/tests/test_consolidate_research.py +48 -0
  205. package/.claude/tests/test_export_contracts.py +43 -0
  206. package/.claude/tests/test_extract_data_model.py +33 -0
  207. package/.claude/tests/test_generate_quickstart.py +50 -0
  208. package/.claude/tests/test_generate_research_tasks.py +52 -0
  209. package/.claude/tsc-cache/6e64f818-6398-49ca-8623-581a9af85c44/edited-files.log +1 -0
  210. package/.claude/tsc-cache/795ba6e3-b98a-423b-bab2-51aa62812569/affected-repos.txt +1 -0
  211. package/.claude/tsc-cache/795ba6e3-b98a-423b-bab2-51aa62812569/edited-files.log +1 -0
  212. package/.claude/tsc-cache/ae335694-be5a-4ba4-a1a0-b676c09a7906/affected-repos.txt +1 -0
  213. package/.claude/tsc-cache/ae335694-be5a-4ba4-a1a0-b676c09a7906/edited-files.log +1 -0
  214. package/CHANGELOG.md +507 -0
  215. package/LICENSE +21 -0
  216. package/README.md +534 -0
  217. package/README.zh-CN.md +530 -0
  218. package/bin/adapt.js +240 -0
  219. package/bin/cc-devflow-cli.js +185 -0
  220. package/bin/cc-devflow.js +78 -0
  221. package/config/adapters.yml +5 -0
  222. package/config/schema/adapters.schema.json +44 -0
  223. package/docs/CLAUDE.md +26 -0
  224. package/docs/commands/README.md +61 -0
  225. package/docs/commands/README.zh-CN.md +55 -0
  226. package/docs/commands/core-roadmap.md +106 -0
  227. package/docs/commands/core-roadmap.zh-CN.md +102 -0
  228. package/docs/commands/core-style.md +405 -0
  229. package/docs/commands/core-style.zh-CN.md +405 -0
  230. package/docs/commands/flow-init.md +134 -0
  231. package/docs/commands/flow-init.zh-CN.md +163 -0
  232. package/docs/commands/flow-new.md +274 -0
  233. package/docs/commands/flow-new.zh-CN.md +270 -0
  234. package/docs/guides/getting-started.md +204 -0
  235. package/docs/guides/getting-started.zh-CN.md +152 -0
  236. package/lib/adapters/adapter-interface.js +57 -0
  237. package/lib/adapters/claude-adapter.js +74 -0
  238. package/lib/adapters/codex-adapter.js +40 -0
  239. package/lib/adapters/config-validator.js +68 -0
  240. package/lib/adapters/logger.js +42 -0
  241. package/lib/adapters/registry.js +153 -0
  242. package/lib/compiler/CLAUDE.md +92 -0
  243. package/lib/compiler/__tests__/drift.test.js +215 -0
  244. package/lib/compiler/__tests__/errors.test.js +184 -0
  245. package/lib/compiler/__tests__/incremental.test.js +174 -0
  246. package/lib/compiler/__tests__/integration.test.js +174 -0
  247. package/lib/compiler/__tests__/manifest.test.js +233 -0
  248. package/lib/compiler/__tests__/parser.test.js +456 -0
  249. package/lib/compiler/__tests__/schemas.test.js +301 -0
  250. package/lib/compiler/__tests__/skills-registry.test.js +125 -0
  251. package/lib/compiler/__tests__/transformer.test.js +286 -0
  252. package/lib/compiler/emitters/antigravity-emitter.js +171 -0
  253. package/lib/compiler/emitters/base-emitter.js +73 -0
  254. package/lib/compiler/emitters/codex-emitter.js +52 -0
  255. package/lib/compiler/emitters/cursor-emitter.js +31 -0
  256. package/lib/compiler/emitters/index.js +50 -0
  257. package/lib/compiler/emitters/qwen-emitter.js +39 -0
  258. package/lib/compiler/errors.js +119 -0
  259. package/lib/compiler/index.js +256 -0
  260. package/lib/compiler/manifest.js +242 -0
  261. package/lib/compiler/parser.js +258 -0
  262. package/lib/compiler/platforms.js +113 -0
  263. package/lib/compiler/resource-copier.js +320 -0
  264. package/lib/compiler/rules-emitters/__tests__/antigravity-rules-emitter.test.js +191 -0
  265. package/lib/compiler/rules-emitters/__tests__/codex-rules-emitter.test.js +109 -0
  266. package/lib/compiler/rules-emitters/__tests__/cursor-rules-emitter.test.js +123 -0
  267. package/lib/compiler/rules-emitters/__tests__/qwen-rules-emitter.test.js +123 -0
  268. package/lib/compiler/rules-emitters/antigravity-rules-emitter.js +253 -0
  269. package/lib/compiler/rules-emitters/base-rules-emitter.js +83 -0
  270. package/lib/compiler/rules-emitters/codex-rules-emitter.js +116 -0
  271. package/lib/compiler/rules-emitters/cursor-rules-emitter.js +98 -0
  272. package/lib/compiler/rules-emitters/index.js +71 -0
  273. package/lib/compiler/rules-emitters/qwen-rules-emitter.js +70 -0
  274. package/lib/compiler/schemas.js +144 -0
  275. package/lib/compiler/skills-registry.js +225 -0
  276. package/lib/compiler/transformer.js +236 -0
  277. package/package.json +50 -0
@@ -0,0 +1,463 @@
1
+ #!/usr/bin/env bash
2
+ # =============================================================================
3
+ # generate-clarification-report.sh - 澄清报告生成
4
+ # =============================================================================
5
+ # Purpose: 基于会话数据生成 Markdown 报告 (借鉴 spec-kit /speckit.clarify 模式)
6
+ # Usage: generate-clarification-report.sh --session .session.json --output clarifications/
7
+ # Output: Markdown file (clarifications/[timestamp]-flow-clarify.md)
8
+ # Exit codes: 0=success, 1=invalid session, 2=fatal
9
+ # =============================================================================
10
+
11
+ set -euo pipefail
12
+
13
+ SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
14
+ source "${SCRIPT_DIR}/common.sh"
15
+
16
+ # Configuration
17
+ SESSION_FILE=""
18
+ OUTPUT_DIR=""
19
+
20
+ # 维度名称映射
21
+ readonly DIM_NAMES=(
22
+ "" "Functional Scope" "Data Model" "UX Flow" "Non-Functional Quality"
23
+ "Integration & Dependencies" "Edge Cases" "Constraints & Tradeoffs"
24
+ "Terminology" "Completion Signals" "Misc & Placeholders" "Security & Privacy"
25
+ )
26
+
27
+ # =============================================================================
28
+ # Coverage Status 计算
29
+ # =============================================================================
30
+ calculate_coverage_status() {
31
+ local session_data="$1"
32
+
33
+ # 统计各类别状态
34
+ local clear_count=0
35
+ local resolved_count=0
36
+ local deferred_count=0
37
+ local outstanding_count=0
38
+
39
+ local dimensions
40
+ dimensions=$(echo "$session_data" | jq -c '.scanResults.dimensions // []')
41
+
42
+ local questions
43
+ questions=$(echo "$session_data" | jq -c '.questions // []')
44
+
45
+ # 遍历维度判断状态
46
+ for i in {1..11}; do
47
+ local dim_status
48
+ dim_status=$(echo "$dimensions" | jq -r ".[$((i-1))].status // \"clear\"")
49
+ local dim_issues
50
+ dim_issues=$(echo "$dimensions" | jq ".[$((i-1))].issues | length // 0")
51
+
52
+ # 检查是否有对应问题已回答
53
+ local dim_resolved
54
+ dim_resolved=$(echo "$questions" | jq "[.[] | select(.dimensionId == $i and .answer != null)] | length")
55
+
56
+ if [[ "$dim_status" == "clear" ]] || [[ "$dim_issues" -eq 0 ]]; then
57
+ clear_count=$((clear_count + 1))
58
+ elif [[ "$dim_resolved" -gt 0 ]]; then
59
+ resolved_count=$((resolved_count + 1))
60
+ elif [[ "$dim_issues" -gt 0 ]]; then
61
+ outstanding_count=$((outstanding_count + 1))
62
+ fi
63
+ done
64
+
65
+ echo "${clear_count}|${resolved_count}|${deferred_count}|${outstanding_count}"
66
+ }
67
+
68
+ # =============================================================================
69
+ # T046: 报告模板生成 (增强版,借鉴 spec-kit)
70
+ # =============================================================================
71
+ generate_report_template() {
72
+ local session_data="$1"
73
+
74
+ local session_id req_id status created_at
75
+ session_id=$(echo "$session_data" | jq -r '.sessionId // "unknown"')
76
+ req_id=$(echo "$session_data" | jq -r '.reqId // "REQ-XXX"')
77
+ status=$(echo "$session_data" | jq -r '.status // "unknown"')
78
+ created_at=$(echo "$session_data" | jq -r '.createdAt // ""')
79
+
80
+ local questions_count questions_answered
81
+ questions_count=$(echo "$session_data" | jq '.questions | length')
82
+ questions_answered=$(echo "$session_data" | jq '[.questions[] | select(.answer != null)] | length')
83
+
84
+ # 计算扫描时间
85
+ local scan_duration_ms
86
+ scan_duration_ms=$(echo "$session_data" | jq -r '.scanResults.scanDurationMs // 0')
87
+ local scan_duration_sec=$((scan_duration_ms / 1000))
88
+
89
+ # 计算覆盖状态
90
+ local coverage
91
+ coverage=$(calculate_coverage_status "$session_data")
92
+ local clear_count resolved_count deferred_count outstanding_count
93
+ IFS='|' read -r clear_count resolved_count deferred_count outstanding_count <<< "$coverage"
94
+
95
+ local today_date
96
+ today_date=$(TZ='Asia/Shanghai' date '+%Y-%m-%d')
97
+
98
+ cat << EOF
99
+ # Clarification Report: ${req_id}
100
+
101
+ **Session ID**: ${session_id}
102
+ **Date**: ${today_date}
103
+ **Duration**: ${scan_duration_sec}s (scan) + interactive Q&A
104
+ **Status**: ${status}
105
+
106
+ ---
107
+
108
+ ## Metadata
109
+
110
+ | Field | Value |
111
+ |-------|-------|
112
+ | Requirement | ${req_id} |
113
+ | Session ID | ${session_id} |
114
+ | Questions | ${questions_answered} / ${questions_count} |
115
+ | Generated | $(get_beijing_time_iso) |
116
+
117
+ ---
118
+
119
+ ## Coverage Summary
120
+
121
+ **Legend**: ✅ Clear | ⚠️ Partial | ❌ Missing | 🔄 Resolved | ⏭️ Deferred
122
+
123
+ EOF
124
+
125
+ # 生成维度覆盖表格
126
+ echo "| Category | Status | Issues | Resolution |"
127
+ echo "|----------|--------|--------|------------|"
128
+
129
+ local dimensions
130
+ dimensions=$(echo "$session_data" | jq -c '.scanResults.dimensions // []')
131
+ local questions
132
+ questions=$(echo "$session_data" | jq -c '.questions // []')
133
+
134
+ for i in {1..11}; do
135
+ local dim_name="${DIM_NAMES[$i]}"
136
+ local dim_status
137
+ dim_status=$(echo "$dimensions" | jq -r ".[$((i-1))].status // \"clear\"")
138
+ local dim_issues
139
+ dim_issues=$(echo "$dimensions" | jq ".[$((i-1))].issues | length // 0")
140
+
141
+ # 检查是否有对应问题已回答
142
+ local dim_resolved
143
+ dim_resolved=$(echo "$questions" | jq "[.[] | select(.dimensionId == $i and .answer != null)] | length")
144
+
145
+ local status_icon resolution
146
+ if [[ "$dim_status" == "clear" ]] || [[ "$dim_issues" -eq 0 ]]; then
147
+ status_icon="✅ Clear"
148
+ resolution="-"
149
+ elif [[ "$dim_resolved" -gt 0 ]]; then
150
+ status_icon="🔄 Resolved"
151
+ resolution="Q&A addressed"
152
+ elif [[ "$dim_status" == "ambiguous" ]]; then
153
+ status_icon="⚠️ Partial"
154
+ resolution="Outstanding"
155
+ else
156
+ status_icon="❌ Missing"
157
+ resolution="Needs attention"
158
+ fi
159
+
160
+ echo "| ${dim_name} | ${status_icon} | ${dim_issues} | ${resolution} |"
161
+ done
162
+
163
+ cat << EOF
164
+
165
+ **Summary**:
166
+ - ✅ Clear: ${clear_count} categories
167
+ - 🔄 Resolved: ${resolved_count} categories
168
+ - ⏭️ Deferred: ${deferred_count} categories
169
+ - ❌ Outstanding: ${outstanding_count} categories
170
+
171
+ ---
172
+
173
+ ## High-Priority Issues (Priority ≥ 50)
174
+
175
+ EOF
176
+
177
+ # 高优先级问题
178
+ local high_priority_issues
179
+ high_priority_issues=$(echo "$session_data" | jq '[
180
+ .scanResults.dimensions[]? |
181
+ .dimensionId as $dimId |
182
+ .name as $dimName |
183
+ (.issues // [])[]? |
184
+ select(.priority >= 50) |
185
+ . + {dimensionId: $dimId, dimensionName: $dimName}
186
+ ]')
187
+ local hp_count
188
+ hp_count=$(echo "$high_priority_issues" | jq 'length')
189
+
190
+ if [[ "$hp_count" -gt 0 ]]; then
191
+ echo "| Issue ID | Dimension | Description | Impact | Uncertainty | Priority |"
192
+ echo "|----------|-----------|-------------|--------|-------------|----------|"
193
+ echo "$high_priority_issues" | jq -r '.[] | "| \(.issueId // "N/A") | \(.dimensionName // (.dimensionId | tostring) // "-") | \(.description // "-" | .[0:50])... | \(.impact // 0)/10 | \(.uncertainty // 0)/10 | \(.priority // 0) |"'
194
+ else
195
+ echo "No high-priority issues detected."
196
+ fi
197
+
198
+ cat << EOF
199
+
200
+ ---
201
+
202
+ ## Clarification Session
203
+
204
+ ### Session ${today_date}
205
+
206
+ EOF
207
+
208
+ # 生成每个问题的详细信息 (spec-kit Q→A 格式)
209
+ local idx=0
210
+ while IFS= read -r question; do
211
+ [[ -z "$question" ]] && continue
212
+ idx=$((idx + 1))
213
+ local qid text answer rationale dim_id recommended rec_rationale question_type
214
+ qid=$(echo "$question" | jq -r '.questionId')
215
+ text=$(echo "$question" | jq -r '.text')
216
+ answer=$(echo "$question" | jq -r '.answer // "Not answered"')
217
+ rationale=$(echo "$question" | jq -r '.rationale // "N/A"')
218
+ dim_id=$(echo "$question" | jq -r '.dimensionId')
219
+ recommended=$(echo "$question" | jq -r '.recommendedOption // "N/A"')
220
+ rec_rationale=$(echo "$question" | jq -r '.recommendedRationale // "Based on industry best practices"')
221
+ question_type=$(echo "$question" | jq -r '.type // "multiple_choice"')
222
+
223
+ local dim_name="${DIM_NAMES[$dim_id]:-Unknown}"
224
+
225
+ cat << EOF
226
+ #### ${qid}: ${dim_name}
227
+
228
+ **Question**: ${text}
229
+
230
+ **Type**: ${question_type}
231
+
232
+ EOF
233
+
234
+ if [[ "$question_type" == "multiple_choice" ]]; then
235
+ echo "| Option | Description |"
236
+ echo "|--------|-------------|"
237
+ echo "$question" | jq -r '.options // [] | .[] | "| **\(.optionId)** | \(.text) - \(.description // "") |"'
238
+
239
+ cat << EOF
240
+
241
+ **Recommended**: ${recommended} ⭐
242
+ > ${rec_rationale}
243
+
244
+ EOF
245
+ else
246
+ cat << EOF
247
+ **Format**: Short answer (≤5 words)
248
+
249
+ EOF
250
+ fi
251
+
252
+ cat << EOF
253
+ **User Answer**: ${answer}
254
+
255
+ **Rationale**: ${rationale}
256
+
257
+ - Q: ${text} → A: ${answer}
258
+
259
+ ---
260
+
261
+ EOF
262
+ done < <(echo "$session_data" | jq -c '.questions // [] | .[]')
263
+
264
+ if [[ "$questions_count" -eq 0 ]]; then
265
+ echo "No questions generated - research.md appears sufficiently clear."
266
+ echo ""
267
+ echo "---"
268
+ echo ""
269
+ fi
270
+
271
+ cat << EOF
272
+ ## Validation Checklist
273
+
274
+ - [x] CLR-001 All Q&A pairs recorded in session section
275
+ - [x] CLR-002 No duplicate questions asked
276
+ - [x] CLR-003 Questions ≤ 5 (actual: ${questions_count})
277
+ - [x] CLR-004 Coverage summary complete for all 11 dimensions
278
+ - [ ] CLR-005 No contradictory statements remain in research.md
279
+ - [ ] CLR-006 Terminology consistent across updated sections
280
+
281
+ ---
282
+
283
+ ## Next Steps
284
+
285
+ EOF
286
+
287
+ if [[ "$questions_answered" -eq "$questions_count" ]] && [[ "$questions_count" -gt 0 ]]; then
288
+ cat << EOF
289
+ ✅ **Clarification complete**. All ${questions_answered} questions answered.
290
+
291
+ Recommended next command: \`/flow-prd ${req_id}\`
292
+ EOF
293
+ elif [[ "$questions_count" -eq 0 ]]; then
294
+ cat << EOF
295
+ ✅ **No critical ambiguities detected**. research.md is sufficiently specified.
296
+
297
+ Recommended next command: \`/flow-prd ${req_id}\`
298
+ EOF
299
+ else
300
+ cat << EOF
301
+ ⚠️ **Clarification incomplete**. ${questions_answered}/${questions_count} questions answered.
302
+
303
+ Options:
304
+ 1. Continue with \`/flow-clarify ${req_id}\` to complete remaining questions
305
+ 2. Proceed with \`/flow-prd ${req_id}\` (acceptable risk if remaining items are low-impact)
306
+ EOF
307
+ fi
308
+
309
+ if [[ "$outstanding_count" -gt 0 ]]; then
310
+ cat << EOF
311
+
312
+ > ⚠️ **${outstanding_count} categories** have outstanding issues that may need attention during planning phase.
313
+ EOF
314
+ fi
315
+
316
+ cat << EOF
317
+
318
+ ---
319
+
320
+ **Generated by**: clarify-analyst agent
321
+ **Report Version**: 1.1.0
322
+ **Template**: CC-DevFlow Clarification Report (based on spec-kit pattern)
323
+ EOF
324
+ }
325
+
326
+ # =============================================================================
327
+ # T047: 模板渲染
328
+ # =============================================================================
329
+ render_report() {
330
+ local session_file="$1"
331
+
332
+ # 读取会话数据
333
+ if [[ ! -f "$session_file" ]]; then
334
+ echo "Error: Session file not found: $session_file" >&2
335
+ return 1
336
+ fi
337
+
338
+ local session_data
339
+ session_data=$(cat "$session_file")
340
+
341
+ # 验证 JSON
342
+ if ! echo "$session_data" | jq -e '.' >/dev/null 2>&1; then
343
+ echo "Error: Invalid JSON in session file" >&2
344
+ return 1
345
+ fi
346
+
347
+ # 生成报告
348
+ generate_report_template "$session_data"
349
+ }
350
+
351
+ # =============================================================================
352
+ # T048: 报告写入
353
+ # =============================================================================
354
+ write_report() {
355
+ local content="$1"
356
+ local output_dir="$2"
357
+ local session_id="$3"
358
+
359
+ # 确保输出目录存在
360
+ mkdir -p "$output_dir"
361
+
362
+ # 生成文件名
363
+ local timestamp
364
+ timestamp=$(TZ='Asia/Shanghai' date '+%Y%m%d-%H%M%S')
365
+ local filename="${timestamp}-flow-clarify.md"
366
+ local filepath="${output_dir}/${filename}"
367
+
368
+ # 写入文件
369
+ echo "$content" > "$filepath"
370
+
371
+ # 验证写入
372
+ if [[ -f "$filepath" ]]; then
373
+ echo "$filepath"
374
+ return 0
375
+ else
376
+ echo "Error: Failed to write report" >&2
377
+ return 1
378
+ fi
379
+ }
380
+
381
+ # =============================================================================
382
+ # T049: 主入口
383
+ # =============================================================================
384
+ parse_args() {
385
+ local session_file=""
386
+ local output_dir=""
387
+
388
+ while [[ $# -gt 0 ]]; do
389
+ case "$1" in
390
+ --session)
391
+ session_file="$2"
392
+ shift 2
393
+ ;;
394
+ --output)
395
+ output_dir="$2"
396
+ shift 2
397
+ ;;
398
+ --help|-h)
399
+ echo "Usage: generate-clarification-report.sh --session .session.json [--output clarifications/]"
400
+ exit 0
401
+ ;;
402
+ *)
403
+ shift
404
+ ;;
405
+ esac
406
+ done
407
+
408
+ echo "$session_file|$output_dir"
409
+ }
410
+
411
+ main() {
412
+ # 解析参数
413
+ local parsed
414
+ parsed=$(parse_args "$@")
415
+ local session_file output_dir
416
+ IFS='|' read -r session_file output_dir <<< "$parsed"
417
+
418
+ # 验证输入
419
+ if [[ -z "$session_file" ]]; then
420
+ echo '{"error": {"code": "MISSING_SESSION", "message": "--session is required"}}' >&2
421
+ exit 2
422
+ fi
423
+
424
+ if [[ ! -f "$session_file" ]]; then
425
+ echo '{"error": {"code": "SESSION_NOT_FOUND", "message": "Session file not found"}}' >&2
426
+ exit 1
427
+ fi
428
+
429
+ # 确定输出目录
430
+ if [[ -z "$output_dir" ]]; then
431
+ # 从会话文件路径推断
432
+ output_dir=$(dirname "$session_file")
433
+ fi
434
+
435
+ # 读取会话 ID
436
+ local session_id
437
+ session_id=$(jq -r '.sessionId // "unknown"' "$session_file")
438
+
439
+ # 渲染报告
440
+ local report_content
441
+ report_content=$(render_report "$session_file")
442
+
443
+ if [[ -z "$report_content" ]]; then
444
+ echo '{"error": {"code": "RENDER_FAILED", "message": "Failed to render report"}}' >&2
445
+ exit 2
446
+ fi
447
+
448
+ # 写入报告
449
+ local report_path
450
+ report_path=$(write_report "$report_content" "$output_dir" "$session_id")
451
+
452
+ if [[ -n "$report_path" ]]; then
453
+ echo "{\"success\": true, \"reportPath\": \"$report_path\"}"
454
+ else
455
+ echo '{"error": {"code": "WRITE_FAILED", "message": "Failed to write report"}}' >&2
456
+ exit 2
457
+ fi
458
+ }
459
+
460
+ # 只在直接执行时运行 main
461
+ if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
462
+ main "$@"
463
+ fi
@@ -0,0 +1,146 @@
1
+ #!/usr/bin/env bash
2
+ # shellcheck disable=SC2312
3
+
4
+ set -euo pipefail
5
+
6
+ usage() {
7
+ cat <<'USAGE'
8
+ Usage: scripts/bash/generate-quickstart.sh <requirement-dir>
9
+
10
+ Generates a quickstart.md file summarizing environment setup, commands, and
11
+ test execution instructions derived from TECH_DESIGN.md and existing project
12
+ scripts.
13
+ USAGE
14
+ }
15
+
16
+ if [[ "${1:-}" == "-h" || "${1:-}" == "--help" ]]; then
17
+ usage
18
+ exit 0
19
+ fi
20
+
21
+ if [[ $# -lt 1 ]]; then
22
+ echo "Error: requirement directory is required." >&2
23
+ usage
24
+ exit 1
25
+ fi
26
+
27
+ REQ_DIR="$1"
28
+ if [[ ! -d "$REQ_DIR" ]]; then
29
+ echo "Error: requirement directory '$REQ_DIR' does not exist." >&2
30
+ exit 1
31
+ fi
32
+
33
+ python3 - "$REQ_DIR" <<'PY'
34
+ from __future__ import annotations
35
+
36
+ import json
37
+ import os
38
+ import re
39
+ import shutil
40
+ import sys
41
+ from datetime import datetime, timezone
42
+ from pathlib import Path
43
+
44
+ req_dir = Path(sys.argv[1]).resolve()
45
+ tech_design = req_dir / "TECH_DESIGN.md"
46
+ output_path = req_dir / "quickstart.md"
47
+
48
+ if not tech_design.exists():
49
+ print(f"Error: {tech_design} not found.", file=sys.stderr)
50
+ sys.exit(1)
51
+
52
+ content = tech_design.read_text(encoding="utf-8", errors="ignore")
53
+
54
+ def extract_section(title: str) -> str | None:
55
+ pattern = re.compile(rf"(^##\s+{re.escape(title)}.*?)(?=^##\s+|\Z)", re.MULTILINE | re.DOTALL)
56
+ match = pattern.search(content)
57
+ return match.group(1).strip() if match else None
58
+
59
+ env_section = extract_section("2. Technology Stack")
60
+ tests_section = extract_section("6. Performance Design")
61
+ security_section = extract_section("5. Security Design")
62
+
63
+ setup_commands = []
64
+ test_commands = []
65
+
66
+ def search_upwards(filename: str) -> Path | None:
67
+ for parent in [req_dir] + list(req_dir.parents):
68
+ candidate = parent / filename
69
+ if candidate.exists():
70
+ return candidate
71
+ return None
72
+
73
+ package_json = search_upwards("package.json")
74
+ if package_json:
75
+ data = json.loads(package_json.read_text(encoding="utf-8"))
76
+ scripts = data.get("scripts", {})
77
+ for key in ("dev", "start", "lint", "build", "test", "test:integration"):
78
+ if key in scripts:
79
+ command = f"npm run {key}"
80
+ (test_commands if "test" in key else setup_commands).append(command)
81
+ # Also surface underlying script command values for clarity (e.g., pytest)
82
+ for key, val in scripts.items():
83
+ if re.search(r"\bpytest\b", val):
84
+ test_commands.append(val)
85
+
86
+ makefile = search_upwards("Makefile")
87
+ if makefile:
88
+ content = makefile.read_text(encoding="utf-8", errors="ignore")
89
+ for target in ("setup", "serve", "dev", "test", "test-integration"):
90
+ if re.search(rf"^{target}:", content, re.MULTILINE):
91
+ command = f"make {target}"
92
+ (test_commands if "test" in target else setup_commands).append(command)
93
+
94
+ generated_at = datetime.now(timezone.utc).isoformat()
95
+
96
+ lines = [
97
+ f"# Quickstart — {req_dir.name}",
98
+ "",
99
+ f"_Generated {generated_at}_",
100
+ "",
101
+ "## 环境准备",
102
+ "",
103
+ ]
104
+
105
+ if env_section:
106
+ lines.append(env_section)
107
+ else:
108
+ lines.append("- 参考 TECH_DESIGN.md 中的技术栈章节配置环境。")
109
+ lines.append("")
110
+ lines.append("## 初始化步骤")
111
+ lines.append("")
112
+
113
+ if setup_commands:
114
+ for cmd in dict.fromkeys(setup_commands):
115
+ lines.append(f"- `{cmd}`")
116
+ else:
117
+ lines.append("- TODO: 填写项目初始化命令(例如安装依赖、启动服务)。")
118
+
119
+ lines.append("")
120
+ lines.append("## 测试命令")
121
+ lines.append("")
122
+ if test_commands:
123
+ for cmd in dict.fromkeys(test_commands):
124
+ lines.append(f"- `{cmd}`")
125
+ else:
126
+ lines.append("- TODO: 填写单元/集成测试命令。")
127
+
128
+ lines.append("")
129
+ lines.append("## 安全验证提示")
130
+ lines.append("")
131
+ if security_section:
132
+ lines.append(security_section)
133
+ else:
134
+ lines.append("- TODO: 补充安全配置与验证步骤。")
135
+
136
+ lines.append("")
137
+ lines.append("## 性能基线 (可选)")
138
+ lines.append("")
139
+ if tests_section:
140
+ lines.append(tests_section)
141
+ else:
142
+ lines.append("- TODO: 如有性能要求,补充压测或指标监控步骤。")
143
+
144
+ output_path.write_text("\n".join(lines) + "\n", encoding="utf-8")
145
+ print(f"Wrote quickstart guide → {output_path}")
146
+ PY