@paths.design/caws-cli 10.2.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 (421) hide show
  1. package/README.md +125 -374
  2. package/dist/index.js +43 -785
  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/agents.js +0 -124
  183. package/dist/commands/archive.js +0 -500
  184. package/dist/commands/burnup.js +0 -198
  185. package/dist/commands/diagnose.js +0 -525
  186. package/dist/commands/evaluate.js +0 -314
  187. package/dist/commands/gates.js +0 -149
  188. package/dist/commands/init.js +0 -857
  189. package/dist/commands/iterate.js +0 -417
  190. package/dist/commands/mode.js +0 -269
  191. package/dist/commands/parallel.js +0 -242
  192. package/dist/commands/plan.js +0 -438
  193. package/dist/commands/provenance.js +0 -1143
  194. package/dist/commands/quality-monitor.js +0 -284
  195. package/dist/commands/scope.js +0 -264
  196. package/dist/commands/session.js +0 -312
  197. package/dist/commands/sidecar.js +0 -74
  198. package/dist/commands/specs.js +0 -1656
  199. package/dist/commands/status.js +0 -1172
  200. package/dist/commands/templates.js +0 -237
  201. package/dist/commands/tool.js +0 -136
  202. package/dist/commands/tutorial.js +0 -480
  203. package/dist/commands/validate.js +0 -357
  204. package/dist/commands/verify-acs.js +0 -443
  205. package/dist/commands/waivers.js +0 -599
  206. package/dist/commands/workflow.js +0 -243
  207. package/dist/commands/worktree.js +0 -502
  208. package/dist/config/lite-scope.js +0 -158
  209. package/dist/config/modes.js +0 -347
  210. package/dist/constants/spec-types.js +0 -65
  211. package/dist/gates/budget-limit.js +0 -121
  212. package/dist/gates/feedback.js +0 -260
  213. package/dist/gates/format.js +0 -179
  214. package/dist/gates/god-object.js +0 -117
  215. package/dist/gates/pipeline.js +0 -167
  216. package/dist/gates/scope-boundary.js +0 -112
  217. package/dist/gates/spec-completeness.js +0 -109
  218. package/dist/gates/todo-detection.js +0 -205
  219. package/dist/generators/jest-config-generator.js +0 -242
  220. package/dist/generators/working-spec.js +0 -237
  221. package/dist/minimal-cli.js +0 -88
  222. package/dist/parallel/parallel-manager.js +0 -433
  223. package/dist/policy/PolicyManager.js +0 -470
  224. package/dist/scaffold/claude-hooks.js +0 -443
  225. package/dist/scaffold/cursor-hooks.js +0 -177
  226. package/dist/scaffold/git-hooks.js +0 -928
  227. package/dist/scaffold/index.js +0 -794
  228. package/dist/session/session-manager.js +0 -653
  229. package/dist/sidecars/index.js +0 -33
  230. package/dist/sidecars/listeners.js +0 -40
  231. package/dist/sidecars/provenance-summary.js +0 -238
  232. package/dist/sidecars/quality-gaps.js +0 -258
  233. package/dist/sidecars/schema.js +0 -149
  234. package/dist/sidecars/spec-drift.js +0 -151
  235. package/dist/sidecars/waiver-draft.js +0 -176
  236. package/dist/spec/SpecFileManager.js +0 -419
  237. package/dist/templates/.caws/schemas/policy.schema.json +0 -117
  238. package/dist/templates/.caws/schemas/scope.schema.json +0 -52
  239. package/dist/templates/.caws/schemas/waivers.schema.json +0 -106
  240. package/dist/templates/.caws/schemas/working-spec.schema.json +0 -340
  241. package/dist/templates/.caws/schemas/worktrees.schema.json +0 -38
  242. package/dist/templates/.caws/templates/working-spec.template.yml +0 -80
  243. package/dist/templates/.caws/tools/README.md +0 -18
  244. package/dist/templates/.caws/tools/scope-guard.js +0 -203
  245. package/dist/templates/.caws/tools-allow.json +0 -331
  246. package/dist/templates/.caws/waivers.yml +0 -19
  247. package/dist/templates/.claude/README.md +0 -190
  248. package/dist/templates/.claude/hooks/audit.sh +0 -121
  249. package/dist/templates/.claude/hooks/block-dangerous.sh +0 -203
  250. package/dist/templates/.claude/hooks/classify_command.py +0 -592
  251. package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  252. package/dist/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  253. package/dist/templates/.claude/hooks/naming-check.sh +0 -100
  254. package/dist/templates/.claude/hooks/protected-paths.sh +0 -39
  255. package/dist/templates/.claude/hooks/quality-check.sh +0 -81
  256. package/dist/templates/.claude/hooks/scan-secrets.sh +0 -85
  257. package/dist/templates/.claude/hooks/scope-guard.sh +0 -381
  258. package/dist/templates/.claude/hooks/session-caws-status.sh +0 -117
  259. package/dist/templates/.claude/hooks/session-log.sh +0 -634
  260. package/dist/templates/.claude/hooks/simplification-guard.sh +0 -92
  261. package/dist/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  262. package/dist/templates/.claude/hooks/test_classify_command.py +0 -370
  263. package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  264. package/dist/templates/.claude/hooks/validate-spec.sh +0 -76
  265. package/dist/templates/.claude/hooks/worktree-guard.sh +0 -220
  266. package/dist/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  267. package/dist/templates/.claude/rules/git-safety.md +0 -26
  268. package/dist/templates/.claude/rules/worktree-isolation.md +0 -101
  269. package/dist/templates/.claude/settings.json +0 -141
  270. package/dist/templates/.cursor/README.md +0 -299
  271. package/dist/templates/.cursor/hooks/audit.sh +0 -55
  272. package/dist/templates/.cursor/hooks/block-dangerous.sh +0 -84
  273. package/dist/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  274. package/dist/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  275. package/dist/templates/.cursor/hooks/format.sh +0 -38
  276. package/dist/templates/.cursor/hooks/naming-check.sh +0 -64
  277. package/dist/templates/.cursor/hooks/scan-secrets.sh +0 -51
  278. package/dist/templates/.cursor/hooks/scope-guard.sh +0 -52
  279. package/dist/templates/.cursor/hooks/session-log.sh +0 -924
  280. package/dist/templates/.cursor/hooks/validate-spec.sh +0 -83
  281. package/dist/templates/.cursor/hooks.json +0 -76
  282. package/dist/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  283. package/dist/templates/.cursor/rules/01-working-style.mdc +0 -50
  284. package/dist/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  285. package/dist/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  286. package/dist/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  287. package/dist/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  288. package/dist/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  289. package/dist/templates/.cursor/rules/07-process-ops.mdc +0 -20
  290. package/dist/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  291. package/dist/templates/.cursor/rules/09-docstrings.mdc +0 -89
  292. package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  293. package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  294. package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  295. package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  296. package/dist/templates/.cursor/rules/README.md +0 -148
  297. package/dist/templates/.github/copilot-instructions.md +0 -82
  298. package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  299. package/dist/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  300. package/dist/templates/.junie/guidelines.md +0 -73
  301. package/dist/templates/.vscode/launch.json +0 -17
  302. package/dist/templates/.vscode/settings.json +0 -95
  303. package/dist/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  304. package/dist/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  305. package/dist/templates/CLAUDE.md +0 -196
  306. package/dist/templates/COMMIT_CONVENTIONS.md +0 -86
  307. package/dist/templates/OIDC_SETUP.md +0 -300
  308. package/dist/templates/agents.md +0 -171
  309. package/dist/templates/codemod/README.md +0 -1
  310. package/dist/templates/codemod/test.js +0 -93
  311. package/dist/templates/docs/README.md +0 -151
  312. package/dist/templates/scripts/new_feature.sh +0 -80
  313. package/dist/templates/scripts/quality-gates/check-god-objects.js +0 -146
  314. package/dist/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  315. package/dist/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
  316. package/dist/test-analysis.js +0 -786
  317. package/dist/tool-interface.js +0 -314
  318. package/dist/tool-loader.js +0 -303
  319. package/dist/tool-validator.js +0 -393
  320. package/dist/utils/agent-display.js +0 -210
  321. package/dist/utils/agent-session.js +0 -344
  322. package/dist/utils/async-utils.js +0 -188
  323. package/dist/utils/command-wrapper.js +0 -200
  324. package/dist/utils/event-log.js +0 -584
  325. package/dist/utils/event-renderer.js +0 -521
  326. package/dist/utils/finalization.js +0 -230
  327. package/dist/utils/git-lock.js +0 -119
  328. package/dist/utils/gitignore-updater.js +0 -158
  329. package/dist/utils/ide-detection.js +0 -133
  330. package/dist/utils/lifecycle-events.js +0 -94
  331. package/dist/utils/project-analysis.js +0 -367
  332. package/dist/utils/promise-utils.js +0 -72
  333. package/dist/utils/quality-gates-errors.js +0 -520
  334. package/dist/utils/quality-gates-utils.js +0 -387
  335. package/dist/utils/schema-validator.js +0 -50
  336. package/dist/utils/spec-resolver.js +0 -711
  337. package/dist/utils/typescript-detector.js +0 -369
  338. package/dist/utils/working-state.js +0 -530
  339. package/dist/utils/yaml-validation.js +0 -156
  340. package/dist/validation/spec-validation.js +0 -924
  341. package/dist/waivers-manager.js +0 -732
  342. package/dist/worktree/worktree-manager.js +0 -1735
  343. package/templates/.caws/schemas/policy.schema.json +0 -117
  344. package/templates/.caws/schemas/scope.schema.json +0 -52
  345. package/templates/.caws/schemas/waivers.schema.json +0 -106
  346. package/templates/.caws/schemas/working-spec.schema.json +0 -340
  347. package/templates/.caws/schemas/worktrees.schema.json +0 -38
  348. package/templates/.caws/templates/working-spec.template.yml +0 -80
  349. package/templates/.caws/tools/README.md +0 -18
  350. package/templates/.caws/tools/scope-guard.js +0 -203
  351. package/templates/.caws/tools-allow.json +0 -331
  352. package/templates/.caws/waivers.yml +0 -19
  353. package/templates/.claude/README.md +0 -190
  354. package/templates/.claude/hooks/audit.sh +0 -121
  355. package/templates/.claude/hooks/block-dangerous.sh +0 -203
  356. package/templates/.claude/hooks/classify_command.py +0 -592
  357. package/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  358. package/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  359. package/templates/.claude/hooks/naming-check.sh +0 -100
  360. package/templates/.claude/hooks/protected-paths.sh +0 -39
  361. package/templates/.claude/hooks/quality-check.sh +0 -81
  362. package/templates/.claude/hooks/scan-secrets.sh +0 -85
  363. package/templates/.claude/hooks/scope-guard.sh +0 -381
  364. package/templates/.claude/hooks/session-caws-status.sh +0 -117
  365. package/templates/.claude/hooks/session-log.sh +0 -634
  366. package/templates/.claude/hooks/simplification-guard.sh +0 -92
  367. package/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  368. package/templates/.claude/hooks/test_classify_command.py +0 -370
  369. package/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  370. package/templates/.claude/hooks/validate-spec.sh +0 -76
  371. package/templates/.claude/hooks/worktree-guard.sh +0 -220
  372. package/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  373. package/templates/.claude/rules/git-safety.md +0 -26
  374. package/templates/.claude/rules/worktree-isolation.md +0 -101
  375. package/templates/.claude/settings.json +0 -141
  376. package/templates/.cursor/README.md +0 -299
  377. package/templates/.cursor/hooks/audit.sh +0 -55
  378. package/templates/.cursor/hooks/block-dangerous.sh +0 -84
  379. package/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  380. package/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  381. package/templates/.cursor/hooks/format.sh +0 -38
  382. package/templates/.cursor/hooks/naming-check.sh +0 -64
  383. package/templates/.cursor/hooks/scan-secrets.sh +0 -51
  384. package/templates/.cursor/hooks/scope-guard.sh +0 -52
  385. package/templates/.cursor/hooks/session-log.sh +0 -924
  386. package/templates/.cursor/hooks/validate-spec.sh +0 -83
  387. package/templates/.cursor/hooks.json +0 -76
  388. package/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  389. package/templates/.cursor/rules/01-working-style.mdc +0 -50
  390. package/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  391. package/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  392. package/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  393. package/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  394. package/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  395. package/templates/.cursor/rules/07-process-ops.mdc +0 -20
  396. package/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  397. package/templates/.cursor/rules/09-docstrings.mdc +0 -89
  398. package/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  399. package/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  400. package/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  401. package/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  402. package/templates/.cursor/rules/README.md +0 -148
  403. package/templates/.github/copilot-instructions.md +0 -82
  404. package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  405. package/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  406. package/templates/.junie/guidelines.md +0 -73
  407. package/templates/.vscode/launch.json +0 -17
  408. package/templates/.vscode/settings.json +0 -95
  409. package/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  410. package/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  411. package/templates/CLAUDE.md +0 -196
  412. package/templates/COMMIT_CONVENTIONS.md +0 -86
  413. package/templates/OIDC_SETUP.md +0 -300
  414. package/templates/agents.md +0 -171
  415. package/templates/codemod/README.md +0 -1
  416. package/templates/codemod/test.js +0 -93
  417. package/templates/docs/README.md +0 -151
  418. package/templates/scripts/new_feature.sh +0 -80
  419. package/templates/scripts/quality-gates/check-god-objects.js +0 -146
  420. package/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  421. 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
- };