@paths.design/caws-cli 10.1.0 → 11.0.0

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 (419) hide show
  1. package/README.md +125 -374
  2. package/dist/index.js +43 -756
  3. package/dist/shell/binding/resolve-binding.d.ts +4 -0
  4. package/dist/shell/binding/resolve-binding.d.ts.map +1 -0
  5. package/dist/shell/binding/resolve-binding.js +228 -0
  6. package/dist/shell/binding/resolve-binding.js.map +1 -0
  7. package/dist/shell/binding/types.d.ts +42 -0
  8. package/dist/shell/binding/types.d.ts.map +1 -0
  9. package/dist/shell/binding/types.js +21 -0
  10. package/dist/shell/binding/types.js.map +1 -0
  11. package/dist/shell/commands/claim.d.ts +14 -0
  12. package/dist/shell/commands/claim.d.ts.map +1 -0
  13. package/dist/shell/commands/claim.js +197 -0
  14. package/dist/shell/commands/claim.js.map +1 -0
  15. package/dist/shell/commands/doctor.d.ts +13 -0
  16. package/dist/shell/commands/doctor.d.ts.map +1 -0
  17. package/dist/shell/commands/doctor.js +97 -0
  18. package/dist/shell/commands/doctor.js.map +1 -0
  19. package/dist/shell/commands/evidence.d.ts +28 -0
  20. package/dist/shell/commands/evidence.d.ts.map +1 -0
  21. package/dist/shell/commands/evidence.js +166 -0
  22. package/dist/shell/commands/evidence.js.map +1 -0
  23. package/dist/shell/commands/gates.d.ts +19 -0
  24. package/dist/shell/commands/gates.d.ts.map +1 -0
  25. package/dist/shell/commands/gates.js +181 -0
  26. package/dist/shell/commands/gates.js.map +1 -0
  27. package/dist/shell/commands/init.d.ts +8 -0
  28. package/dist/shell/commands/init.d.ts.map +1 -0
  29. package/dist/shell/commands/init.js +64 -0
  30. package/dist/shell/commands/init.js.map +1 -0
  31. package/dist/shell/commands/scope.d.ts +11 -0
  32. package/dist/shell/commands/scope.d.ts.map +1 -0
  33. package/dist/shell/commands/scope.js +92 -0
  34. package/dist/shell/commands/scope.js.map +1 -0
  35. package/dist/shell/commands/status.d.ts +15 -0
  36. package/dist/shell/commands/status.d.ts.map +1 -0
  37. package/dist/shell/commands/status.js +106 -0
  38. package/dist/shell/commands/status.js.map +1 -0
  39. package/dist/shell/commands/waiver.d.ts +38 -0
  40. package/dist/shell/commands/waiver.d.ts.map +1 -0
  41. package/dist/shell/commands/waiver.js +240 -0
  42. package/dist/shell/commands/waiver.js.map +1 -0
  43. package/dist/shell/gates/disposition.d.ts +23 -0
  44. package/dist/shell/gates/disposition.d.ts.map +1 -0
  45. package/dist/shell/gates/disposition.js +87 -0
  46. package/dist/shell/gates/disposition.js.map +1 -0
  47. package/dist/shell/gates/gate-result-contract.d.ts +39 -0
  48. package/dist/shell/gates/gate-result-contract.d.ts.map +1 -0
  49. package/dist/shell/gates/gate-result-contract.js +150 -0
  50. package/dist/shell/gates/gate-result-contract.js.map +1 -0
  51. package/dist/shell/gates/quality-gates-adapter.d.ts +55 -0
  52. package/dist/shell/gates/quality-gates-adapter.d.ts.map +1 -0
  53. package/dist/shell/gates/quality-gates-adapter.js +161 -0
  54. package/dist/shell/gates/quality-gates-adapter.js.map +1 -0
  55. package/dist/shell/gates/waiver-filter.d.ts +58 -0
  56. package/dist/shell/gates/waiver-filter.d.ts.map +1 -0
  57. package/dist/shell/gates/waiver-filter.js +119 -0
  58. package/dist/shell/gates/waiver-filter.js.map +1 -0
  59. package/dist/shell/index.d.ts +50 -0
  60. package/dist/shell/index.d.ts.map +1 -0
  61. package/dist/shell/index.js +73 -0
  62. package/dist/shell/index.js.map +1 -0
  63. package/dist/shell/register.d.ts +11 -0
  64. package/dist/shell/register.d.ts.map +1 -0
  65. package/dist/shell/register.js +274 -0
  66. package/dist/shell/register.js.map +1 -0
  67. package/dist/shell/render/claim.d.ts +22 -0
  68. package/dist/shell/render/claim.d.ts.map +1 -0
  69. package/dist/shell/render/claim.js +75 -0
  70. package/dist/shell/render/claim.js.map +1 -0
  71. package/dist/shell/render/decision.d.ts +15 -0
  72. package/dist/shell/render/decision.d.ts.map +1 -0
  73. package/dist/shell/render/decision.js +66 -0
  74. package/dist/shell/render/decision.js.map +1 -0
  75. package/dist/shell/render/diagnostic.d.ts +19 -0
  76. package/dist/shell/render/diagnostic.d.ts.map +1 -0
  77. package/dist/shell/render/diagnostic.js +76 -0
  78. package/dist/shell/render/diagnostic.js.map +1 -0
  79. package/dist/shell/render/finding.d.ts +15 -0
  80. package/dist/shell/render/finding.d.ts.map +1 -0
  81. package/dist/shell/render/finding.js +57 -0
  82. package/dist/shell/render/finding.js.map +1 -0
  83. package/dist/shell/render/gates.d.ts +3 -0
  84. package/dist/shell/render/gates.d.ts.map +1 -0
  85. package/dist/shell/render/gates.js +56 -0
  86. package/dist/shell/render/gates.js.map +1 -0
  87. package/dist/shell/render/init.d.ts +11 -0
  88. package/dist/shell/render/init.d.ts.map +1 -0
  89. package/dist/shell/render/init.js +32 -0
  90. package/dist/shell/render/init.js.map +1 -0
  91. package/dist/shell/render/status.d.ts +26 -0
  92. package/dist/shell/render/status.d.ts.map +1 -0
  93. package/dist/shell/render/status.js +143 -0
  94. package/dist/shell/render/status.js.map +1 -0
  95. package/dist/shell/render/waiver.d.ts +21 -0
  96. package/dist/shell/render/waiver.d.ts.map +1 -0
  97. package/dist/shell/render/waiver.js +94 -0
  98. package/dist/shell/render/waiver.js.map +1 -0
  99. package/dist/shell/rules.d.ts +37 -0
  100. package/dist/shell/rules.d.ts.map +1 -0
  101. package/dist/shell/rules.js +51 -0
  102. package/dist/shell/rules.js.map +1 -0
  103. package/dist/shell/session/actor.d.ts +14 -0
  104. package/dist/shell/session/actor.d.ts.map +1 -0
  105. package/dist/shell/session/actor.js +34 -0
  106. package/dist/shell/session/actor.js.map +1 -0
  107. package/dist/shell/session/resolve-session.d.ts +5 -0
  108. package/dist/shell/session/resolve-session.d.ts.map +1 -0
  109. package/dist/shell/session/resolve-session.js +239 -0
  110. package/dist/shell/session/resolve-session.js.map +1 -0
  111. package/dist/shell/session/types.d.ts +56 -0
  112. package/dist/shell/session/types.d.ts.map +1 -0
  113. package/dist/shell/session/types.js +15 -0
  114. package/dist/shell/session/types.js.map +1 -0
  115. package/dist/store/agents-store.d.ts +3 -0
  116. package/dist/store/agents-store.d.ts.map +1 -0
  117. package/dist/store/agents-store.js +63 -0
  118. package/dist/store/agents-store.js.map +1 -0
  119. package/dist/store/apply-patch.d.ts +16 -0
  120. package/dist/store/apply-patch.d.ts.map +1 -0
  121. package/dist/store/apply-patch.js +191 -0
  122. package/dist/store/apply-patch.js.map +1 -0
  123. package/dist/store/atomic-write.d.ts +16 -0
  124. package/dist/store/atomic-write.d.ts.map +1 -0
  125. package/dist/store/atomic-write.js +132 -0
  126. package/dist/store/atomic-write.js.map +1 -0
  127. package/dist/store/doctor-snapshot.d.ts +20 -0
  128. package/dist/store/doctor-snapshot.d.ts.map +1 -0
  129. package/dist/store/doctor-snapshot.js +176 -0
  130. package/dist/store/doctor-snapshot.js.map +1 -0
  131. package/dist/store/events-store.d.ts +33 -0
  132. package/dist/store/events-store.d.ts.map +1 -0
  133. package/dist/store/events-store.js +297 -0
  134. package/dist/store/events-store.js.map +1 -0
  135. package/dist/store/index.d.ts +21 -0
  136. package/dist/store/index.d.ts.map +1 -0
  137. package/dist/store/index.js +47 -0
  138. package/dist/store/index.js.map +1 -0
  139. package/dist/store/init-store.d.ts +21 -0
  140. package/dist/store/init-store.d.ts.map +1 -0
  141. package/dist/store/init-store.js +295 -0
  142. package/dist/store/init-store.js.map +1 -0
  143. package/dist/store/json-store.d.ts +3 -0
  144. package/dist/store/json-store.d.ts.map +1 -0
  145. package/dist/store/json-store.js +65 -0
  146. package/dist/store/json-store.js.map +1 -0
  147. package/dist/store/policy-store.d.ts +3 -0
  148. package/dist/store/policy-store.d.ts.map +1 -0
  149. package/dist/store/policy-store.js +65 -0
  150. package/dist/store/policy-store.js.map +1 -0
  151. package/dist/store/repo-root.d.ts +46 -0
  152. package/dist/store/repo-root.d.ts.map +1 -0
  153. package/dist/store/repo-root.js +145 -0
  154. package/dist/store/repo-root.js.map +1 -0
  155. package/dist/store/rules.d.ts +53 -0
  156. package/dist/store/rules.d.ts.map +1 -0
  157. package/dist/store/rules.js +78 -0
  158. package/dist/store/rules.js.map +1 -0
  159. package/dist/store/specs-store.d.ts +3 -0
  160. package/dist/store/specs-store.d.ts.map +1 -0
  161. package/dist/store/specs-store.js +131 -0
  162. package/dist/store/specs-store.js.map +1 -0
  163. package/dist/store/types.d.ts +84 -0
  164. package/dist/store/types.d.ts.map +1 -0
  165. package/dist/store/types.js +14 -0
  166. package/dist/store/types.js.map +1 -0
  167. package/dist/store/waivers-store.d.ts +25 -0
  168. package/dist/store/waivers-store.d.ts.map +1 -0
  169. package/dist/store/waivers-store.js +232 -0
  170. package/dist/store/waivers-store.js.map +1 -0
  171. package/dist/store/worktrees-store.d.ts +3 -0
  172. package/dist/store/worktrees-store.d.ts.map +1 -0
  173. package/dist/store/worktrees-store.js +62 -0
  174. package/dist/store/worktrees-store.js.map +1 -0
  175. package/dist/store/yaml-store.d.ts +9 -0
  176. package/dist/store/yaml-store.d.ts.map +1 -0
  177. package/dist/store/yaml-store.js +121 -0
  178. package/dist/store/yaml-store.js.map +1 -0
  179. package/package.json +15 -13
  180. package/dist/budget-derivation.js +0 -751
  181. package/dist/cicd-optimizer.js +0 -504
  182. package/dist/commands/archive.js +0 -500
  183. package/dist/commands/burnup.js +0 -198
  184. package/dist/commands/diagnose.js +0 -525
  185. package/dist/commands/evaluate.js +0 -314
  186. package/dist/commands/gates.js +0 -149
  187. package/dist/commands/init.js +0 -857
  188. package/dist/commands/iterate.js +0 -417
  189. package/dist/commands/mode.js +0 -269
  190. package/dist/commands/parallel.js +0 -242
  191. package/dist/commands/plan.js +0 -438
  192. package/dist/commands/provenance.js +0 -1143
  193. package/dist/commands/quality-monitor.js +0 -284
  194. package/dist/commands/scope.js +0 -264
  195. package/dist/commands/session.js +0 -312
  196. package/dist/commands/sidecar.js +0 -74
  197. package/dist/commands/specs.js +0 -1448
  198. package/dist/commands/status.js +0 -1151
  199. package/dist/commands/templates.js +0 -237
  200. package/dist/commands/tool.js +0 -136
  201. package/dist/commands/tutorial.js +0 -480
  202. package/dist/commands/validate.js +0 -357
  203. package/dist/commands/verify-acs.js +0 -443
  204. package/dist/commands/waivers.js +0 -599
  205. package/dist/commands/workflow.js +0 -243
  206. package/dist/commands/worktree.js +0 -386
  207. package/dist/config/lite-scope.js +0 -158
  208. package/dist/config/modes.js +0 -347
  209. package/dist/constants/spec-types.js +0 -65
  210. package/dist/gates/budget-limit.js +0 -121
  211. package/dist/gates/feedback.js +0 -260
  212. package/dist/gates/format.js +0 -179
  213. package/dist/gates/god-object.js +0 -117
  214. package/dist/gates/pipeline.js +0 -167
  215. package/dist/gates/scope-boundary.js +0 -93
  216. package/dist/gates/spec-completeness.js +0 -109
  217. package/dist/gates/todo-detection.js +0 -205
  218. package/dist/generators/jest-config-generator.js +0 -242
  219. package/dist/generators/working-spec.js +0 -237
  220. package/dist/minimal-cli.js +0 -88
  221. package/dist/parallel/parallel-manager.js +0 -433
  222. package/dist/policy/PolicyManager.js +0 -465
  223. package/dist/scaffold/claude-hooks.js +0 -443
  224. package/dist/scaffold/cursor-hooks.js +0 -177
  225. package/dist/scaffold/git-hooks.js +0 -928
  226. package/dist/scaffold/index.js +0 -794
  227. package/dist/session/session-manager.js +0 -653
  228. package/dist/sidecars/index.js +0 -33
  229. package/dist/sidecars/listeners.js +0 -40
  230. package/dist/sidecars/provenance-summary.js +0 -238
  231. package/dist/sidecars/quality-gaps.js +0 -258
  232. package/dist/sidecars/schema.js +0 -149
  233. package/dist/sidecars/spec-drift.js +0 -151
  234. package/dist/sidecars/waiver-draft.js +0 -176
  235. package/dist/spec/SpecFileManager.js +0 -419
  236. package/dist/templates/.caws/schemas/policy.schema.json +0 -112
  237. package/dist/templates/.caws/schemas/scope.schema.json +0 -52
  238. package/dist/templates/.caws/schemas/waivers.schema.json +0 -106
  239. package/dist/templates/.caws/schemas/working-spec.schema.json +0 -340
  240. package/dist/templates/.caws/schemas/worktrees.schema.json +0 -38
  241. package/dist/templates/.caws/templates/working-spec.template.yml +0 -80
  242. package/dist/templates/.caws/tools/README.md +0 -18
  243. package/dist/templates/.caws/tools/scope-guard.js +0 -203
  244. package/dist/templates/.caws/tools-allow.json +0 -331
  245. package/dist/templates/.caws/waivers.yml +0 -19
  246. package/dist/templates/.claude/README.md +0 -190
  247. package/dist/templates/.claude/hooks/audit.sh +0 -121
  248. package/dist/templates/.claude/hooks/block-dangerous.sh +0 -203
  249. package/dist/templates/.claude/hooks/classify_command.py +0 -592
  250. package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  251. package/dist/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  252. package/dist/templates/.claude/hooks/naming-check.sh +0 -100
  253. package/dist/templates/.claude/hooks/protected-paths.sh +0 -39
  254. package/dist/templates/.claude/hooks/quality-check.sh +0 -81
  255. package/dist/templates/.claude/hooks/scan-secrets.sh +0 -85
  256. package/dist/templates/.claude/hooks/scope-guard.sh +0 -381
  257. package/dist/templates/.claude/hooks/session-caws-status.sh +0 -117
  258. package/dist/templates/.claude/hooks/session-log.sh +0 -634
  259. package/dist/templates/.claude/hooks/simplification-guard.sh +0 -92
  260. package/dist/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  261. package/dist/templates/.claude/hooks/test_classify_command.py +0 -370
  262. package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  263. package/dist/templates/.claude/hooks/validate-spec.sh +0 -76
  264. package/dist/templates/.claude/hooks/worktree-guard.sh +0 -220
  265. package/dist/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  266. package/dist/templates/.claude/rules/git-safety.md +0 -26
  267. package/dist/templates/.claude/rules/worktree-isolation.md +0 -83
  268. package/dist/templates/.claude/settings.json +0 -141
  269. package/dist/templates/.cursor/README.md +0 -299
  270. package/dist/templates/.cursor/hooks/audit.sh +0 -55
  271. package/dist/templates/.cursor/hooks/block-dangerous.sh +0 -84
  272. package/dist/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  273. package/dist/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  274. package/dist/templates/.cursor/hooks/format.sh +0 -38
  275. package/dist/templates/.cursor/hooks/naming-check.sh +0 -64
  276. package/dist/templates/.cursor/hooks/scan-secrets.sh +0 -51
  277. package/dist/templates/.cursor/hooks/scope-guard.sh +0 -52
  278. package/dist/templates/.cursor/hooks/session-log.sh +0 -924
  279. package/dist/templates/.cursor/hooks/validate-spec.sh +0 -83
  280. package/dist/templates/.cursor/hooks.json +0 -76
  281. package/dist/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  282. package/dist/templates/.cursor/rules/01-working-style.mdc +0 -50
  283. package/dist/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  284. package/dist/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  285. package/dist/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  286. package/dist/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  287. package/dist/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  288. package/dist/templates/.cursor/rules/07-process-ops.mdc +0 -20
  289. package/dist/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  290. package/dist/templates/.cursor/rules/09-docstrings.mdc +0 -89
  291. package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  292. package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  293. package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  294. package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  295. package/dist/templates/.cursor/rules/README.md +0 -148
  296. package/dist/templates/.github/copilot-instructions.md +0 -82
  297. package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  298. package/dist/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  299. package/dist/templates/.junie/guidelines.md +0 -73
  300. package/dist/templates/.vscode/launch.json +0 -17
  301. package/dist/templates/.vscode/settings.json +0 -95
  302. package/dist/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  303. package/dist/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  304. package/dist/templates/CLAUDE.md +0 -174
  305. package/dist/templates/COMMIT_CONVENTIONS.md +0 -86
  306. package/dist/templates/OIDC_SETUP.md +0 -300
  307. package/dist/templates/agents.md +0 -145
  308. package/dist/templates/codemod/README.md +0 -1
  309. package/dist/templates/codemod/test.js +0 -93
  310. package/dist/templates/docs/README.md +0 -151
  311. package/dist/templates/scripts/new_feature.sh +0 -80
  312. package/dist/templates/scripts/quality-gates/check-god-objects.js +0 -146
  313. package/dist/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  314. package/dist/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
  315. package/dist/test-analysis.js +0 -786
  316. package/dist/tool-interface.js +0 -314
  317. package/dist/tool-loader.js +0 -303
  318. package/dist/tool-validator.js +0 -393
  319. package/dist/utils/agent-session.js +0 -202
  320. package/dist/utils/async-utils.js +0 -188
  321. package/dist/utils/command-wrapper.js +0 -200
  322. package/dist/utils/event-log.js +0 -584
  323. package/dist/utils/event-renderer.js +0 -521
  324. package/dist/utils/finalization.js +0 -230
  325. package/dist/utils/git-lock.js +0 -119
  326. package/dist/utils/gitignore-updater.js +0 -158
  327. package/dist/utils/ide-detection.js +0 -133
  328. package/dist/utils/lifecycle-events.js +0 -94
  329. package/dist/utils/project-analysis.js +0 -367
  330. package/dist/utils/promise-utils.js +0 -72
  331. package/dist/utils/quality-gates-errors.js +0 -520
  332. package/dist/utils/quality-gates-utils.js +0 -387
  333. package/dist/utils/schema-validator.js +0 -50
  334. package/dist/utils/spec-resolver.js +0 -711
  335. package/dist/utils/typescript-detector.js +0 -369
  336. package/dist/utils/working-state.js +0 -530
  337. package/dist/utils/yaml-validation.js +0 -156
  338. package/dist/validation/spec-validation.js +0 -921
  339. package/dist/waivers-manager.js +0 -732
  340. package/dist/worktree/worktree-manager.js +0 -1374
  341. package/templates/.caws/schemas/policy.schema.json +0 -112
  342. package/templates/.caws/schemas/scope.schema.json +0 -52
  343. package/templates/.caws/schemas/waivers.schema.json +0 -106
  344. package/templates/.caws/schemas/working-spec.schema.json +0 -340
  345. package/templates/.caws/schemas/worktrees.schema.json +0 -38
  346. package/templates/.caws/templates/working-spec.template.yml +0 -80
  347. package/templates/.caws/tools/README.md +0 -18
  348. package/templates/.caws/tools/scope-guard.js +0 -203
  349. package/templates/.caws/tools-allow.json +0 -331
  350. package/templates/.caws/waivers.yml +0 -19
  351. package/templates/.claude/README.md +0 -190
  352. package/templates/.claude/hooks/audit.sh +0 -121
  353. package/templates/.claude/hooks/block-dangerous.sh +0 -203
  354. package/templates/.claude/hooks/classify_command.py +0 -592
  355. package/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  356. package/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  357. package/templates/.claude/hooks/naming-check.sh +0 -100
  358. package/templates/.claude/hooks/protected-paths.sh +0 -39
  359. package/templates/.claude/hooks/quality-check.sh +0 -81
  360. package/templates/.claude/hooks/scan-secrets.sh +0 -85
  361. package/templates/.claude/hooks/scope-guard.sh +0 -381
  362. package/templates/.claude/hooks/session-caws-status.sh +0 -117
  363. package/templates/.claude/hooks/session-log.sh +0 -634
  364. package/templates/.claude/hooks/simplification-guard.sh +0 -92
  365. package/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  366. package/templates/.claude/hooks/test_classify_command.py +0 -370
  367. package/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  368. package/templates/.claude/hooks/validate-spec.sh +0 -76
  369. package/templates/.claude/hooks/worktree-guard.sh +0 -220
  370. package/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  371. package/templates/.claude/rules/git-safety.md +0 -26
  372. package/templates/.claude/rules/worktree-isolation.md +0 -83
  373. package/templates/.claude/settings.json +0 -141
  374. package/templates/.cursor/README.md +0 -299
  375. package/templates/.cursor/hooks/audit.sh +0 -55
  376. package/templates/.cursor/hooks/block-dangerous.sh +0 -84
  377. package/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  378. package/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  379. package/templates/.cursor/hooks/format.sh +0 -38
  380. package/templates/.cursor/hooks/naming-check.sh +0 -64
  381. package/templates/.cursor/hooks/scan-secrets.sh +0 -51
  382. package/templates/.cursor/hooks/scope-guard.sh +0 -52
  383. package/templates/.cursor/hooks/session-log.sh +0 -924
  384. package/templates/.cursor/hooks/validate-spec.sh +0 -83
  385. package/templates/.cursor/hooks.json +0 -76
  386. package/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  387. package/templates/.cursor/rules/01-working-style.mdc +0 -50
  388. package/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  389. package/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  390. package/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  391. package/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  392. package/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  393. package/templates/.cursor/rules/07-process-ops.mdc +0 -20
  394. package/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  395. package/templates/.cursor/rules/09-docstrings.mdc +0 -89
  396. package/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  397. package/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  398. package/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  399. package/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  400. package/templates/.cursor/rules/README.md +0 -148
  401. package/templates/.github/copilot-instructions.md +0 -82
  402. package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  403. package/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  404. package/templates/.junie/guidelines.md +0 -73
  405. package/templates/.vscode/launch.json +0 -17
  406. package/templates/.vscode/settings.json +0 -95
  407. package/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  408. package/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  409. package/templates/CLAUDE.md +0 -174
  410. package/templates/COMMIT_CONVENTIONS.md +0 -86
  411. package/templates/OIDC_SETUP.md +0 -300
  412. package/templates/agents.md +0 -145
  413. package/templates/codemod/README.md +0 -1
  414. package/templates/codemod/test.js +0 -93
  415. package/templates/docs/README.md +0 -151
  416. package/templates/scripts/new_feature.sh +0 -80
  417. package/templates/scripts/quality-gates/check-god-objects.js +0 -146
  418. package/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  419. package/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
@@ -1,711 +0,0 @@
1
- /**
2
- * @fileoverview Spec Resolution System
3
- * Resolves spec files with priority: feature-specific > working-spec.yaml
4
- * Enables multi-agent workflows where each agent works on their own spec
5
- * @author @darianrosebrook
6
- */
7
-
8
- const fs = require('fs-extra');
9
- const path = require('path');
10
- const yaml = require('js-yaml');
11
- const chalk = require('chalk');
12
-
13
- // Import SPEC_TYPES from constants for consistent display
14
- const { SPEC_TYPES } = require('../constants/spec-types');
15
- const { findProjectRoot } = require('./detection');
16
- const { createValidator, getSchemaPath } = require('./schema-validator');
17
-
18
- /**
19
- * Validate a spec object against the working-spec schema.
20
- * Throws with schema errors included in the message.
21
- * @param {Object} spec - Parsed spec object
22
- * @param {string} specPath - Path to the spec file (for error context)
23
- */
24
- function validateSpecSchema(spec, specPath) {
25
- try {
26
- const schemaPath = getSchemaPath('working-spec.schema.json', getProjectRoot());
27
- const validate = createValidator(schemaPath);
28
- const result = validate(spec);
29
- if (!result.valid) {
30
- const errorDetails = result.errors
31
- .map(e => ` ${e.path}: ${e.message}`)
32
- .join('\n');
33
- // Schema violations are warnings, not fatal — the spec is still loadable.
34
- // Commands like validate will report these; other commands shouldn't be blocked.
35
- if (process.env.CAWS_QUIET !== '1') {
36
- console.warn(`Schema warnings for ${specPath}:\n${errorDetails}`);
37
- }
38
- }
39
- } catch (schemaErr) {
40
- // Schema loading/compilation errors are non-fatal — warn and continue
41
- console.warn('Could not validate spec schema:', schemaErr.message);
42
- }
43
- }
44
-
45
- /**
46
- * Spec resolution priority:
47
- * 1. .caws/specs/<spec-id>.yaml (feature-specific, multi-agent safe)
48
- * 2. .caws/working-spec.yaml (legacy, single-agent only)
49
- */
50
- const SPECS_DIR = '.caws/specs';
51
- const LEGACY_SPEC = '.caws/working-spec.yaml';
52
- const SPECS_REGISTRY = '.caws/specs/registry.json';
53
-
54
- /**
55
- * Get the project root for spec resolution.
56
- * Caches per process to avoid repeated filesystem walks.
57
- */
58
- let _cachedProjectRoot = null;
59
- function getProjectRoot() {
60
- if (!_cachedProjectRoot) {
61
- _cachedProjectRoot = findProjectRoot();
62
- }
63
- return _cachedProjectRoot;
64
- }
65
-
66
- /**
67
- * Resolve spec file path based on priority
68
- * @param {Object} options - Resolution options
69
- * @param {string} [options.specId] - Feature-specific spec ID (e.g., 'user-auth', 'FEAT-001')
70
- * @param {string} [options.specFile] - Explicit file path override
71
- * @param {boolean} [options.warnLegacy=true] - Warn when falling back to legacy spec
72
- * @param {boolean} [options.interactive=false] - Use interactive spec selection for multiple specs
73
- * @param {boolean} [options.quiet=false] - Suppress informational logging for machine-readable output
74
- * @returns {Promise<{path: string, type: 'feature' | 'legacy', spec: Object}>}
75
- */
76
- async function resolveSpec(options = {}) {
77
- const { specId, specFile, warnLegacy = true, interactive = false, quiet = false } = options;
78
-
79
- // 1. Explicit file path takes highest priority
80
- if (specFile) {
81
- const explicitPath = path.isAbsolute(specFile) ? specFile : path.join(getProjectRoot(), specFile);
82
-
83
- if (await fs.pathExists(explicitPath)) {
84
- const yaml = require('js-yaml');
85
- const content = await fs.readFile(explicitPath, 'utf8');
86
- let spec;
87
- try {
88
- spec = yaml.load(content);
89
- } catch (yamlError) {
90
- throw new Error(`Invalid YAML in spec file ${explicitPath}: ${yamlError.message}`);
91
- }
92
- validateSpecSchema(spec, explicitPath);
93
-
94
- return {
95
- path: explicitPath,
96
- type: explicitPath.includes('/specs/') ? 'feature' : 'legacy',
97
- spec,
98
- };
99
- }
100
-
101
- throw new Error(`Spec file not found: ${explicitPath}`);
102
- }
103
-
104
- // 2. Feature-specific spec (preferred for multi-agent)
105
- if (specId) {
106
- const featurePath = path.join(getProjectRoot(), SPECS_DIR, `${specId}.yaml`);
107
-
108
- if (await fs.pathExists(featurePath)) {
109
- const yaml = require('js-yaml');
110
- const content = await fs.readFile(featurePath, 'utf8');
111
- const spec = yaml.load(content);
112
- validateSpecSchema(spec, featurePath);
113
-
114
- if (!quiet) {
115
- console.log(chalk.green(`Using feature-specific spec: ${specId}`));
116
- }
117
-
118
- return {
119
- path: featurePath,
120
- type: 'feature',
121
- spec,
122
- };
123
- }
124
-
125
- throw new Error(
126
- `Feature spec '${specId}' not found. Create it with: caws specs create ${specId}`
127
- );
128
- }
129
-
130
- // 3. Auto-detect from registry or list specs
131
- const registry = await loadSpecsRegistry();
132
- const specIds = Object.keys(registry.specs ?? {});
133
-
134
- if (specIds.length === 1) {
135
- // Single spec - use it automatically
136
- const singleSpecId = specIds[0];
137
- const singleSpecPath = path.join(getProjectRoot(), SPECS_DIR, registry.specs[singleSpecId].path);
138
-
139
- if (await fs.pathExists(singleSpecPath)) {
140
- const yaml = require('js-yaml');
141
- const content = await fs.readFile(singleSpecPath, 'utf8');
142
- const spec = yaml.load(content);
143
- validateSpecSchema(spec, singleSpecPath);
144
-
145
- if (!quiet) {
146
- console.log(chalk.blue(`Auto-detected single spec: ${singleSpecId}`));
147
- }
148
-
149
- return {
150
- path: singleSpecPath,
151
- type: 'feature',
152
- spec,
153
- };
154
- }
155
- } else if (specIds.length > 1) {
156
- // Multiple specs - require explicit selection with enhanced guidance
157
- if (!quiet) {
158
- console.error(chalk.red('Multiple specs detected. Please specify which one:'));
159
- }
160
-
161
- // Show specs with details
162
- const specsInfo = [];
163
- for (const id of specIds) {
164
- const specPath = path.join(getProjectRoot(), SPECS_DIR, registry.specs[id].path);
165
- try {
166
- const content = await fs.readFile(specPath, 'utf8');
167
- let spec;
168
- try {
169
- spec = yaml.load(content);
170
- } catch (yamlError) {
171
- if (!quiet) {
172
- console.log(chalk.yellow(` - ${id} (YAML syntax error: ${yamlError.message})`));
173
- }
174
- specsInfo.push({ id, type: 'unknown', status: 'unknown', title: 'YAML error' });
175
- continue;
176
- }
177
- const status = spec.status || 'draft';
178
- const type = spec.type || 'feature';
179
- const statusColor =
180
- status === 'active' ? chalk.green : status === 'completed' ? chalk.blue : chalk.yellow;
181
- const typeColor = SPEC_TYPES[type] ? SPEC_TYPES[type].color : chalk.white;
182
-
183
- if (!quiet) {
184
- console.log(
185
- chalk.yellow(
186
- ` - ${id} ${typeColor(`(${type})`)} ${statusColor(`[${status}]`)} - ${spec.title || 'Untitled'}`
187
- )
188
- );
189
- }
190
- specsInfo.push({ id, type, status, title: spec.title || 'Untitled' });
191
- } catch (error) {
192
- if (!quiet) {
193
- console.log(chalk.yellow(` - ${id} (error loading details: ${error.message})`));
194
- }
195
- specsInfo.push({ id, type: 'unknown', status: 'unknown', title: 'Error loading' });
196
- }
197
- }
198
-
199
- // Interactive mode
200
- if (interactive) {
201
- try {
202
- const selectedSpecId = await interactiveSpecSelection(specIds);
203
-
204
- // Recursively resolve with the selected spec ID
205
- return await resolveSpec({
206
- specId: selectedSpecId,
207
- warnLegacy,
208
- interactive: false, // Prevent infinite recursion
209
- quiet,
210
- });
211
- } catch (error) {
212
- throw new Error(`Interactive selection failed: ${error.message}`);
213
- }
214
- }
215
-
216
- if (!quiet) {
217
- console.log(chalk.blue('\n Usage: caws <command> --spec-id <spec-id>'));
218
- console.log(chalk.gray(` Example: caws validate --spec-id ${specIds[0]}`));
219
- }
220
-
221
- // Suggest most likely spec (active first, then by type priority)
222
- const priorityOrder = { active: 0, draft: 1, completed: 2 };
223
- const sortedSpecs = specIds.sort((a, b) => {
224
- const aSpec = specsInfo.find((s) => s.id === a);
225
- const bSpec = specsInfo.find((s) => s.id === b);
226
- const aPriority = priorityOrder[aSpec?.status] || 999;
227
- const bPriority = priorityOrder[bSpec?.status] || 999;
228
- if (aPriority !== bPriority) return aPriority - bPriority;
229
-
230
- // Then by type (feature > fix > refactor > etc.)
231
- const typePriority = { feature: 0, fix: 1, refactor: 2, chore: 3, docs: 4 };
232
- const aTypePriority = typePriority[aSpec?.type] || 999;
233
- const bTypePriority = typePriority[bSpec?.type] || 999;
234
- return aTypePriority - bTypePriority;
235
- });
236
-
237
- if (!quiet) {
238
- console.log(chalk.green('\nQuick suggestion:'));
239
- console.log(chalk.gray(` Try: caws <command> --spec-id ${sortedSpecs[0]}`));
240
- }
241
-
242
- // Interactive mode suggestion
243
- if (!quiet) {
244
- console.log(chalk.blue('\n Interactive mode: caws <command> --interactive-spec-selection'));
245
- }
246
-
247
- throw new Error('Spec ID required when multiple specs exist');
248
- }
249
-
250
- // 4. Fall back to legacy working-spec.yaml (with warning)
251
- const legacyPath = path.join(getProjectRoot(), LEGACY_SPEC);
252
-
253
- if (await fs.pathExists(legacyPath)) {
254
- const yaml = require('js-yaml');
255
- const content = await fs.readFile(legacyPath, 'utf8');
256
- const spec = yaml.load(content);
257
- validateSpecSchema(spec, legacyPath);
258
-
259
- if (warnLegacy && !quiet) {
260
- console.log(chalk.yellow('Using legacy working-spec.yaml'));
261
- console.log(chalk.gray(' For multi-agent workflows, use feature-specific specs:'));
262
- console.log(chalk.blue(' caws specs create <feature-id>'));
263
- console.log('');
264
- }
265
-
266
- return {
267
- path: legacyPath,
268
- type: 'legacy',
269
- spec,
270
- };
271
- }
272
-
273
- // 5. No specs found
274
- throw new Error(
275
- 'No CAWS spec found. Initialize with: caws init or create a feature spec: caws specs create <id>'
276
- );
277
- }
278
-
279
- /**
280
- * Load specs registry
281
- * @returns {Promise<Object>} Registry data
282
- */
283
- async function loadSpecsRegistry() {
284
- const registryPath = path.join(getProjectRoot(), SPECS_REGISTRY);
285
-
286
- if (!(await fs.pathExists(registryPath))) {
287
- return {
288
- version: '1.0.0',
289
- specs: {},
290
- lastUpdated: new Date().toISOString(),
291
- };
292
- }
293
-
294
- try {
295
- const registry = await fs.readJson(registryPath);
296
- const sanitizedSpecs = {};
297
-
298
- for (const [id, entry] of Object.entries(registry.specs || {})) {
299
- const specPath = path.join(getProjectRoot(), SPECS_DIR, entry.path);
300
- if (await fs.pathExists(specPath)) {
301
- sanitizedSpecs[id] = entry;
302
- }
303
- }
304
-
305
- return {
306
- ...registry,
307
- specs: sanitizedSpecs,
308
- };
309
- } catch (error) {
310
- return {
311
- version: '1.0.0',
312
- specs: {},
313
- lastUpdated: new Date().toISOString(),
314
- };
315
- }
316
- }
317
-
318
- /**
319
- * List all available specs
320
- * @returns {Promise<Array<{id: string, path: string, type: string}>>}
321
- */
322
- async function listAvailableSpecs() {
323
- const specs = [];
324
-
325
- // Check feature-specific specs
326
- const specsDir = path.join(getProjectRoot(), SPECS_DIR);
327
- if (await fs.pathExists(specsDir)) {
328
- const files = await fs.readdir(specsDir);
329
- const yamlFiles = files.filter((f) => f.endsWith('.yaml') || f.endsWith('.yml'));
330
-
331
- for (const file of yamlFiles) {
332
- if (file === 'registry.json') continue;
333
-
334
- const specPath = path.join(specsDir, file);
335
- try {
336
- const yaml = require('js-yaml');
337
- const content = await fs.readFile(specPath, 'utf8');
338
- const spec = yaml.load(content);
339
-
340
- specs.push({
341
- id: spec.id || path.basename(file, path.extname(file)),
342
- path: path.relative(getProjectRoot(), specPath),
343
- type: 'feature',
344
- title: spec.title || 'Untitled',
345
- });
346
- } catch (error) {
347
- // Skip invalid specs
348
- }
349
- }
350
- }
351
-
352
- // Check legacy working-spec.yaml
353
- const legacyPath = path.join(getProjectRoot(), LEGACY_SPEC);
354
- if (await fs.pathExists(legacyPath)) {
355
- try {
356
- const yaml = require('js-yaml');
357
- const content = await fs.readFile(legacyPath, 'utf8');
358
- const spec = yaml.load(content);
359
-
360
- specs.push({
361
- id: spec.id || 'working-spec',
362
- path: LEGACY_SPEC,
363
- type: 'legacy',
364
- title: spec.title || 'Legacy Working Spec',
365
- });
366
- } catch (error) {
367
- // Skip invalid spec
368
- }
369
- }
370
-
371
- return specs;
372
- }
373
-
374
- /**
375
- * Interactive spec selection using readline
376
- * @param {string[]} specIds - Available spec IDs
377
- * @returns {Promise<string>} Selected spec ID
378
- */
379
- async function interactiveSpecSelection(specIds) {
380
- return new Promise((resolve, reject) => {
381
- const readline = require('readline');
382
-
383
- console.log(chalk.blue('\nInteractive Spec Selection'));
384
- console.log(chalk.gray('Select which spec to use:\n'));
385
-
386
- specIds.forEach((id, index) => {
387
- console.log(chalk.yellow(`${index + 1}. ${id}`));
388
- });
389
-
390
- console.log(chalk.gray('\nEnter number (1-' + specIds.length + ') or spec ID directly: '));
391
-
392
- const rl = readline.createInterface({
393
- input: process.stdin,
394
- output: process.stdout,
395
- });
396
-
397
- rl.question('> ', (answer) => {
398
- rl.close();
399
-
400
- const trimmed = answer.trim();
401
-
402
- // Check if it's a number
403
- const num = parseInt(trimmed);
404
- if (num >= 1 && num <= specIds.length) {
405
- resolve(specIds[num - 1]);
406
- return;
407
- }
408
-
409
- // Check if it's a direct spec ID
410
- if (specIds.includes(trimmed)) {
411
- resolve(trimmed);
412
- return;
413
- }
414
-
415
- reject(new Error(`Invalid selection: ${trimmed}. Please choose a valid spec ID.`));
416
- });
417
- });
418
- }
419
-
420
- /**
421
- * Check if project is using multi-spec architecture
422
- * @returns {Promise<{isMultiSpec: boolean, specCount: number, needsMigration: boolean}>}
423
- */
424
- async function checkMultiSpecStatus() {
425
- const registry = await loadSpecsRegistry();
426
- const hasFeatureSpecs = Object.keys(registry.specs ?? {}).length > 0;
427
- const legacyPath = path.join(getProjectRoot(), LEGACY_SPEC);
428
- const hasLegacySpec = await fs.pathExists(legacyPath);
429
-
430
- return {
431
- isMultiSpec: hasFeatureSpecs,
432
- specCount: Object.keys(registry.specs ?? {}).length,
433
- needsMigration: hasLegacySpec && !hasFeatureSpecs,
434
- };
435
- }
436
-
437
- /**
438
- * Check for scope conflicts between specs
439
- * @param {string[]} specIds - Array of spec IDs to check
440
- * @returns {Promise<Array<{spec1: string, spec2: string, conflicts: string[]}>>} Array of conflicts
441
- */
442
- async function checkScopeConflicts(specIds) {
443
- const conflicts = [];
444
- const specScopes = [];
445
-
446
- // Load registry once
447
- const registry = await loadSpecsRegistry();
448
-
449
- // Load all specs and their scopes
450
- for (const id of specIds) {
451
- const entry = registry.specs[id];
452
- if (!entry) continue;
453
-
454
- const specPath = path.join(getProjectRoot(), SPECS_DIR, entry.path);
455
-
456
- try {
457
- const content = await fs.readFile(specPath, 'utf8');
458
- let spec;
459
- try {
460
- spec = yaml.load(content);
461
- } catch (yamlError) {
462
- const relativePath = path.relative(getProjectRoot(), specPath);
463
- throw new Error(
464
- `Invalid YAML syntax in ${relativePath}: ${yamlError.message}\n` +
465
- (yamlError.mark
466
- ? ` Line ${yamlError.mark.line + 1}, Column ${yamlError.mark.column + 1}\n`
467
- : '') +
468
- (yamlError.mark?.snippet ? ` ${yamlError.mark.snippet}\n` : '') +
469
- `Fix YAML syntax errors or use 'caws specs create <id>' for proper structure`
470
- );
471
- }
472
-
473
- specScopes.push({
474
- id,
475
- scope: spec.scope || { in: [], out: [] },
476
- title: spec.title || id,
477
- });
478
- } catch (error) {
479
- // Skip specs that can't be loaded
480
- continue;
481
- }
482
- }
483
-
484
- // Check for conflicts between each pair of specs
485
- for (let i = 0; i < specScopes.length; i++) {
486
- for (let j = i + 1; j < specScopes.length; j++) {
487
- const spec1 = specScopes[i];
488
- const spec2 = specScopes[j];
489
-
490
- const spec1Paths = new Set(spec1.scope.in || []);
491
- const spec2Paths = new Set(spec2.scope.in || []);
492
-
493
- // Find overlapping paths
494
- const overlappingPaths = [];
495
- for (const path1 of spec1Paths) {
496
- for (const path2 of spec2Paths) {
497
- if (pathsOverlap(path1, path2)) {
498
- overlappingPaths.push(`${path1} ↔ ${path2}`);
499
- }
500
- }
501
- }
502
-
503
- if (overlappingPaths.length > 0) {
504
- conflicts.push({
505
- spec1: spec1.id,
506
- spec2: spec2.id,
507
- conflicts: overlappingPaths,
508
- severity: 'warning', // Could be 'error' for stricter enforcement
509
- });
510
- }
511
- }
512
- }
513
-
514
- return conflicts;
515
- }
516
-
517
- /**
518
- * Check if two paths overlap (simplified implementation)
519
- * @param {string} path1 - First path
520
- * @param {string} path2 - Second path
521
- * @returns {boolean} True if paths overlap
522
- */
523
- function pathsOverlap(path1, path2) {
524
- // Normalize paths (remove leading/trailing slashes)
525
- const normalizePath = (p) => p.replace(/^\/+|\/+$/g, '');
526
-
527
- const normalized1 = normalizePath(path1);
528
- const normalized2 = normalizePath(path2);
529
-
530
- // Check for exact match
531
- if (normalized1 === normalized2) {
532
- return true;
533
- }
534
-
535
- // Handle wildcard patterns
536
- const hasWildcard = (p) => p.includes('*');
537
-
538
- if (hasWildcard(normalized1) || hasWildcard(normalized2)) {
539
- // Convert wildcards to regex patterns
540
- const toRegex = (p) => {
541
- // Escape dots first
542
- let result = p.replace(/\./g, '\\.');
543
-
544
- // Handle ** patterns (match any path including zero segments)
545
- result = result.replace(/\*\*/g, '(?:.*/)?');
546
-
547
- // Handle single * patterns (match any non-slash characters)
548
- result = result.replace(/\*/g, '[^/]*');
549
-
550
- // Fix patterns like src/auth/**/*.js to match src/auth/login.js
551
- // The pattern (?:.*/)?[^/]* should become .*[^/]* for direct filename matching
552
- result = result.replace(/(\?:.*\/)?[^/]*/g, '.*[^/]*');
553
-
554
- // Also fix patterns like (?:.[^/]*/)?/[^/]* to match direct filenames
555
- result = result.replace(/(?:\..*\/)?[^/]*/g, '.*[^/]*');
556
-
557
- return result;
558
- };
559
-
560
- // Check if either path matches the other's pattern
561
- if (hasWildcard(normalized1)) {
562
- const regex1 = new RegExp('^' + toRegex(normalized1) + '$');
563
- if (regex1.test(normalized2)) return true;
564
- }
565
-
566
- if (hasWildcard(normalized2)) {
567
- const regex2 = new RegExp('^' + toRegex(normalized2) + '$');
568
- if (regex2.test(normalized1)) return true;
569
- }
570
-
571
- return false;
572
- }
573
-
574
- // Simple substring check for non-wildcard paths
575
- return normalized1.includes(normalized2) || normalized2.includes(normalized1);
576
- }
577
-
578
- /**
579
- * Suggest migration from legacy to multi-spec
580
- * @returns {Promise<void>}
581
- */
582
- async function suggestMigration() {
583
- const status = await checkMultiSpecStatus();
584
-
585
- if (status.needsMigration) {
586
- console.log(chalk.yellow('\nMigration Recommended: Single-Spec → Multi-Spec'));
587
- console.log(chalk.gray(' Your project uses the legacy working-spec.yaml'));
588
- console.log(chalk.gray(' For multi-agent workflows, migrate to feature-specific specs:\n'));
589
- console.log(chalk.blue(' 1. caws specs create <feature-id>'));
590
- console.log(chalk.blue(' 2. Copy relevant content from working-spec.yaml'));
591
- console.log(chalk.blue(' 3. Update agents to use --spec-id <feature-id>'));
592
- console.log(chalk.gray('\n See: docs/guides/multi-agent-migration.md\n'));
593
- }
594
- }
595
-
596
- // Feature breakdown logic (moved from specs.js to avoid circular dependency)
597
- function suggestFeatureBreakdown(legacySpec) {
598
- const features = [];
599
-
600
- if (!legacySpec) {
601
- return features;
602
- }
603
-
604
- if (legacySpec.acceptance && legacySpec.acceptance.length > 0) {
605
- // Group acceptance criteria by logical features
606
- const criteriaByFeature = {};
607
-
608
- legacySpec.acceptance.forEach((criterion, index) => {
609
- // Simple heuristic: extract feature from criterion description (check all fields)
610
- const fullDescription = [
611
- criterion.given || '',
612
- criterion.when || '',
613
- criterion.then || '',
614
- criterion.description || '',
615
- criterion.title || `A${index + 1}`,
616
- ].join(' ');
617
- const words = fullDescription.toLowerCase().split(' ');
618
-
619
- // Look for common feature keywords
620
- const featureKeywords = {
621
- auth: 'Authentication',
622
- login: 'Authentication',
623
- payment: 'Payment System',
624
- billing: 'Billing',
625
- dashboard: 'Dashboard',
626
- admin: 'Admin Panel',
627
- api: 'API',
628
- database: 'Data Layer',
629
- ui: 'User Interface',
630
- email: 'Email System',
631
- notification: 'Notifications',
632
- report: 'Reporting',
633
- search: 'Search',
634
- filter: 'Filtering',
635
- user: 'User Management',
636
- };
637
-
638
- let featureKey = 'general';
639
- let featureTitle = 'General Features';
640
-
641
- for (const [keyword, title] of Object.entries(featureKeywords)) {
642
- if (words.some((word) => word.includes(keyword))) {
643
- featureKey = keyword;
644
- featureTitle = title;
645
- break;
646
- }
647
- }
648
-
649
- if (!criteriaByFeature[featureKey]) {
650
- criteriaByFeature[featureKey] = {
651
- id: featureKey,
652
- title: featureTitle,
653
- criteria: [],
654
- scope: { in: [], out: [] },
655
- };
656
- }
657
-
658
- criteriaByFeature[featureKey].criteria.push(criterion);
659
- });
660
-
661
- // Convert to feature objects
662
- Object.values(criteriaByFeature).forEach((feature) => {
663
- // Suggest scope based on feature type
664
- const scopeSuggestions = {
665
- user: { in: ['src/users/', 'tests/users/'], out: ['src/payments/', 'src/admin/'] },
666
- auth: { in: ['src/auth/', 'tests/auth/'], out: ['src/payments/', 'src/admin/'] },
667
- payment: { in: ['src/payments/', 'tests/payments/'], out: ['src/users/', 'src/admin/'] },
668
- dashboard: {
669
- in: ['src/dashboard/', 'tests/dashboard/'],
670
- out: ['src/payments/', 'src/users/'],
671
- },
672
- admin: { in: ['src/admin/', 'tests/admin/'], out: ['src/payments/', 'src/users/'] },
673
- api: { in: ['src/api/', 'tests/api/'], out: ['src/dashboard/', 'src/admin/'] },
674
- general: { in: ['src/', 'tests/'], out: [] },
675
- };
676
-
677
- const suggestion = scopeSuggestions[feature.id] || scopeSuggestions.general;
678
- feature.scope = suggestion;
679
-
680
- features.push(feature);
681
- });
682
- } else {
683
- // Fallback: create a single feature
684
- features.push({
685
- id: 'main-feature',
686
- title: legacySpec.title || 'Main Feature',
687
- criteria: legacySpec.acceptance || [],
688
- scope: {
689
- in: ['src/', 'tests/'],
690
- out: [],
691
- },
692
- });
693
- }
694
-
695
- return features;
696
- }
697
-
698
- module.exports = {
699
- resolveSpec,
700
- listAvailableSpecs,
701
- checkMultiSpecStatus,
702
- checkScopeConflicts,
703
- suggestMigration,
704
- interactiveSpecSelection,
705
- loadSpecsRegistry,
706
- suggestFeatureBreakdown,
707
- pathsOverlap,
708
- SPECS_DIR,
709
- LEGACY_SPEC,
710
- SPECS_REGISTRY,
711
- };