@paths.design/caws-cli 10.2.0 → 11.1.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 (493) hide show
  1. package/README.md +125 -374
  2. package/dist/index.js +45 -787
  3. package/dist/init/harness-detect.d.ts +18 -0
  4. package/dist/init/harness-detect.d.ts.map +1 -0
  5. package/dist/init/harness-detect.js +90 -0
  6. package/dist/init/harness-detect.js.map +1 -0
  7. package/dist/init/hook-install.d.ts +53 -0
  8. package/dist/init/hook-install.d.ts.map +1 -0
  9. package/dist/init/hook-install.js +421 -0
  10. package/dist/init/hook-install.js.map +1 -0
  11. package/dist/init/hook-packs/manifest-claude-code.d.ts +4 -0
  12. package/dist/init/hook-packs/manifest-claude-code.d.ts.map +1 -0
  13. package/dist/init/hook-packs/manifest-claude-code.js +190 -0
  14. package/dist/init/hook-packs/manifest-claude-code.js.map +1 -0
  15. package/dist/init/hook-packs/register.d.ts +19 -0
  16. package/dist/init/hook-packs/register.d.ts.map +1 -0
  17. package/dist/init/hook-packs/register.js +37 -0
  18. package/dist/init/hook-packs/register.js.map +1 -0
  19. package/dist/init/hook-packs/types.d.ts +123 -0
  20. package/dist/init/hook-packs/types.d.ts.map +1 -0
  21. package/dist/init/hook-packs/types.js +29 -0
  22. package/dist/init/hook-packs/types.js.map +1 -0
  23. package/dist/shell/binding/resolve-binding.d.ts +4 -0
  24. package/dist/shell/binding/resolve-binding.d.ts.map +1 -0
  25. package/dist/shell/binding/resolve-binding.js +228 -0
  26. package/dist/shell/binding/resolve-binding.js.map +1 -0
  27. package/dist/shell/binding/types.d.ts +42 -0
  28. package/dist/shell/binding/types.d.ts.map +1 -0
  29. package/dist/shell/binding/types.js +21 -0
  30. package/dist/shell/binding/types.js.map +1 -0
  31. package/dist/shell/commands/claim.d.ts +14 -0
  32. package/dist/shell/commands/claim.d.ts.map +1 -0
  33. package/dist/shell/commands/claim.js +197 -0
  34. package/dist/shell/commands/claim.js.map +1 -0
  35. package/dist/shell/commands/doctor.d.ts +13 -0
  36. package/dist/shell/commands/doctor.d.ts.map +1 -0
  37. package/dist/shell/commands/doctor.js +97 -0
  38. package/dist/shell/commands/doctor.js.map +1 -0
  39. package/dist/shell/commands/evidence.d.ts +28 -0
  40. package/dist/shell/commands/evidence.d.ts.map +1 -0
  41. package/dist/shell/commands/evidence.js +166 -0
  42. package/dist/shell/commands/evidence.js.map +1 -0
  43. package/dist/shell/commands/gates.d.ts +19 -0
  44. package/dist/shell/commands/gates.d.ts.map +1 -0
  45. package/dist/shell/commands/gates.js +208 -0
  46. package/dist/shell/commands/gates.js.map +1 -0
  47. package/dist/shell/commands/init.d.ts +17 -0
  48. package/dist/shell/commands/init.d.ts.map +1 -0
  49. package/dist/shell/commands/init.js +168 -0
  50. package/dist/shell/commands/init.js.map +1 -0
  51. package/dist/shell/commands/scope.d.ts +11 -0
  52. package/dist/shell/commands/scope.d.ts.map +1 -0
  53. package/dist/shell/commands/scope.js +92 -0
  54. package/dist/shell/commands/scope.js.map +1 -0
  55. package/dist/shell/commands/specs.d.ts +41 -0
  56. package/dist/shell/commands/specs.d.ts.map +1 -0
  57. package/dist/shell/commands/specs.js +264 -0
  58. package/dist/shell/commands/specs.js.map +1 -0
  59. package/dist/shell/commands/status.d.ts +15 -0
  60. package/dist/shell/commands/status.d.ts.map +1 -0
  61. package/dist/shell/commands/status.js +106 -0
  62. package/dist/shell/commands/status.js.map +1 -0
  63. package/dist/shell/commands/waiver.d.ts +38 -0
  64. package/dist/shell/commands/waiver.d.ts.map +1 -0
  65. package/dist/shell/commands/waiver.js +240 -0
  66. package/dist/shell/commands/waiver.js.map +1 -0
  67. package/dist/shell/commands/worktree.d.ts +38 -0
  68. package/dist/shell/commands/worktree.d.ts.map +1 -0
  69. package/dist/shell/commands/worktree.js +286 -0
  70. package/dist/shell/commands/worktree.js.map +1 -0
  71. package/dist/shell/gates/disposition.d.ts +23 -0
  72. package/dist/shell/gates/disposition.d.ts.map +1 -0
  73. package/dist/shell/gates/disposition.js +117 -0
  74. package/dist/shell/gates/disposition.js.map +1 -0
  75. package/dist/shell/gates/gate-result-contract.d.ts +39 -0
  76. package/dist/shell/gates/gate-result-contract.d.ts.map +1 -0
  77. package/dist/shell/gates/gate-result-contract.js +150 -0
  78. package/dist/shell/gates/gate-result-contract.js.map +1 -0
  79. package/dist/shell/gates/local-evaluators/budget-limit.d.ts +24 -0
  80. package/dist/shell/gates/local-evaluators/budget-limit.d.ts.map +1 -0
  81. package/dist/shell/gates/local-evaluators/budget-limit.js +67 -0
  82. package/dist/shell/gates/local-evaluators/budget-limit.js.map +1 -0
  83. package/dist/shell/gates/local-evaluators/diff-helpers.d.ts +25 -0
  84. package/dist/shell/gates/local-evaluators/diff-helpers.d.ts.map +1 -0
  85. package/dist/shell/gates/local-evaluators/diff-helpers.js +74 -0
  86. package/dist/shell/gates/local-evaluators/diff-helpers.js.map +1 -0
  87. package/dist/shell/gates/local-evaluators/index.d.ts +28 -0
  88. package/dist/shell/gates/local-evaluators/index.d.ts.map +1 -0
  89. package/dist/shell/gates/local-evaluators/index.js +67 -0
  90. package/dist/shell/gates/local-evaluators/index.js.map +1 -0
  91. package/dist/shell/gates/local-evaluators/scope-boundary.d.ts +23 -0
  92. package/dist/shell/gates/local-evaluators/scope-boundary.d.ts.map +1 -0
  93. package/dist/shell/gates/local-evaluators/scope-boundary.js +67 -0
  94. package/dist/shell/gates/local-evaluators/scope-boundary.js.map +1 -0
  95. package/dist/shell/gates/local-evaluators/spec-completeness.d.ts +12 -0
  96. package/dist/shell/gates/local-evaluators/spec-completeness.d.ts.map +1 -0
  97. package/dist/shell/gates/local-evaluators/spec-completeness.js +73 -0
  98. package/dist/shell/gates/local-evaluators/spec-completeness.js.map +1 -0
  99. package/dist/shell/gates/quality-gates-adapter.d.ts +55 -0
  100. package/dist/shell/gates/quality-gates-adapter.d.ts.map +1 -0
  101. package/dist/shell/gates/quality-gates-adapter.js +161 -0
  102. package/dist/shell/gates/quality-gates-adapter.js.map +1 -0
  103. package/dist/shell/gates/waiver-filter.d.ts +58 -0
  104. package/dist/shell/gates/waiver-filter.d.ts.map +1 -0
  105. package/dist/shell/gates/waiver-filter.js +119 -0
  106. package/dist/shell/gates/waiver-filter.js.map +1 -0
  107. package/dist/shell/index.d.ts +54 -0
  108. package/dist/shell/index.d.ts.map +1 -0
  109. package/dist/shell/index.js +85 -0
  110. package/dist/shell/index.js.map +1 -0
  111. package/dist/shell/register.d.ts +11 -0
  112. package/dist/shell/register.d.ts.map +1 -0
  113. package/dist/shell/register.js +464 -0
  114. package/dist/shell/register.js.map +1 -0
  115. package/dist/shell/render/claim.d.ts +22 -0
  116. package/dist/shell/render/claim.d.ts.map +1 -0
  117. package/dist/shell/render/claim.js +75 -0
  118. package/dist/shell/render/claim.js.map +1 -0
  119. package/dist/shell/render/decision.d.ts +15 -0
  120. package/dist/shell/render/decision.d.ts.map +1 -0
  121. package/dist/shell/render/decision.js +66 -0
  122. package/dist/shell/render/decision.js.map +1 -0
  123. package/dist/shell/render/diagnostic.d.ts +19 -0
  124. package/dist/shell/render/diagnostic.d.ts.map +1 -0
  125. package/dist/shell/render/diagnostic.js +76 -0
  126. package/dist/shell/render/diagnostic.js.map +1 -0
  127. package/dist/shell/render/finding.d.ts +15 -0
  128. package/dist/shell/render/finding.d.ts.map +1 -0
  129. package/dist/shell/render/finding.js +57 -0
  130. package/dist/shell/render/finding.js.map +1 -0
  131. package/dist/shell/render/gates.d.ts +3 -0
  132. package/dist/shell/render/gates.d.ts.map +1 -0
  133. package/dist/shell/render/gates.js +56 -0
  134. package/dist/shell/render/gates.js.map +1 -0
  135. package/dist/shell/render/init-hook-pack.d.ts +16 -0
  136. package/dist/shell/render/init-hook-pack.d.ts.map +1 -0
  137. package/dist/shell/render/init-hook-pack.js +206 -0
  138. package/dist/shell/render/init-hook-pack.js.map +1 -0
  139. package/dist/shell/render/init.d.ts +11 -0
  140. package/dist/shell/render/init.d.ts.map +1 -0
  141. package/dist/shell/render/init.js +32 -0
  142. package/dist/shell/render/init.js.map +1 -0
  143. package/dist/shell/render/status.d.ts +26 -0
  144. package/dist/shell/render/status.d.ts.map +1 -0
  145. package/dist/shell/render/status.js +143 -0
  146. package/dist/shell/render/status.js.map +1 -0
  147. package/dist/shell/render/waiver.d.ts +21 -0
  148. package/dist/shell/render/waiver.d.ts.map +1 -0
  149. package/dist/shell/render/waiver.js +94 -0
  150. package/dist/shell/render/waiver.js.map +1 -0
  151. package/dist/shell/rules.d.ts +37 -0
  152. package/dist/shell/rules.d.ts.map +1 -0
  153. package/dist/shell/rules.js +51 -0
  154. package/dist/shell/rules.js.map +1 -0
  155. package/dist/shell/session/actor.d.ts +14 -0
  156. package/dist/shell/session/actor.d.ts.map +1 -0
  157. package/dist/shell/session/actor.js +34 -0
  158. package/dist/shell/session/actor.js.map +1 -0
  159. package/dist/shell/session/resolve-session.d.ts +5 -0
  160. package/dist/shell/session/resolve-session.d.ts.map +1 -0
  161. package/dist/shell/session/resolve-session.js +239 -0
  162. package/dist/shell/session/resolve-session.js.map +1 -0
  163. package/dist/shell/session/types.d.ts +56 -0
  164. package/dist/shell/session/types.d.ts.map +1 -0
  165. package/dist/shell/session/types.js +15 -0
  166. package/dist/shell/session/types.js.map +1 -0
  167. package/dist/store/agents-store.d.ts +3 -0
  168. package/dist/store/agents-store.d.ts.map +1 -0
  169. package/dist/store/agents-store.js +63 -0
  170. package/dist/store/agents-store.js.map +1 -0
  171. package/dist/store/apply-patch.d.ts +16 -0
  172. package/dist/store/apply-patch.d.ts.map +1 -0
  173. package/dist/store/apply-patch.js +191 -0
  174. package/dist/store/apply-patch.js.map +1 -0
  175. package/dist/store/atomic-write.d.ts +34 -0
  176. package/dist/store/atomic-write.d.ts.map +1 -0
  177. package/dist/store/atomic-write.js +174 -0
  178. package/dist/store/atomic-write.js.map +1 -0
  179. package/dist/store/doctor-snapshot.d.ts +20 -0
  180. package/dist/store/doctor-snapshot.d.ts.map +1 -0
  181. package/dist/store/doctor-snapshot.js +176 -0
  182. package/dist/store/doctor-snapshot.js.map +1 -0
  183. package/dist/store/events-store.d.ts +33 -0
  184. package/dist/store/events-store.d.ts.map +1 -0
  185. package/dist/store/events-store.js +297 -0
  186. package/dist/store/events-store.js.map +1 -0
  187. package/dist/store/index.d.ts +21 -0
  188. package/dist/store/index.d.ts.map +1 -0
  189. package/dist/store/index.js +47 -0
  190. package/dist/store/index.js.map +1 -0
  191. package/dist/store/init-store.d.ts +21 -0
  192. package/dist/store/init-store.d.ts.map +1 -0
  193. package/dist/store/init-store.js +295 -0
  194. package/dist/store/init-store.js.map +1 -0
  195. package/dist/store/json-store.d.ts +3 -0
  196. package/dist/store/json-store.d.ts.map +1 -0
  197. package/dist/store/json-store.js +65 -0
  198. package/dist/store/json-store.js.map +1 -0
  199. package/dist/store/lifecycle-lock.d.ts +34 -0
  200. package/dist/store/lifecycle-lock.d.ts.map +1 -0
  201. package/dist/store/lifecycle-lock.js +168 -0
  202. package/dist/store/lifecycle-lock.js.map +1 -0
  203. package/dist/store/lifecycle-transaction.d.ts +79 -0
  204. package/dist/store/lifecycle-transaction.d.ts.map +1 -0
  205. package/dist/store/lifecycle-transaction.js +319 -0
  206. package/dist/store/lifecycle-transaction.js.map +1 -0
  207. package/dist/store/policy-store.d.ts +3 -0
  208. package/dist/store/policy-store.d.ts.map +1 -0
  209. package/dist/store/policy-store.js +65 -0
  210. package/dist/store/policy-store.js.map +1 -0
  211. package/dist/store/repo-root.d.ts +46 -0
  212. package/dist/store/repo-root.d.ts.map +1 -0
  213. package/dist/store/repo-root.js +145 -0
  214. package/dist/store/repo-root.js.map +1 -0
  215. package/dist/store/rules.d.ts +69 -0
  216. package/dist/store/rules.d.ts.map +1 -0
  217. package/dist/store/rules.js +95 -0
  218. package/dist/store/rules.js.map +1 -0
  219. package/dist/store/specs-store.d.ts +3 -0
  220. package/dist/store/specs-store.d.ts.map +1 -0
  221. package/dist/store/specs-store.js +131 -0
  222. package/dist/store/specs-store.js.map +1 -0
  223. package/dist/store/specs-writer.d.ts +61 -0
  224. package/dist/store/specs-writer.d.ts.map +1 -0
  225. package/dist/store/specs-writer.js +506 -0
  226. package/dist/store/specs-writer.js.map +1 -0
  227. package/dist/store/types.d.ts +84 -0
  228. package/dist/store/types.d.ts.map +1 -0
  229. package/dist/store/types.js +14 -0
  230. package/dist/store/types.js.map +1 -0
  231. package/dist/store/waivers-store.d.ts +25 -0
  232. package/dist/store/waivers-store.d.ts.map +1 -0
  233. package/dist/store/waivers-store.js +232 -0
  234. package/dist/store/waivers-store.js.map +1 -0
  235. package/dist/store/worktrees-store.d.ts +3 -0
  236. package/dist/store/worktrees-store.d.ts.map +1 -0
  237. package/dist/store/worktrees-store.js +62 -0
  238. package/dist/store/worktrees-store.js.map +1 -0
  239. package/dist/store/worktrees-writer.d.ts +77 -0
  240. package/dist/store/worktrees-writer.d.ts.map +1 -0
  241. package/dist/store/worktrees-writer.js +674 -0
  242. package/dist/store/worktrees-writer.js.map +1 -0
  243. package/dist/store/yaml-patch.d.ts +7 -0
  244. package/dist/store/yaml-patch.d.ts.map +1 -0
  245. package/dist/store/yaml-patch.js +250 -0
  246. package/dist/store/yaml-patch.js.map +1 -0
  247. package/dist/store/yaml-store.d.ts +9 -0
  248. package/dist/store/yaml-store.d.ts.map +1 -0
  249. package/dist/store/yaml-store.js +121 -0
  250. package/dist/store/yaml-store.js.map +1 -0
  251. package/package.json +15 -13
  252. package/dist/budget-derivation.js +0 -751
  253. package/dist/cicd-optimizer.js +0 -504
  254. package/dist/commands/agents.js +0 -124
  255. package/dist/commands/archive.js +0 -500
  256. package/dist/commands/burnup.js +0 -198
  257. package/dist/commands/diagnose.js +0 -525
  258. package/dist/commands/evaluate.js +0 -314
  259. package/dist/commands/gates.js +0 -149
  260. package/dist/commands/init.js +0 -857
  261. package/dist/commands/iterate.js +0 -417
  262. package/dist/commands/mode.js +0 -269
  263. package/dist/commands/parallel.js +0 -242
  264. package/dist/commands/plan.js +0 -438
  265. package/dist/commands/provenance.js +0 -1143
  266. package/dist/commands/quality-monitor.js +0 -284
  267. package/dist/commands/scope.js +0 -264
  268. package/dist/commands/session.js +0 -312
  269. package/dist/commands/sidecar.js +0 -74
  270. package/dist/commands/specs.js +0 -1656
  271. package/dist/commands/status.js +0 -1172
  272. package/dist/commands/templates.js +0 -237
  273. package/dist/commands/tool.js +0 -136
  274. package/dist/commands/tutorial.js +0 -480
  275. package/dist/commands/validate.js +0 -357
  276. package/dist/commands/verify-acs.js +0 -443
  277. package/dist/commands/waivers.js +0 -599
  278. package/dist/commands/workflow.js +0 -243
  279. package/dist/commands/worktree.js +0 -502
  280. package/dist/config/lite-scope.js +0 -158
  281. package/dist/config/modes.js +0 -347
  282. package/dist/constants/spec-types.js +0 -65
  283. package/dist/gates/budget-limit.js +0 -121
  284. package/dist/gates/feedback.js +0 -260
  285. package/dist/gates/format.js +0 -179
  286. package/dist/gates/god-object.js +0 -117
  287. package/dist/gates/pipeline.js +0 -167
  288. package/dist/gates/scope-boundary.js +0 -112
  289. package/dist/gates/spec-completeness.js +0 -109
  290. package/dist/gates/todo-detection.js +0 -205
  291. package/dist/generators/jest-config-generator.js +0 -242
  292. package/dist/generators/working-spec.js +0 -237
  293. package/dist/minimal-cli.js +0 -88
  294. package/dist/parallel/parallel-manager.js +0 -433
  295. package/dist/policy/PolicyManager.js +0 -470
  296. package/dist/scaffold/claude-hooks.js +0 -443
  297. package/dist/scaffold/cursor-hooks.js +0 -177
  298. package/dist/scaffold/git-hooks.js +0 -928
  299. package/dist/scaffold/index.js +0 -794
  300. package/dist/session/session-manager.js +0 -653
  301. package/dist/sidecars/index.js +0 -33
  302. package/dist/sidecars/listeners.js +0 -40
  303. package/dist/sidecars/provenance-summary.js +0 -238
  304. package/dist/sidecars/quality-gaps.js +0 -258
  305. package/dist/sidecars/schema.js +0 -149
  306. package/dist/sidecars/spec-drift.js +0 -151
  307. package/dist/sidecars/waiver-draft.js +0 -176
  308. package/dist/spec/SpecFileManager.js +0 -419
  309. package/dist/templates/.caws/schemas/policy.schema.json +0 -117
  310. package/dist/templates/.caws/schemas/scope.schema.json +0 -52
  311. package/dist/templates/.caws/schemas/waivers.schema.json +0 -106
  312. package/dist/templates/.caws/schemas/working-spec.schema.json +0 -340
  313. package/dist/templates/.caws/schemas/worktrees.schema.json +0 -38
  314. package/dist/templates/.caws/templates/working-spec.template.yml +0 -80
  315. package/dist/templates/.caws/tools/README.md +0 -18
  316. package/dist/templates/.caws/tools/scope-guard.js +0 -203
  317. package/dist/templates/.caws/tools-allow.json +0 -331
  318. package/dist/templates/.caws/waivers.yml +0 -19
  319. package/dist/templates/.claude/README.md +0 -190
  320. package/dist/templates/.claude/hooks/audit.sh +0 -121
  321. package/dist/templates/.claude/hooks/block-dangerous.sh +0 -203
  322. package/dist/templates/.claude/hooks/classify_command.py +0 -592
  323. package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  324. package/dist/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  325. package/dist/templates/.claude/hooks/naming-check.sh +0 -100
  326. package/dist/templates/.claude/hooks/protected-paths.sh +0 -39
  327. package/dist/templates/.claude/hooks/quality-check.sh +0 -81
  328. package/dist/templates/.claude/hooks/scan-secrets.sh +0 -85
  329. package/dist/templates/.claude/hooks/scope-guard.sh +0 -381
  330. package/dist/templates/.claude/hooks/session-caws-status.sh +0 -117
  331. package/dist/templates/.claude/hooks/session-log.sh +0 -634
  332. package/dist/templates/.claude/hooks/simplification-guard.sh +0 -92
  333. package/dist/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  334. package/dist/templates/.claude/hooks/test_classify_command.py +0 -370
  335. package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  336. package/dist/templates/.claude/hooks/validate-spec.sh +0 -76
  337. package/dist/templates/.claude/hooks/worktree-guard.sh +0 -220
  338. package/dist/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  339. package/dist/templates/.claude/rules/git-safety.md +0 -26
  340. package/dist/templates/.claude/rules/worktree-isolation.md +0 -101
  341. package/dist/templates/.claude/settings.json +0 -141
  342. package/dist/templates/.cursor/README.md +0 -299
  343. package/dist/templates/.cursor/hooks/audit.sh +0 -55
  344. package/dist/templates/.cursor/hooks/block-dangerous.sh +0 -84
  345. package/dist/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  346. package/dist/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  347. package/dist/templates/.cursor/hooks/format.sh +0 -38
  348. package/dist/templates/.cursor/hooks/naming-check.sh +0 -64
  349. package/dist/templates/.cursor/hooks/scan-secrets.sh +0 -51
  350. package/dist/templates/.cursor/hooks/scope-guard.sh +0 -52
  351. package/dist/templates/.cursor/hooks/session-log.sh +0 -924
  352. package/dist/templates/.cursor/hooks/validate-spec.sh +0 -83
  353. package/dist/templates/.cursor/hooks.json +0 -76
  354. package/dist/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  355. package/dist/templates/.cursor/rules/01-working-style.mdc +0 -50
  356. package/dist/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  357. package/dist/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  358. package/dist/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  359. package/dist/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  360. package/dist/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  361. package/dist/templates/.cursor/rules/07-process-ops.mdc +0 -20
  362. package/dist/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  363. package/dist/templates/.cursor/rules/09-docstrings.mdc +0 -89
  364. package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  365. package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  366. package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  367. package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  368. package/dist/templates/.cursor/rules/README.md +0 -148
  369. package/dist/templates/.github/copilot-instructions.md +0 -82
  370. package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  371. package/dist/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  372. package/dist/templates/.junie/guidelines.md +0 -73
  373. package/dist/templates/.vscode/launch.json +0 -17
  374. package/dist/templates/.vscode/settings.json +0 -95
  375. package/dist/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  376. package/dist/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  377. package/dist/templates/CLAUDE.md +0 -196
  378. package/dist/templates/COMMIT_CONVENTIONS.md +0 -86
  379. package/dist/templates/OIDC_SETUP.md +0 -300
  380. package/dist/templates/agents.md +0 -171
  381. package/dist/templates/codemod/README.md +0 -1
  382. package/dist/templates/codemod/test.js +0 -93
  383. package/dist/templates/docs/README.md +0 -151
  384. package/dist/templates/scripts/new_feature.sh +0 -80
  385. package/dist/templates/scripts/quality-gates/check-god-objects.js +0 -146
  386. package/dist/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  387. package/dist/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
  388. package/dist/test-analysis.js +0 -786
  389. package/dist/tool-interface.js +0 -314
  390. package/dist/tool-loader.js +0 -303
  391. package/dist/tool-validator.js +0 -393
  392. package/dist/utils/agent-display.js +0 -210
  393. package/dist/utils/agent-session.js +0 -344
  394. package/dist/utils/async-utils.js +0 -188
  395. package/dist/utils/command-wrapper.js +0 -200
  396. package/dist/utils/event-log.js +0 -584
  397. package/dist/utils/event-renderer.js +0 -521
  398. package/dist/utils/finalization.js +0 -230
  399. package/dist/utils/git-lock.js +0 -119
  400. package/dist/utils/gitignore-updater.js +0 -158
  401. package/dist/utils/ide-detection.js +0 -133
  402. package/dist/utils/lifecycle-events.js +0 -94
  403. package/dist/utils/project-analysis.js +0 -367
  404. package/dist/utils/promise-utils.js +0 -72
  405. package/dist/utils/quality-gates-errors.js +0 -520
  406. package/dist/utils/quality-gates-utils.js +0 -387
  407. package/dist/utils/schema-validator.js +0 -50
  408. package/dist/utils/spec-resolver.js +0 -711
  409. package/dist/utils/typescript-detector.js +0 -369
  410. package/dist/utils/working-state.js +0 -530
  411. package/dist/utils/yaml-validation.js +0 -156
  412. package/dist/validation/spec-validation.js +0 -924
  413. package/dist/waivers-manager.js +0 -732
  414. package/dist/worktree/worktree-manager.js +0 -1735
  415. package/templates/.caws/schemas/policy.schema.json +0 -117
  416. package/templates/.caws/schemas/scope.schema.json +0 -52
  417. package/templates/.caws/schemas/waivers.schema.json +0 -106
  418. package/templates/.caws/schemas/working-spec.schema.json +0 -340
  419. package/templates/.caws/schemas/worktrees.schema.json +0 -38
  420. package/templates/.caws/templates/working-spec.template.yml +0 -80
  421. package/templates/.caws/tools/README.md +0 -18
  422. package/templates/.caws/tools/scope-guard.js +0 -203
  423. package/templates/.caws/tools-allow.json +0 -331
  424. package/templates/.caws/waivers.yml +0 -19
  425. package/templates/.claude/README.md +0 -190
  426. package/templates/.claude/hooks/audit.sh +0 -121
  427. package/templates/.claude/hooks/block-dangerous.sh +0 -203
  428. package/templates/.claude/hooks/classify_command.py +0 -592
  429. package/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  430. package/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  431. package/templates/.claude/hooks/naming-check.sh +0 -100
  432. package/templates/.claude/hooks/protected-paths.sh +0 -39
  433. package/templates/.claude/hooks/quality-check.sh +0 -81
  434. package/templates/.claude/hooks/scan-secrets.sh +0 -85
  435. package/templates/.claude/hooks/scope-guard.sh +0 -381
  436. package/templates/.claude/hooks/session-caws-status.sh +0 -117
  437. package/templates/.claude/hooks/session-log.sh +0 -634
  438. package/templates/.claude/hooks/simplification-guard.sh +0 -92
  439. package/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  440. package/templates/.claude/hooks/test_classify_command.py +0 -370
  441. package/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  442. package/templates/.claude/hooks/validate-spec.sh +0 -76
  443. package/templates/.claude/hooks/worktree-guard.sh +0 -220
  444. package/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  445. package/templates/.claude/rules/git-safety.md +0 -26
  446. package/templates/.claude/rules/worktree-isolation.md +0 -101
  447. package/templates/.claude/settings.json +0 -141
  448. package/templates/.cursor/README.md +0 -299
  449. package/templates/.cursor/hooks/audit.sh +0 -55
  450. package/templates/.cursor/hooks/block-dangerous.sh +0 -84
  451. package/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  452. package/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  453. package/templates/.cursor/hooks/format.sh +0 -38
  454. package/templates/.cursor/hooks/naming-check.sh +0 -64
  455. package/templates/.cursor/hooks/scan-secrets.sh +0 -51
  456. package/templates/.cursor/hooks/scope-guard.sh +0 -52
  457. package/templates/.cursor/hooks/session-log.sh +0 -924
  458. package/templates/.cursor/hooks/validate-spec.sh +0 -83
  459. package/templates/.cursor/hooks.json +0 -76
  460. package/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  461. package/templates/.cursor/rules/01-working-style.mdc +0 -50
  462. package/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  463. package/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  464. package/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  465. package/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  466. package/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  467. package/templates/.cursor/rules/07-process-ops.mdc +0 -20
  468. package/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  469. package/templates/.cursor/rules/09-docstrings.mdc +0 -89
  470. package/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  471. package/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  472. package/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  473. package/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  474. package/templates/.cursor/rules/README.md +0 -148
  475. package/templates/.github/copilot-instructions.md +0 -82
  476. package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  477. package/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  478. package/templates/.junie/guidelines.md +0 -73
  479. package/templates/.vscode/launch.json +0 -17
  480. package/templates/.vscode/settings.json +0 -95
  481. package/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  482. package/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  483. package/templates/CLAUDE.md +0 -196
  484. package/templates/COMMIT_CONVENTIONS.md +0 -86
  485. package/templates/OIDC_SETUP.md +0 -300
  486. package/templates/agents.md +0 -171
  487. package/templates/codemod/README.md +0 -1
  488. package/templates/codemod/test.js +0 -93
  489. package/templates/docs/README.md +0 -151
  490. package/templates/scripts/new_feature.sh +0 -80
  491. package/templates/scripts/quality-gates/check-god-objects.js +0 -146
  492. package/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  493. 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
- };