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,601 @@
1
+ #!/usr/bin/env bash
2
+ # =============================================================================
3
+ # run-clarify-scan.sh - 11 维度并行歧义扫描
4
+ # =============================================================================
5
+ # Purpose: 执行 11 维度并行扫描,识别 research.md 中的歧义
6
+ # Usage: run-clarify-scan.sh REQ-XXX [--parallel] [--timeout 20] [--dimension all|1-11]
7
+ # Output: JSON (stdout) - DimensionScanResult[]
8
+ # Exit codes: 0=success, 1=partial (some dimensions timeout), 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
+ # =============================================================================
17
+ # 跨平台时间戳 (epoch ms)
18
+ # =============================================================================
19
+ now_ms() {
20
+ if command -v python3 >/dev/null 2>&1; then
21
+ python3 -c 'import time; print(int(time.time() * 1000))'
22
+ return 0
23
+ fi
24
+ echo $(( $(date +%s) * 1000 ))
25
+ }
26
+
27
+ # =============================================================================
28
+ # Constants (T012)
29
+ # =============================================================================
30
+ readonly DIMENSIONS=(
31
+ "Functional Scope"
32
+ "Data Model"
33
+ "UX Flow"
34
+ "Non-Functional Quality"
35
+ "Integration & Dependencies"
36
+ "Edge Cases"
37
+ "Constraints & Tradeoffs"
38
+ "Terminology"
39
+ "Completion Signals"
40
+ "Misc & Placeholders"
41
+ "Security & Privacy"
42
+ )
43
+
44
+ readonly DIMENSION_FOCUS=(
45
+ "核心目标、成功标准、排除声明"
46
+ "实体、属性、关系、生命周期"
47
+ "用户旅程、错误状态、空状态"
48
+ "延迟、吞吐量、可用性、可观测性"
49
+ "外部 API、故障模式、协议"
50
+ "负面场景、限流、冲突解决"
51
+ "技术约束、拒绝方案"
52
+ "规范术语、避免同义词"
53
+ "验收标准、完成定义"
54
+ "TODO 标记、模糊形容词"
55
+ "AuthN/AuthZ、数据保护、合规"
56
+ )
57
+
58
+ # Configuration
59
+ DEFAULT_TIMEOUT=20
60
+ PARALLEL_MODE=true
61
+ SELECTED_DIMENSION="all"
62
+
63
+ # =============================================================================
64
+ # T007: API Key 验证
65
+ # =============================================================================
66
+ check_api_key() {
67
+ if [[ -z "${CLAUDE_API_KEY:-}" ]]; then
68
+ echo '{"error": {"code": "API_KEY_MISSING", "message": "CLAUDE_API_KEY not set"}}' >&2
69
+ echo "提示: export CLAUDE_API_KEY='sk-ant-...'" >&2
70
+ return 1
71
+ fi
72
+
73
+ # 验证格式 (sk-ant-* 开头)
74
+ if [[ ! "${CLAUDE_API_KEY}" =~ ^sk-ant- ]]; then
75
+ echo '{"error": {"code": "API_KEY_INVALID", "message": "CLAUDE_API_KEY format invalid"}}' >&2
76
+ return 1
77
+ fi
78
+ return 0
79
+ }
80
+
81
+ # =============================================================================
82
+ # API Key 快速检测 (不输出错误)
83
+ # =============================================================================
84
+ has_valid_api_key() {
85
+ [[ -n "${CLAUDE_API_KEY:-}" ]] && [[ "${CLAUDE_API_KEY}" =~ ^sk-ant- ]]
86
+ }
87
+
88
+ # =============================================================================
89
+ # 无 API Key 降级:规则引擎歧义扫描
90
+ # =============================================================================
91
+ contains_any() {
92
+ local haystack="$1"
93
+ shift
94
+ local needle
95
+ for needle in "$@"; do
96
+ if [[ -n "$needle" ]] && [[ "$haystack" == *"$needle"* ]]; then
97
+ return 0
98
+ fi
99
+ done
100
+ return 1
101
+ }
102
+
103
+ scan_dimension_heuristic() {
104
+ local dim_id="$1"
105
+ local dim_name="$2"
106
+ local research_content="$3"
107
+
108
+ local description impact uncertainty
109
+ description=""
110
+ impact=5
111
+ uncertainty=5
112
+
113
+ case "$dim_id" in
114
+ 1)
115
+ if ! contains_any "$research_content" "In Scope" "Out of Scope" "in-scope" "out-of-scope" "范围" "不做" "非目标" "success" "Success" "验收" "完成定义"; then
116
+ description="功能范围边界与成功标准不清:缺少 In Scope / Out of Scope、可度量验收信号"
117
+ impact=9
118
+ uncertainty=8
119
+ fi
120
+ ;;
121
+ 2)
122
+ if ! contains_any "$research_content" "ExecutionResult" "Context" "context" "capabilities" "capability" "schema" "字段" "数据模型"; then
123
+ description="接口数据模型未定义:Context / ExecutionResult / capabilities 的结构与生命周期缺失"
124
+ impact=7
125
+ uncertainty=7
126
+ fi
127
+ ;;
128
+ 3)
129
+ if ! contains_any "$research_content" "UX" "用户" "交互" "CLI" "参数" "flag" "配置" "auto" "自动"; then
130
+ description="调用方如何选择/覆盖 Adapter 未定义:自动探测 vs 配置文件 vs CLI 参数"
131
+ impact=7
132
+ uncertainty=6
133
+ fi
134
+ ;;
135
+ 4)
136
+ if ! contains_any "$research_content" "性能" "latency" "timeout" "SLA" "可观测" "logging" "metrics" "稳定" "reliability"; then
137
+ description="非功能目标缺失:性能/超时/错误恢复/可观测性要求未量化"
138
+ impact=6
139
+ uncertainty=7
140
+ fi
141
+ ;;
142
+ 5)
143
+ if ! contains_any "$research_content" ".claude" "commands" "agents" "scripts" "集成" "integration" "entry" "registry"; then
144
+ description="集成点不清:与现有命令/Agent/脚本的调用链、初始化时机、配置来源未明确"
145
+ impact=8
146
+ uncertainty=7
147
+ fi
148
+ ;;
149
+ 6)
150
+ if ! contains_any "$research_content" "fallback" "优先级" "priority" "冲突" "conflict" "multiple" "edge" "降级"; then
151
+ description="边界场景缺失:多个 Adapter 命中/无 Adapter 命中/探测冲突时的降级与报错策略"
152
+ impact=7
153
+ uncertainty=7
154
+ fi
155
+ ;;
156
+ 7)
157
+ if ! contains_any "$research_content" "constraint" "tradeoff" "取舍" "约束" "兼容" "backward" "breaking" "向后"; then
158
+ description="约束与取舍未声明:是否必须保持 Claude Code 行为完全兼容?允许哪些破坏性重构?"
159
+ impact=6
160
+ uncertainty=6
161
+ fi
162
+ ;;
163
+ 8)
164
+ if [[ "$research_content" =~ RM-[0-9]+ ]]; then
165
+ description="术语/编号不一致:文档出现 RM-* 与 REQ-* 混用,需统一命名与引用"
166
+ impact=5
167
+ uncertainty=6
168
+ fi
169
+ ;;
170
+ 9)
171
+ if ! contains_any "$research_content" "验收" "完成" "Definition" "DoD" "Acceptance" "测试" "交付"; then
172
+ description="完成定义不清:需要可验证交付物与验收清单(接口、注册表、默认适配器、测试)"
173
+ impact=8
174
+ uncertainty=8
175
+ fi
176
+ ;;
177
+ 10)
178
+ if contains_any "$research_content" "TODO" "TBD" "None requested" "not clear" "不清楚"; then
179
+ description="存在占位/来源缺失:外部参考/输入不足,决策依据可能不够可追溯"
180
+ impact=5
181
+ uncertainty=7
182
+ fi
183
+ ;;
184
+ 11)
185
+ if ! contains_any "$research_content" "security" "Security" "privacy" "Privacy" "权限" "Auth" "鉴权" "审计"; then
186
+ description="安全边界未定义:Adapter 能力(shell/filesystem/network)最小化、审计与隔离策略缺失"
187
+ impact=8
188
+ uncertainty=7
189
+ fi
190
+ ;;
191
+ *)
192
+ ;;
193
+ esac
194
+
195
+ if [[ -z "$description" ]]; then
196
+ jq -n '{"issues": []}'
197
+ return 0
198
+ fi
199
+
200
+ jq -n \
201
+ --arg desc "$description" \
202
+ --argjson impact "$impact" \
203
+ --argjson uncertainty "$uncertainty" \
204
+ '{
205
+ issues: [
206
+ {
207
+ description: $desc,
208
+ impact: $impact,
209
+ uncertainty: $uncertainty
210
+ }
211
+ ]
212
+ }'
213
+ }
214
+
215
+ # =============================================================================
216
+ # T008: 用户答案验证
217
+ # =============================================================================
218
+ validate_answer() {
219
+ local type="$1" # "multiple_choice" | "short_answer"
220
+ local answer="$2"
221
+ local options="${3:-A,B,C,D,E}"
222
+
223
+ case "$type" in
224
+ "multiple_choice")
225
+ # 必须是单字符 A-E (大小写)
226
+ if [[ ! "$answer" =~ ^[A-Ea-e]$ ]]; then
227
+ return 1
228
+ fi
229
+ # 转换为大写并检查是否在有效选项中
230
+ local upper_answer
231
+ upper_answer=$(echo "$answer" | tr '[:lower:]' '[:upper:]')
232
+ if [[ ! ",$options," == *",$upper_answer,"* ]]; then
233
+ return 1
234
+ fi
235
+ ;;
236
+ "short_answer")
237
+ # ≤5 个单词
238
+ local word_count
239
+ word_count=$(echo "$answer" | wc -w | tr -d ' ')
240
+ if [[ "$word_count" -gt 5 ]]; then
241
+ return 1
242
+ fi
243
+ # 仅允许字母数字 + 空格 + 基本符号
244
+ if [[ ! "$answer" =~ ^[a-zA-Z0-9\ \<\>\.\-\_\,]+$ ]]; then
245
+ return 1
246
+ fi
247
+ ;;
248
+ *)
249
+ return 1
250
+ ;;
251
+ esac
252
+ return 0
253
+ }
254
+
255
+ # =============================================================================
256
+ # T009: 会话状态读取
257
+ # =============================================================================
258
+ load_session() {
259
+ local session_file="$1"
260
+
261
+ if [[ ! -f "$session_file" ]]; then
262
+ echo '{"error": {"code": "SESSION_NOT_FOUND", "message": "Session file not found"}}' >&2
263
+ return 1
264
+ fi
265
+
266
+ # 验证 JSON 格式
267
+ if ! jq -e '.' "$session_file" >/dev/null 2>&1; then
268
+ echo '{"error": {"code": "SESSION_CORRUPT", "message": "Invalid JSON in session file"}}' >&2
269
+ return 1
270
+ fi
271
+
272
+ cat "$session_file"
273
+ }
274
+
275
+ # =============================================================================
276
+ # T010: 会话状态保存
277
+ # =============================================================================
278
+ save_session() {
279
+ local session_file="$1"
280
+ local session_data="$2"
281
+
282
+ # 更新 updatedAt 时间戳
283
+ local updated_data
284
+ updated_data=$(echo "$session_data" | jq --arg ts "$(get_beijing_time_iso)" '.updatedAt = $ts')
285
+
286
+ # 原子写入 (临时文件 + mv)
287
+ local temp_file="${session_file}.tmp"
288
+ echo "$updated_data" | jq '.' > "$temp_file"
289
+ mv "$temp_file" "$session_file"
290
+ }
291
+
292
+ # =============================================================================
293
+ # T011: Claude API 调用封装
294
+ # =============================================================================
295
+ call_claude_api() {
296
+ local model="$1" # "claude-4-5-haiku" | "claude-sonnet-4-5-20241022"
297
+ local system_prompt="$2"
298
+ local user_content="$3"
299
+ local max_tokens="${4:-1000}"
300
+ local timeout="${5:-$DEFAULT_TIMEOUT}"
301
+
302
+ local retry_count=0
303
+ local max_retries=3
304
+ local result=""
305
+
306
+ while [[ $retry_count -lt $max_retries ]]; do
307
+ result=$(timeout "${timeout}s" curl -s \
308
+ -X POST "https://api.anthropic.com/v1/messages" \
309
+ -H "Authorization: Bearer ${CLAUDE_API_KEY}" \
310
+ -H "Content-Type: application/json" \
311
+ -H "anthropic-version: 2023-06-01" \
312
+ -d "$(jq -n \
313
+ --arg model "$model" \
314
+ --arg system "$system_prompt" \
315
+ --arg content "$user_content" \
316
+ --argjson max_tokens "$max_tokens" \
317
+ '{
318
+ model: $model,
319
+ max_tokens: $max_tokens,
320
+ system: $system,
321
+ messages: [{role: "user", content: $content}]
322
+ }')" 2>/dev/null) || true
323
+
324
+ # 检查是否成功
325
+ if [[ -n "$result" ]] && echo "$result" | jq -e '.content[0].text' >/dev/null 2>&1; then
326
+ echo "$result" | jq -r '.content[0].text'
327
+ return 0
328
+ fi
329
+
330
+ # 检查是否是 rate limit
331
+ if echo "$result" | jq -e '.error.type == "rate_limit_error"' >/dev/null 2>&1; then
332
+ retry_count=$((retry_count + 1))
333
+ sleep $((2 ** retry_count)) # 指数退避
334
+ continue
335
+ fi
336
+
337
+ # 其他错误,退出
338
+ break
339
+ done
340
+
341
+ # 返回错误
342
+ echo '{"error": "API call failed after retries"}'
343
+ return 1
344
+ }
345
+
346
+ # =============================================================================
347
+ # T018: 单维度扫描
348
+ # =============================================================================
349
+ scan_dimension() {
350
+ local dim_id="$1"
351
+ local dim_name="$2"
352
+ local dim_focus="$3"
353
+ local research_content="$4"
354
+ local timeout_sec="${5:-$DEFAULT_TIMEOUT}"
355
+
356
+ local start_time
357
+ start_time=$(now_ms)
358
+
359
+ local system_prompt="You are a requirements ambiguity scanner for dimension: ${dim_name}.
360
+ Focus areas: ${dim_focus}
361
+
362
+ Analyze the research document and identify:
363
+ 1. Missing information (gaps that need clarification)
364
+ 2. Vague terminology (e.g., 'fast', 'robust' without metrics)
365
+ 3. Implicit assumptions (things taken for granted)
366
+ 4. Contradictions with other sections
367
+
368
+ Output ONLY valid JSON:
369
+ {\"issues\": [{\"description\": \"...\", \"impact\": 1-10, \"uncertainty\": 1-10}]}
370
+
371
+ If no issues found, return: {\"issues\": []}"
372
+
373
+ local result
374
+ if has_valid_api_key; then
375
+ result=$(call_claude_api "claude-haiku-4-5-20241022" "$system_prompt" "$research_content" 1000 "$timeout_sec") || true
376
+
377
+ # API 不可用/超时/输出无效时,降级到规则引擎
378
+ if [[ -z "$result" ]] || [[ "$result" == *'"error"'* ]] || ! echo "$result" | jq -e '.issues' >/dev/null 2>&1; then
379
+ result=$(scan_dimension_heuristic "$dim_id" "$dim_name" "$research_content")
380
+ fi
381
+ else
382
+ result=$(scan_dimension_heuristic "$dim_id" "$dim_name" "$research_content")
383
+ fi
384
+
385
+ local end_time
386
+ end_time=$(now_ms)
387
+ local duration=$((end_time - start_time))
388
+
389
+ # 解析结果
390
+ local status="clear"
391
+ local issues="[]"
392
+
393
+ if [[ -z "$result" ]] || [[ "$result" == *'"error"'* ]]; then
394
+ status="timeout"
395
+ elif echo "$result" | jq -e '.issues' >/dev/null 2>&1; then
396
+ issues=$(echo "$result" | jq '.issues')
397
+ local issue_count
398
+ issue_count=$(echo "$issues" | jq 'length')
399
+ if [[ "$issue_count" -gt 0 ]]; then
400
+ status="ambiguous"
401
+ # 添加 issueId 和 priority
402
+ issues=$(echo "$issues" | jq --argjson dim "$dim_id" '
403
+ to_entries | map(
404
+ .value + {
405
+ issueId: ("dim-" + ($dim | tostring) + "-issue-" + ((.key + 1) | tostring)),
406
+ priority: ((.value.impact // 5) * (.value.uncertainty // 5))
407
+ }
408
+ )
409
+ ')
410
+ fi
411
+ fi
412
+
413
+ # 输出结果 JSON
414
+ jq -n \
415
+ --argjson dimId "$dim_id" \
416
+ --arg name "$dim_name" \
417
+ --arg status "$status" \
418
+ --argjson issues "$issues" \
419
+ --argjson scanTimeMs "$duration" \
420
+ '{
421
+ dimensionId: $dimId,
422
+ name: $name,
423
+ status: $status,
424
+ issues: $issues,
425
+ scanTimeMs: $scanTimeMs
426
+ }'
427
+ }
428
+
429
+ # =============================================================================
430
+ # T019: 并行扫描主函数
431
+ # =============================================================================
432
+ scan_all_dimensions() {
433
+ local req_id="$1"
434
+ local timeout_sec="$2"
435
+ local single_dim="${3:-all}"
436
+
437
+ local repo_root
438
+ repo_root=$(get_repo_root)
439
+ local req_dir
440
+ req_dir=$(get_req_dir "$repo_root" "$req_id")
441
+ local research_file="$req_dir/research/research.md"
442
+
443
+ if [[ ! -f "$research_file" ]]; then
444
+ echo '{"error": {"code": "MISSING_RESEARCH", "message": "research.md not found"}}' >&2
445
+ return 2
446
+ fi
447
+
448
+ local research_content
449
+ research_content=$(cat "$research_file")
450
+
451
+ local session_id
452
+ session_id="$(TZ='Asia/Shanghai' date '+%Y%m%d-%H%M%S')-${req_id}"
453
+ local start_time
454
+ start_time=$(now_ms)
455
+
456
+ local results=()
457
+ local pids=()
458
+ local temp_dir
459
+ temp_dir=$(mktemp -d)
460
+
461
+ # 确定要扫描的维度
462
+ local dims_to_scan=()
463
+ if [[ "$single_dim" == "all" ]]; then
464
+ for i in "${!DIMENSIONS[@]}"; do
465
+ dims_to_scan+=("$((i + 1))")
466
+ done
467
+ else
468
+ dims_to_scan=("$single_dim")
469
+ fi
470
+
471
+ # 启动并行扫描
472
+ for dim_id in "${dims_to_scan[@]}"; do
473
+ local idx=$((dim_id - 1))
474
+ local dim_name="${DIMENSIONS[$idx]}"
475
+ local dim_focus="${DIMENSION_FOCUS[$idx]}"
476
+
477
+ (
478
+ scan_dimension "$dim_id" "$dim_name" "$dim_focus" "$research_content" "$timeout_sec" \
479
+ > "$temp_dir/dim_${dim_id}.json"
480
+ ) &
481
+ pids+=($!)
482
+ done
483
+
484
+ # 等待所有完成
485
+ local has_timeout=false
486
+ for pid in "${pids[@]}"; do
487
+ wait "$pid" || has_timeout=true
488
+ done
489
+
490
+ # 收集结果
491
+ local all_results="["
492
+ local first=true
493
+ for dim_id in "${dims_to_scan[@]}"; do
494
+ if [[ -f "$temp_dir/dim_${dim_id}.json" ]]; then
495
+ if [[ "$first" == "true" ]]; then
496
+ first=false
497
+ else
498
+ all_results+=","
499
+ fi
500
+ all_results+=$(cat "$temp_dir/dim_${dim_id}.json")
501
+
502
+ # 检查是否有 timeout
503
+ if jq -e '.status == "timeout"' "$temp_dir/dim_${dim_id}.json" >/dev/null 2>&1; then
504
+ has_timeout=true
505
+ fi
506
+ fi
507
+ done
508
+ all_results+="]"
509
+
510
+ # 清理临时文件
511
+ rm -rf "$temp_dir"
512
+
513
+ local end_time
514
+ end_time=$(now_ms)
515
+ local total_duration=$((end_time - start_time))
516
+
517
+ # T020: 格式化输出
518
+ jq -n \
519
+ --arg sessionId "$session_id" \
520
+ --argjson scanDurationMs "$total_duration" \
521
+ --argjson dimensions "$all_results" \
522
+ '{
523
+ sessionId: $sessionId,
524
+ scanDurationMs: $scanDurationMs,
525
+ dimensions: $dimensions
526
+ }'
527
+
528
+ if [[ "$has_timeout" == "true" ]]; then
529
+ return 1
530
+ fi
531
+ return 0
532
+ }
533
+
534
+ # =============================================================================
535
+ # T021: 主入口
536
+ # =============================================================================
537
+ parse_args() {
538
+ local req_id=""
539
+ local timeout="$DEFAULT_TIMEOUT"
540
+ local dimension="all"
541
+
542
+ while [[ $# -gt 0 ]]; do
543
+ case "$1" in
544
+ --parallel)
545
+ PARALLEL_MODE=true
546
+ shift
547
+ ;;
548
+ --timeout)
549
+ timeout="$2"
550
+ shift 2
551
+ ;;
552
+ --dimension)
553
+ dimension="$2"
554
+ shift 2
555
+ ;;
556
+ --help|-h)
557
+ echo "Usage: run-clarify-scan.sh REQ-XXX [--parallel] [--timeout N] [--dimension all|1-11]"
558
+ exit 0
559
+ ;;
560
+ *)
561
+ if [[ -z "$req_id" ]]; then
562
+ req_id="$1"
563
+ fi
564
+ shift
565
+ ;;
566
+ esac
567
+ done
568
+
569
+ echo "$req_id|$timeout|$dimension"
570
+ }
571
+
572
+ main() {
573
+ if ! has_valid_api_key; then
574
+ echo "Warning: CLAUDE_API_KEY not set or invalid; using heuristic scan" >&2
575
+ fi
576
+
577
+ # 解析参数
578
+ local parsed
579
+ parsed=$(parse_args "$@")
580
+ local req_id timeout dimension
581
+ IFS='|' read -r req_id timeout dimension <<< "$parsed"
582
+
583
+ # 验证 REQ_ID
584
+ if [[ -z "$req_id" ]]; then
585
+ req_id=$(get_current_req_id)
586
+ fi
587
+
588
+ if ! validate_req_id "$req_id" 2>/dev/null; then
589
+ echo '{"error": {"code": "INVALID_REQ_ID", "message": "Invalid or missing REQ_ID"}}' >&2
590
+ exit 2
591
+ fi
592
+
593
+ # 执行扫描
594
+ scan_all_dimensions "$req_id" "$timeout" "$dimension"
595
+ exit $?
596
+ }
597
+
598
+ # 只在直接执行时运行 main
599
+ if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
600
+ main "$@"
601
+ fi
@@ -0,0 +1,62 @@
1
+ #!/usr/bin/env bash
2
+
3
+ set -euo pipefail
4
+
5
+ usage() {
6
+ cat <<'USAGE'
7
+ Usage: scripts/bash/run-high-review.sh --input "<text>" [--severity high]
8
+
9
+ Generates a placeholder high-rigor review report in reviews/high-review-<timestamp>.md.
10
+ USAGE
11
+ }
12
+
13
+ INPUT=""
14
+
15
+ while [[ $# -gt 0 ]]; do
16
+ case "$1" in
17
+ --input)
18
+ shift
19
+ INPUT="${1:-}"
20
+ ;;
21
+ --severity)
22
+ shift
23
+ # value ignored in placeholder
24
+ ;;
25
+ -h|--help)
26
+ usage
27
+ exit 0
28
+ ;;
29
+ *)
30
+ echo "Unknown argument: $1" >&2
31
+ usage
32
+ exit 1
33
+ ;;
34
+ esac
35
+ shift
36
+ done
37
+
38
+ if [[ -z "$INPUT" ]]; then
39
+ echo "Error: --input payload required." >&2
40
+ usage
41
+ exit 1
42
+ fi
43
+
44
+ REVIEW_DIR="reviews"
45
+ mkdir -p "$REVIEW_DIR"
46
+ timestamp="$(date -u +"%Y%m%d-%H%M%S")"
47
+ report="$REVIEW_DIR/high-review-${timestamp}.md"
48
+
49
+ cat > "$report" <<EOF
50
+ # High-Rigor Code Review (Placeholder)
51
+
52
+ Input:
53
+ \`\`\`
54
+ $INPUT
55
+ \`\`\`
56
+
57
+ Status: TODO - integrate review engine.
58
+
59
+ Please replace this placeholder with the actual review pipeline output.
60
+ EOF
61
+
62
+ echo "Review report generated at $report"