cc-devflow 4.5.10 → 4.5.12

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 (187) hide show
  1. package/.claude/skills/cc-act/CHANGELOG.md +23 -0
  2. package/.claude/skills/cc-act/PLAYBOOK.md +17 -269
  3. package/.claude/skills/cc-act/SKILL.md +38 -418
  4. package/.claude/skills/cc-act/assets/PROJECT_POSTMORTEM_INDEX_TEMPLATE.md +2 -13
  5. package/.claude/skills/cc-act/assets/PROJECT_POSTMORTEM_TEMPLATE.md +1 -9
  6. package/.claude/skills/cc-act/assets/PR_BRIEF_TEMPLATE.md +21 -177
  7. package/.claude/skills/cc-act/references/closure-contract.md +12 -63
  8. package/.claude/skills/cc-act/references/git-commit-guidelines.md +5 -5
  9. package/.claude/skills/cc-act/scripts/cc-act-common.sh +5 -322
  10. package/.claude/skills/cc-act/scripts/detect-ship-target.sh +11 -2
  11. package/.claude/skills/cc-act/scripts/inspect-git-index.sh +58 -0
  12. package/.claude/skills/cc-act/scripts/render-pr-brief.sh +40 -440
  13. package/.claude/skills/cc-act/scripts/verify-act-gate.sh +10 -50
  14. package/.claude/skills/cc-check/CHANGELOG.md +24 -0
  15. package/.claude/skills/cc-check/PLAYBOOK.md +19 -273
  16. package/.claude/skills/cc-check/SKILL.md +33 -454
  17. package/.claude/skills/cc-check/references/review-contract.md +12 -147
  18. package/.claude/skills/cc-dev/CHANGELOG.md +20 -0
  19. package/.claude/skills/cc-dev/PLAYBOOK.md +1 -1
  20. package/.claude/skills/cc-dev/SKILL.md +52 -130
  21. package/.claude/skills/cc-dev/scripts/resolve-cc-devflow.sh +181 -0
  22. package/.claude/skills/cc-do/CHANGELOG.md +17 -0
  23. package/.claude/skills/cc-do/PLAYBOOK.md +19 -113
  24. package/.claude/skills/cc-do/SKILL.md +39 -236
  25. package/.claude/skills/cc-do/references/execution-recovery.md +15 -109
  26. package/.claude/skills/cc-do/scripts/cc-do-common.sh +5 -57
  27. package/.claude/skills/cc-do/scripts/check-task-status.sh +35 -65
  28. package/.claude/skills/cc-do/scripts/mark-task-complete.sh +9 -46
  29. package/.claude/skills/cc-do/scripts/select-ready-tasks.sh +29 -97
  30. package/.claude/skills/cc-investigate/CHANGELOG.md +23 -0
  31. package/.claude/skills/cc-investigate/PLAYBOOK.md +20 -180
  32. package/.claude/skills/cc-investigate/SKILL.md +65 -513
  33. package/.claude/skills/cc-investigate/assets/TASKS_TEMPLATE.md +48 -95
  34. package/.claude/skills/cc-investigate/references/investigation-contract.md +14 -217
  35. package/.claude/skills/cc-next/CHANGELOG.md +6 -0
  36. package/.claude/skills/cc-next/PLAYBOOK.md +12 -8
  37. package/.claude/skills/cc-next/SKILL.md +34 -140
  38. package/.claude/skills/cc-plan/CHANGELOG.md +29 -0
  39. package/.claude/skills/cc-plan/PLAYBOOK.md +22 -161
  40. package/.claude/skills/cc-plan/SKILL.md +47 -640
  41. package/.claude/skills/cc-plan/assets/TASKS_TEMPLATE.md +30 -225
  42. package/.claude/skills/cc-plan/references/planning-contract.md +24 -160
  43. package/.claude/skills/cc-plan/scripts/next-change-key.sh +8 -44
  44. package/.claude/skills/cc-plan/scripts/parse-task-dependencies.js +2 -2
  45. package/.claude/skills/cc-plan/scripts/validate-scope.sh +1 -1
  46. package/.claude/skills/cc-pr-land/SKILL.md +14 -114
  47. package/.claude/skills/cc-pr-review/CHANGELOG.md +4 -0
  48. package/.claude/skills/cc-pr-review/SKILL.md +20 -103
  49. package/.claude/skills/cc-review/CHANGELOG.md +17 -0
  50. package/.claude/skills/cc-review/PLAYBOOK.md +13 -86
  51. package/.claude/skills/cc-review/SKILL.md +53 -241
  52. package/.claude/skills/cc-review/references/e2e-and-plugin-verification.md +2 -2
  53. package/.claude/skills/cc-review/references/implementation-review-branch.md +7 -147
  54. package/.claude/skills/cc-review/references/plan-review-branch.md +5 -147
  55. package/.claude/skills/cc-review/references/review-methods.md +10 -218
  56. package/.claude/skills/cc-review/scripts/collect-review-context.sh +4 -63
  57. package/.claude/skills/cc-roadmap/PLAYBOOK.md +1 -1
  58. package/.claude/skills/cc-roadmap/SKILL.md +3 -3
  59. package/.claude/skills/cc-simplify/CHANGELOG.md +7 -0
  60. package/.claude/skills/cc-simplify/SKILL.md +26 -21
  61. package/.claude/skills/cc-spec-init/PLAYBOOK.md +12 -48
  62. package/.claude/skills/cc-spec-init/SKILL.md +29 -132
  63. package/.claude/skills/cc-spec-init/references/spec-contract.md +8 -17
  64. package/CHANGELOG.md +27 -0
  65. package/README.md +5 -3
  66. package/README.zh-CN.md +5 -3
  67. package/bin/cc-devflow-cli.js +20 -260
  68. package/bin/cc-devflow.js +44 -7
  69. package/docs/commands/README.md +1 -1
  70. package/docs/commands/README.zh-CN.md +1 -1
  71. package/docs/examples/README.md +1 -1
  72. package/docs/examples/START-HERE.md +14 -14
  73. package/docs/examples/example-bindings.json +11 -11
  74. package/docs/examples/full-design-blocked/README.md +4 -6
  75. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/{planning/tasks.md → task.md} +20 -15
  76. package/docs/examples/local-handoff/README.md +8 -11
  77. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/handoff/pr-brief.md +31 -0
  78. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/{planning/tasks.md → task.md} +18 -13
  79. package/docs/examples/pdca-loop/README.md +6 -9
  80. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/handoff/pr-brief.md +9 -11
  81. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/{planning/tasks.md → task.md} +18 -13
  82. package/docs/examples/scripts/check-example-bindings.sh +11 -62
  83. package/docs/guides/artifact-contract.md +10 -36
  84. package/docs/guides/getting-started.md +8 -7
  85. package/docs/guides/getting-started.zh-CN.md +8 -7
  86. package/docs/guides/minimize-artifacts.md +16 -116
  87. package/docs/guides/project-postmortem.md +14 -71
  88. package/lib/compiler/__tests__/skills-registry.test.js +9 -8
  89. package/lib/compiler/resource-copier.js +29 -0
  90. package/lib/skill-runtime/__tests__/archive-change.test.js +2 -2
  91. package/lib/skill-runtime/__tests__/benchmark-skills.test.js +109 -0
  92. package/lib/skill-runtime/__tests__/cli-bootstrap.integration.test.js +14 -4
  93. package/lib/skill-runtime/errors.js +3 -3
  94. package/lib/skill-runtime/index.js +5 -23
  95. package/lib/skill-runtime/paths.js +5 -52
  96. package/lib/skill-runtime/query-registry.js +4 -4
  97. package/lib/skill-runtime/query.js +89 -201
  98. package/lib/skill-runtime/store.js +4 -40
  99. package/lib/skill-runtime/trace.js +2 -2
  100. package/package.json +5 -7
  101. package/.claude/skills/cc-act/assets/PROJECT_POSTMORTEM_PRINCIPLES_TEMPLATE.md +0 -29
  102. package/.claude/skills/cc-act/assets/RELEASE_NOTE_TEMPLATE.md +0 -54
  103. package/.claude/skills/cc-act/scripts/generate-status-report.sh +0 -92
  104. package/.claude/skills/cc-act/scripts/sync-act-docs.sh +0 -355
  105. package/.claude/skills/cc-check/assets/REPORT_CARD_TEMPLATE.json +0 -234
  106. package/.claude/skills/cc-check/scripts/render-report-card.js +0 -438
  107. package/.claude/skills/cc-check/scripts/verify-gate.sh +0 -85
  108. package/.claude/skills/cc-do/scripts/build-task-context.sh +0 -175
  109. package/.claude/skills/cc-do/scripts/record-review-decision.sh +0 -88
  110. package/.claude/skills/cc-do/scripts/recover-workflow.sh +0 -82
  111. package/.claude/skills/cc-do/scripts/run-problem-analysis.sh +0 -70
  112. package/.claude/skills/cc-do/scripts/verify-task-gates.sh +0 -109
  113. package/.claude/skills/cc-do/scripts/write-task-checkpoint.sh +0 -92
  114. package/.claude/skills/cc-investigate/assets/TASK_MANIFEST_TEMPLATE.json +0 -225
  115. package/.claude/skills/cc-plan/assets/TASK_MANIFEST_TEMPLATE.json +0 -179
  116. package/.claude/skills/cc-spec-init/assets/CHANGE_META_TEMPLATE.json +0 -28
  117. package/.claude/skills/cc-spec-init/scripts/validate-spec-links.sh +0 -45
  118. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/planning/design.md +0 -234
  119. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/planning/task-manifest.json +0 -488
  120. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/review/report-card.json +0 -189
  121. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/handoff/resume-index.md +0 -39
  122. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/handoff/status.md +0 -29
  123. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/planning/design.md +0 -123
  124. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/planning/task-manifest.json +0 -292
  125. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/review/report-card.json +0 -136
  126. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/handoff/status.md +0 -29
  127. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/planning/design.md +0 -124
  128. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/planning/task-manifest.json +0 -292
  129. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/review/report-card.json +0 -136
  130. package/docs/get-shit-done-strategy-audit.md +0 -518
  131. package/docs/skill-runtime-migration.md +0 -46
  132. package/lib/skill-runtime/__tests__/approve.test.js +0 -92
  133. package/lib/skill-runtime/__tests__/autopilot.test.js +0 -253
  134. package/lib/skill-runtime/__tests__/benchmark-artifacts.test.js +0 -165
  135. package/lib/skill-runtime/__tests__/delegation.test.js +0 -97
  136. package/lib/skill-runtime/__tests__/dispatch.test.js +0 -237
  137. package/lib/skill-runtime/__tests__/intent.test.js +0 -203
  138. package/lib/skill-runtime/__tests__/lifecycle.test.js +0 -169
  139. package/lib/skill-runtime/__tests__/planner.tdd.test.js +0 -331
  140. package/lib/skill-runtime/__tests__/prepare-pr.test.js +0 -126
  141. package/lib/skill-runtime/__tests__/query.test.js +0 -860
  142. package/lib/skill-runtime/__tests__/readiness.test.js +0 -53
  143. package/lib/skill-runtime/__tests__/release.test.js +0 -85
  144. package/lib/skill-runtime/__tests__/review-check-integration.test.js +0 -148
  145. package/lib/skill-runtime/__tests__/review-records.test.js +0 -619
  146. package/lib/skill-runtime/__tests__/runtime.integration.test.js +0 -351
  147. package/lib/skill-runtime/__tests__/schemas.test.js +0 -337
  148. package/lib/skill-runtime/__tests__/task-contract-migrate.test.js +0 -137
  149. package/lib/skill-runtime/__tests__/task-contract.test.js +0 -783
  150. package/lib/skill-runtime/__tests__/team-state.test.js +0 -51
  151. package/lib/skill-runtime/__tests__/verify-artifacts.test.js +0 -203
  152. package/lib/skill-runtime/__tests__/worker-run.test.js +0 -275
  153. package/lib/skill-runtime/__tests__/worker.test.js +0 -56
  154. package/lib/skill-runtime/__tests__/workflow-context-legacy-fallback.test.js +0 -31
  155. package/lib/skill-runtime/__tests__/workflow-context.test.js +0 -98
  156. package/lib/skill-runtime/artifacts.js +0 -88
  157. package/lib/skill-runtime/context-index.js +0 -545
  158. package/lib/skill-runtime/delegation.js +0 -533
  159. package/lib/skill-runtime/intent.js +0 -309
  160. package/lib/skill-runtime/lifecycle.js +0 -294
  161. package/lib/skill-runtime/operations/CLAUDE.md +0 -19
  162. package/lib/skill-runtime/operations/approve.js +0 -81
  163. package/lib/skill-runtime/operations/autopilot-core.js +0 -337
  164. package/lib/skill-runtime/operations/autopilot-execution.js +0 -307
  165. package/lib/skill-runtime/operations/autopilot-shared.js +0 -48
  166. package/lib/skill-runtime/operations/autopilot.js +0 -163
  167. package/lib/skill-runtime/operations/dispatch.js +0 -416
  168. package/lib/skill-runtime/operations/init.js +0 -60
  169. package/lib/skill-runtime/operations/janitor.js +0 -61
  170. package/lib/skill-runtime/operations/plan.js +0 -59
  171. package/lib/skill-runtime/operations/prepare-pr.js +0 -25
  172. package/lib/skill-runtime/operations/release.js +0 -99
  173. package/lib/skill-runtime/operations/resume.js +0 -126
  174. package/lib/skill-runtime/operations/review-records.js +0 -265
  175. package/lib/skill-runtime/operations/snapshot.js +0 -45
  176. package/lib/skill-runtime/operations/task-contract.js +0 -524
  177. package/lib/skill-runtime/operations/verify.js +0 -170
  178. package/lib/skill-runtime/operations/worker-run.js +0 -531
  179. package/lib/skill-runtime/operations/worker.js +0 -33
  180. package/lib/skill-runtime/planner.js +0 -539
  181. package/lib/skill-runtime/readiness.js +0 -84
  182. package/lib/skill-runtime/review-records.js +0 -123
  183. package/lib/skill-runtime/review.js +0 -855
  184. package/lib/skill-runtime/schemas.js +0 -746
  185. package/lib/skill-runtime/task-contract.js +0 -187
  186. package/lib/skill-runtime/team-state.js +0 -122
  187. package/lib/skill-runtime/workflow-context.js +0 -748
@@ -1,619 +0,0 @@
1
- /**
2
- * [INPUT]: 依赖 lib/skill-runtime/schemas.js 暴露的 ReviewLedgerEventSchema + ReviewFindingsDocSchema 以及对应 parse helpers。
3
- * [OUTPUT]: 通过 zod parse 行为证明 review-ledger.jsonl 每种 event 和 review-findings.json 顶层 doc 的结构契约。
4
- * [POS]: REQ-003-minimize-workflow-artifacts T001 的 Red+Green 合一证据;支撑后续 review CLI 的 schema 边界。
5
- * [PROTOCOL]: 变更时更新此头部,然后检查 CLAUDE.md
6
- */
7
-
8
- const fs = require('fs');
9
- const os = require('os');
10
- const path = require('path');
11
- const { spawnSync } = require('child_process');
12
-
13
- const {
14
- ReviewLedgerEventSchema,
15
- ReviewFindingsDocSchema,
16
- parseReviewLedgerEvent,
17
- parseReviewFindingsDoc
18
- } = require('../schemas');
19
- const { parseReviewLedger } = require('../review-records');
20
-
21
- const CLI = path.resolve(__dirname, '../../../bin/cc-devflow-cli.js');
22
-
23
- const VALID_COMMON = {
24
- schema: 'review-ledger.v2',
25
- change: 'REQ-003-minimize-workflow-artifacts',
26
- reviewId: 'RVW-20260512-001',
27
- createdAt: '2026-05-12T00:00:00.000Z',
28
- createdBy: 'cc-devflow-cli'
29
- };
30
-
31
- function build(eventOverrides) {
32
- return { ...VALID_COMMON, ...eventOverrides };
33
- }
34
-
35
- function readJsonl(filePath) {
36
- return fs.readFileSync(filePath, 'utf8')
37
- .trim()
38
- .split('\n')
39
- .filter(Boolean)
40
- .map((line) => JSON.parse(line));
41
- }
42
-
43
- describe('ReviewLedgerEventSchema', () => {
44
- describe('review-started event', () => {
45
- const validStarted = () => build({
46
- event: 'review-started',
47
- mode: 'implementation',
48
- scope: 'current-diff',
49
- baseSha: 'abc123',
50
- headSha: 'def456',
51
- selectedNodes: ['R001', 'R002'],
52
- skippedNodes: [{ node: 'browser', reason: 'not UI-facing' }],
53
- riskLanes: ['intent-regression', 'contracts-coverage']
54
- });
55
-
56
- test('accepts a well-formed review-started event with selectedNodes and riskLanes', () => {
57
- expect(() => parseReviewLedgerEvent(validStarted())).not.toThrow();
58
- });
59
-
60
- test('rejects review-started event missing baseSha', () => {
61
- const bad = validStarted();
62
- delete bad.baseSha;
63
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/baseSha/);
64
- });
65
-
66
- test('rejects review-started event with unknown mode', () => {
67
- const bad = { ...validStarted(), mode: 'exploration' };
68
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/mode/);
69
- });
70
- });
71
-
72
- describe('review-node-checked event', () => {
73
- const validNode = () => build({
74
- event: 'review-node-checked',
75
- nodeId: 'R001',
76
- mode: 'implementation',
77
- target: 'lib/skill-runtime/workflow-context.js',
78
- status: 'checked',
79
- coverage: ['contract', 'tests'],
80
- evidenceRefs: ['cmd:npm test -- workflow-context'],
81
- findings: [],
82
- next: 'cc-check'
83
- });
84
-
85
- test('accepts a checked node with evidence refs', () => {
86
- expect(() => parseReviewLedgerEvent(validNode())).not.toThrow();
87
- });
88
-
89
- test('rejects a node event with unknown status', () => {
90
- const bad = { ...validNode(), status: 'maybe' };
91
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/status/);
92
- });
93
-
94
- test('rejects a node event with unknown next route', () => {
95
- const bad = { ...validNode(), next: 'cc-unknown' };
96
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/next/);
97
- });
98
- });
99
-
100
- describe('review-finding-added event', () => {
101
- const validFinding = () => build({
102
- event: 'review-finding-added',
103
- findingId: 'F001',
104
- severity: 'important',
105
- confidence: 8,
106
- displayTier: 'blocking',
107
- fingerprint: 'sha256:deadbeef',
108
- scope: 'inside current requirement blast radius',
109
- path: 'lib/skill-runtime/workflow-context.js',
110
- evidence: 'legacyFallback branch still returns exists:false',
111
- recommendation: 'ensure fallback branch probes both design.md and analysis.md',
112
- route: 'cc-do'
113
- });
114
-
115
- test('accepts a well-formed important finding with numeric confidence', () => {
116
- expect(() => parseReviewLedgerEvent(validFinding())).not.toThrow();
117
- });
118
-
119
- test('rejects a finding with unknown severity', () => {
120
- const bad = { ...validFinding(), severity: 'catastrophic' };
121
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/severity/);
122
- });
123
-
124
- test('rejects a finding whose confidence is out of 0..10 range', () => {
125
- const bad = { ...validFinding(), confidence: 11 };
126
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/confidence/);
127
- });
128
-
129
- test('rejects a finding with unknown displayTier', () => {
130
- const bad = { ...validFinding(), displayTier: 'critical-bright-red' };
131
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/displayTier/);
132
- });
133
- });
134
-
135
- describe('review-closed event', () => {
136
- const validClosed = () => build({
137
- event: 'review-closed',
138
- status: 'findings',
139
- blockingCount: 1,
140
- warningCount: 0,
141
- next: 'cc-do'
142
- });
143
-
144
- test('accepts a well-formed review-closed event', () => {
145
- expect(() => parseReviewLedgerEvent(validClosed())).not.toThrow();
146
- });
147
-
148
- test('rejects a close event with unknown status', () => {
149
- const bad = { ...validClosed(), status: 'done' };
150
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/status/);
151
- });
152
-
153
- test('rejects a close event with negative blockingCount', () => {
154
- const bad = { ...validClosed(), blockingCount: -1 };
155
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/blockingCount/);
156
- });
157
- });
158
-
159
- describe('discriminator', () => {
160
- test('rejects an event with unknown literal', () => {
161
- const bad = build({ event: 'review-suspended' });
162
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/event/);
163
- });
164
-
165
- test('rejects an event missing createdBy', () => {
166
- const bad = build({ event: 'review-closed', status: 'clean', blockingCount: 0, warningCount: 0, next: 'cc-check' });
167
- delete bad.createdBy;
168
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/createdBy/);
169
- });
170
-
171
- test('rejects an event with wrong schema string', () => {
172
- const bad = build({ event: 'review-closed', status: 'clean', blockingCount: 0, warningCount: 0, next: 'cc-check' });
173
- bad.schema = 'review-ledger.v1';
174
- expect(() => parseReviewLedgerEvent(bad)).toThrow(/schema/);
175
- });
176
- });
177
- });
178
-
179
- describe('review start CLI', () => {
180
- test('writes one parseable review-started event and prints the reviewId', () => {
181
- const repoRoot = fs.mkdtempSync(path.join(os.tmpdir(), 'cc-devflow-review-start-'));
182
- const changeId = 'REQ-130';
183
- const changeKey = 'REQ-130-review-start';
184
- const result = spawnSync(process.execPath, [
185
- CLI,
186
- 'review',
187
- 'start',
188
- '--cwd',
189
- repoRoot,
190
- '--change',
191
- changeId,
192
- '--change-key',
193
- changeKey,
194
- '--mode',
195
- 'implementation',
196
- '--scope',
197
- 'current-diff',
198
- '--base-sha',
199
- 'abc123',
200
- '--head-sha',
201
- 'def456',
202
- '--selected-node',
203
- 'R001',
204
- '--selected-node',
205
- 'R002',
206
- '--skipped-node',
207
- 'browser:not UI-facing',
208
- '--risk-lane',
209
- 'intent-regression',
210
- '--risk-lane',
211
- 'contracts-coverage'
212
- ], { encoding: 'utf8' });
213
-
214
- expect(result.status).toBe(0);
215
- const stdout = JSON.parse(result.stdout);
216
- expect(stdout.reviewId).toMatch(/^RVW-\d{8}-001$/);
217
-
218
- const ledgerPath = path.join(
219
- repoRoot,
220
- 'devflow',
221
- 'changes',
222
- changeKey,
223
- 'review',
224
- 'review-ledger.jsonl'
225
- );
226
- const events = readJsonl(ledgerPath);
227
- expect(events).toHaveLength(1);
228
-
229
- const event = parseReviewLedgerEvent(events[0]);
230
- expect(event).toMatchObject({
231
- schema: 'review-ledger.v2',
232
- change: changeKey,
233
- reviewId: stdout.reviewId,
234
- createdBy: 'cc-devflow-cli',
235
- event: 'review-started',
236
- mode: 'implementation',
237
- scope: 'current-diff',
238
- baseSha: 'abc123',
239
- headSha: 'def456',
240
- selectedNodes: ['R001', 'R002'],
241
- skippedNodes: [{ node: 'browser', reason: 'not UI-facing' }],
242
- riskLanes: ['intent-regression', 'contracts-coverage']
243
- });
244
-
245
- fs.rmSync(repoRoot, { recursive: true, force: true });
246
- });
247
- });
248
-
249
- describe('review event CLI', () => {
250
- test('appends node, finding, and one close event after review start', () => {
251
- const repoRoot = fs.mkdtempSync(path.join(os.tmpdir(), 'cc-devflow-review-events-'));
252
- const changeId = 'REQ-131';
253
- const changeKey = 'REQ-131-review-events';
254
- const scoped = [
255
- '--cwd',
256
- repoRoot,
257
- '--change',
258
- changeId,
259
- '--change-key',
260
- changeKey
261
- ];
262
- const start = spawnSync(process.execPath, [
263
- CLI,
264
- 'review',
265
- 'start',
266
- ...scoped,
267
- '--base-sha',
268
- 'abc123',
269
- '--head-sha',
270
- 'def456'
271
- ], { encoding: 'utf8' });
272
- const { reviewId } = JSON.parse(start.stdout);
273
-
274
- const recordNode = spawnSync(process.execPath, [
275
- CLI,
276
- 'review',
277
- 'record-node',
278
- ...scoped,
279
- '--review-id',
280
- reviewId,
281
- '--node-id',
282
- 'R001',
283
- '--mode',
284
- 'implementation',
285
- '--target',
286
- 'lib/skill-runtime/workflow-context.js',
287
- '--status',
288
- 'checked',
289
- '--coverage',
290
- 'contract',
291
- '--coverage',
292
- 'tests',
293
- '--evidence-ref',
294
- 'cmd:npm test -- review-records',
295
- '--finding',
296
- 'F001',
297
- '--next',
298
- 'cc-do'
299
- ], { encoding: 'utf8' });
300
- const addFinding = spawnSync(process.execPath, [
301
- CLI,
302
- 'review',
303
- 'add-finding',
304
- ...scoped,
305
- '--review-id',
306
- reviewId,
307
- '--finding-id',
308
- 'F001',
309
- '--severity',
310
- 'important',
311
- '--confidence',
312
- '8',
313
- '--display-tier',
314
- 'blocking',
315
- '--fingerprint',
316
- 'sha256:abc123',
317
- '--scope',
318
- 'inside current requirement blast radius',
319
- '--path',
320
- 'lib/skill-runtime/workflow-context.js',
321
- '--evidence',
322
- 'review start has no follow-up event writer',
323
- '--recommendation',
324
- 'append node and finding events through review CLI',
325
- '--route',
326
- 'cc-do'
327
- ], { encoding: 'utf8' });
328
- const close = spawnSync(process.execPath, [
329
- CLI,
330
- 'review',
331
- 'close',
332
- ...scoped,
333
- '--review-id',
334
- reviewId,
335
- '--status',
336
- 'findings',
337
- '--blocking-count',
338
- '1',
339
- '--warning-count',
340
- '0',
341
- '--next',
342
- 'cc-do'
343
- ], { encoding: 'utf8' });
344
- const duplicateClose = spawnSync(process.execPath, [
345
- CLI,
346
- 'review',
347
- 'close',
348
- ...scoped,
349
- '--review-id',
350
- reviewId,
351
- '--status',
352
- 'findings',
353
- '--blocking-count',
354
- '1',
355
- '--warning-count',
356
- '0',
357
- '--next',
358
- 'cc-do'
359
- ], { encoding: 'utf8' });
360
-
361
- expect(recordNode.status).toBe(0);
362
- expect(addFinding.status).toBe(0);
363
- expect(close.status).toBe(0);
364
- expect(duplicateClose.status).not.toBe(0);
365
-
366
- const ledgerPath = path.join(
367
- repoRoot,
368
- 'devflow',
369
- 'changes',
370
- changeKey,
371
- 'review',
372
- 'review-ledger.jsonl'
373
- );
374
- const events = readJsonl(ledgerPath).map(parseReviewLedgerEvent);
375
- expect(events.map((event) => event.event)).toEqual([
376
- 'review-started',
377
- 'review-node-checked',
378
- 'review-finding-added',
379
- 'review-closed'
380
- ]);
381
- expect(events[1]).toMatchObject({
382
- reviewId,
383
- nodeId: 'R001',
384
- coverage: ['contract', 'tests'],
385
- evidenceRefs: ['cmd:npm test -- review-records'],
386
- findings: ['F001'],
387
- next: 'cc-do'
388
- });
389
- expect(events[2]).toMatchObject({
390
- reviewId,
391
- findingId: 'F001',
392
- severity: 'important',
393
- confidence: 8,
394
- displayTier: 'blocking',
395
- route: 'cc-do'
396
- });
397
- expect(events[3]).toMatchObject({
398
- reviewId,
399
- status: 'findings',
400
- blockingCount: 1,
401
- warningCount: 0,
402
- next: 'cc-do'
403
- });
404
-
405
- fs.rmSync(repoRoot, { recursive: true, force: true });
406
- });
407
- });
408
-
409
- describe('parseReviewLedger', () => {
410
- test('degrades malformed ledger rows to unknown freshness without throwing', () => {
411
- const parsed = parseReviewLedger([
412
- JSON.stringify(build({
413
- event: 'review-started',
414
- mode: 'implementation',
415
- scope: 'current-diff',
416
- baseSha: 'abc123',
417
- headSha: 'def456',
418
- selectedNodes: [],
419
- skippedNodes: [],
420
- riskLanes: []
421
- })),
422
- '{"schema":"review-ledger.v1","event":"review-closed"}'
423
- ].join('\n'));
424
-
425
- expect(parsed.entries).toHaveLength(1);
426
- expect(parsed.errors).toHaveLength(1);
427
- expect(parsed.freshness).toMatchObject({ status: 'unknown' });
428
- });
429
- });
430
-
431
- describe('review render CLI', () => {
432
- test('renders Markdown from ledger events on demand', () => {
433
- const repoRoot = fs.mkdtempSync(path.join(os.tmpdir(), 'cc-devflow-review-render-'));
434
- const changeId = 'REQ-132';
435
- const changeKey = 'REQ-132-review-render';
436
- const outputPath = path.join(repoRoot, 'review-output.md');
437
- const scoped = [
438
- '--cwd',
439
- repoRoot,
440
- '--change',
441
- changeId,
442
- '--change-key',
443
- changeKey
444
- ];
445
- const start = spawnSync(process.execPath, [
446
- CLI,
447
- 'review',
448
- 'start',
449
- ...scoped,
450
- '--base-sha',
451
- 'abc123',
452
- '--head-sha',
453
- 'def456'
454
- ], { encoding: 'utf8' });
455
- const { reviewId } = JSON.parse(start.stdout);
456
-
457
- spawnSync(process.execPath, [
458
- CLI,
459
- 'review',
460
- 'record-node',
461
- ...scoped,
462
- '--review-id',
463
- reviewId,
464
- '--node-id',
465
- 'R001',
466
- '--target',
467
- 'lib/skill-runtime/workflow-context.js',
468
- '--status',
469
- 'checked',
470
- '--coverage',
471
- 'contract',
472
- '--evidence-ref',
473
- 'cmd:npm test -- review-records',
474
- '--next',
475
- 'cc-do'
476
- ], { encoding: 'utf8' });
477
- spawnSync(process.execPath, [
478
- CLI,
479
- 'review',
480
- 'add-finding',
481
- ...scoped,
482
- '--review-id',
483
- reviewId,
484
- '--finding-id',
485
- 'F001',
486
- '--severity',
487
- 'important',
488
- '--confidence',
489
- '8',
490
- '--display-tier',
491
- 'blocking',
492
- '--fingerprint',
493
- 'sha256:abc123',
494
- '--scope',
495
- 'inside current requirement blast radius',
496
- '--path',
497
- 'lib/skill-runtime/workflow-context.js',
498
- '--evidence',
499
- 'review ledger needs Markdown output',
500
- '--recommendation',
501
- 'render a compact review report from ledger events',
502
- '--route',
503
- 'cc-do'
504
- ], { encoding: 'utf8' });
505
- spawnSync(process.execPath, [
506
- CLI,
507
- 'review',
508
- 'close',
509
- ...scoped,
510
- '--review-id',
511
- reviewId,
512
- '--status',
513
- 'findings',
514
- '--blocking-count',
515
- '1',
516
- '--warning-count',
517
- '0',
518
- '--next',
519
- 'cc-do'
520
- ], { encoding: 'utf8' });
521
-
522
- const render = spawnSync(process.execPath, [
523
- CLI,
524
- 'review',
525
- 'render',
526
- ...scoped,
527
- '--review-id',
528
- reviewId,
529
- '--output',
530
- outputPath
531
- ], { encoding: 'utf8' });
532
-
533
- expect(render.status).toBe(0);
534
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining(`reviewId: ${reviewId}`));
535
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining('Output language: en'));
536
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining('## Summary'));
537
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining('## Findings'));
538
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining('F001'));
539
- expect(fs.readFileSync(outputPath, 'utf8')).toEqual(expect.stringContaining('## Ledger Events'));
540
-
541
- fs.rmSync(repoRoot, { recursive: true, force: true });
542
- });
543
- });
544
-
545
- describe('ReviewFindingsDocSchema', () => {
546
- const validDoc = () => ({
547
- schema: 'review-findings.v2',
548
- change: 'REQ-003-minimize-workflow-artifacts',
549
- reviewId: 'RVW-20260512-001',
550
- headSha: 'def456',
551
- freshness: {
552
- status: 'fresh',
553
- reviewedCommit: 'def456',
554
- currentCommit: 'def456',
555
- commitsSinceReview: 0
556
- },
557
- summary: {
558
- status: 'findings',
559
- blockingCount: 1,
560
- warningCount: 0,
561
- next: 'cc-do'
562
- },
563
- findings: [
564
- {
565
- id: 'F001',
566
- severity: 'important',
567
- confidence: 8,
568
- displayTier: 'blocking',
569
- fingerprint: 'sha256:deadbeef',
570
- scope: 'inside current requirement blast radius',
571
- path: 'lib/skill-runtime/workflow-context.js',
572
- evidence: 'legacyFallback branch still returns exists:false',
573
- recommendation: 'ensure fallback branch probes both design.md and analysis.md',
574
- route: 'cc-do'
575
- }
576
- ]
577
- });
578
-
579
- test('accepts a well-formed findings doc with one finding', () => {
580
- expect(() => parseReviewFindingsDoc(validDoc())).not.toThrow();
581
- });
582
-
583
- test('accepts a clean findings doc with empty findings array', () => {
584
- const doc = validDoc();
585
- doc.summary = { status: 'clean', blockingCount: 0, warningCount: 0, next: 'cc-check' };
586
- doc.findings = [];
587
- expect(() => parseReviewFindingsDoc(doc)).not.toThrow();
588
- });
589
-
590
- test('rejects a findings doc with unknown freshness.status', () => {
591
- const bad = validDoc();
592
- bad.freshness.status = 'warm';
593
- expect(() => parseReviewFindingsDoc(bad)).toThrow(/freshness/);
594
- });
595
-
596
- test('rejects a findings doc with unknown summary.status', () => {
597
- const bad = validDoc();
598
- bad.summary.status = 'maybe';
599
- expect(() => parseReviewFindingsDoc(bad)).toThrow(/summary/);
600
- });
601
-
602
- test('rejects a findings doc whose schema string is not review-findings.v2', () => {
603
- const bad = validDoc();
604
- bad.schema = 'review-findings.v1';
605
- expect(() => parseReviewFindingsDoc(bad)).toThrow(/schema/);
606
- });
607
-
608
- test('rejects a finding entry with unknown severity', () => {
609
- const bad = validDoc();
610
- bad.findings[0].severity = 'catastrophic';
611
- expect(() => parseReviewFindingsDoc(bad)).toThrow(/severity/);
612
- });
613
-
614
- test('rejects a finding entry with confidence above 10', () => {
615
- const bad = validDoc();
616
- bad.findings[0].confidence = 12;
617
- expect(() => parseReviewFindingsDoc(bad)).toThrow(/confidence/);
618
- });
619
- });