@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,653 +0,0 @@
1
- /**
2
- * @fileoverview CAWS Session Capsule Manager
3
- * Manages session lifecycle and capsule persistence for multi-agent coordination.
4
- * Each session produces a structured capsule that captures baseline state on entry
5
- * and work summary + verification evidence on exit.
6
- * @author @darianrosebrook
7
- */
8
-
9
- const { execFileSync } = require('child_process');
10
- const fs = require('fs-extra');
11
- const path = require('path');
12
- const crypto = require('crypto');
13
-
14
- const { mergeFilesTouched } = require('../utils/working-state');
15
- const { appendEventSync } = require('../utils/event-log');
16
-
17
- const SESSIONS_DIR = '.caws/sessions';
18
- const REGISTRY_FILE = '.caws/sessions.json';
19
- const CAPSULE_SCHEMA_VERSION = 'caws.capsule.v1';
20
-
21
- /**
22
- * Get the git repository root
23
- * @returns {string} Absolute path to repo root
24
- */
25
- function getRepoRoot() {
26
- return execFileSync('git', ['rev-parse', '--show-toplevel'], {
27
- encoding: 'utf8',
28
- }).trim();
29
- }
30
-
31
- /**
32
- * Get current HEAD revision (short hash)
33
- * @param {string} cwd - Working directory
34
- * @returns {string}
35
- */
36
- function getHeadRev(cwd) {
37
- try {
38
- return execFileSync('git', ['rev-parse', '--short', 'HEAD'], {
39
- encoding: 'utf8',
40
- cwd,
41
- }).trim();
42
- } catch {
43
- return 'unknown';
44
- }
45
- }
46
-
47
- /**
48
- * Get current branch name
49
- * @param {string} cwd - Working directory
50
- * @returns {string}
51
- */
52
- function getCurrentBranch(cwd) {
53
- try {
54
- return execFileSync('git', ['rev-parse', '--abbrev-ref', 'HEAD'], {
55
- encoding: 'utf8',
56
- cwd,
57
- }).trim();
58
- } catch {
59
- return 'detached';
60
- }
61
- }
62
-
63
- /**
64
- * Get dirty files in working tree
65
- * @param {string} cwd - Working directory
66
- * @returns {{ paths: string[], dirty: boolean }}
67
- */
68
- function getWorkspaceFingerprint(cwd) {
69
- try {
70
- const output = execFileSync('git', ['status', '--porcelain'], {
71
- encoding: 'utf8',
72
- cwd,
73
- });
74
- const paths = output
75
- .split('\n')
76
- .filter(Boolean)
77
- .map((line) => line.substring(3).trim());
78
- return { paths_touched: paths, dirty: paths.length > 0 };
79
- } catch {
80
- return { paths_touched: [], dirty: false };
81
- }
82
- }
83
-
84
- /**
85
- * Load the best available spec synchronously (feature specs first, then legacy).
86
- * @param {string} root - Repository root
87
- * @param {string} [specId] - Optional specific spec ID
88
- * @returns {object|null} Parsed spec object or null
89
- */
90
- function loadBestSpecSync(root, specId) {
91
- const yaml = require('js-yaml');
92
-
93
- // If a specific spec ID is requested, load it directly
94
- if (specId) {
95
- for (const ext of ['.yaml', '.yml']) {
96
- const p = path.join(root, '.caws/specs', `${specId}${ext}`);
97
- if (fs.existsSync(p)) {
98
- try { return yaml.load(fs.readFileSync(p, 'utf8')); } catch { /* skip */ }
99
- }
100
- }
101
- }
102
-
103
- // Check registry for active feature specs
104
- const registryPath = path.join(root, '.caws/specs/registry.json');
105
- if (fs.existsSync(registryPath)) {
106
- try {
107
- const registry = JSON.parse(fs.readFileSync(registryPath, 'utf8'));
108
- const specs = registry.specs || {};
109
- const activeIds = Object.keys(specs).filter(
110
- id => specs[id].status !== 'closed' && specs[id].status !== 'archived'
111
- );
112
- for (const id of activeIds) {
113
- const specEntry = specs[id];
114
- const p = path.join(root, '.caws/specs', specEntry.path || `${id}.yaml`);
115
- if (fs.existsSync(p)) {
116
- try { return yaml.load(fs.readFileSync(p, 'utf8')); } catch { /* skip */ }
117
- }
118
- }
119
- } catch { /* fall through */ }
120
- }
121
-
122
- // Legacy fallback
123
- const legacyPath = path.join(root, '.caws/working-spec.yaml');
124
- if (fs.existsSync(legacyPath)) {
125
- try { return yaml.load(fs.readFileSync(legacyPath, 'utf8')); } catch { /* skip */ }
126
- }
127
-
128
- return null;
129
- }
130
-
131
- /**
132
- * Get project name from best available spec or directory
133
- * @param {string} root - Repository root
134
- * @param {string} [specId] - Optional specific spec ID
135
- * @returns {string}
136
- */
137
- function getProjectName(root, specId) {
138
- try {
139
- const spec = loadBestSpecSync(root, specId);
140
- if (spec) {
141
- return spec.title || spec.id || path.basename(root);
142
- }
143
- } catch {
144
- // Fall through
145
- }
146
- return path.basename(root);
147
- }
148
-
149
- /**
150
- * Get skein ID from best available spec
151
- * @param {string} root - Repository root
152
- * @param {string} [specId] - Optional specific spec ID
153
- * @returns {string}
154
- */
155
- function getSkeinId(root, specId) {
156
- try {
157
- const spec = loadBestSpecSync(root, specId);
158
- if (spec) {
159
- return spec.id || 'unknown';
160
- }
161
- } catch {
162
- // Fall through
163
- }
164
- return 'unknown';
165
- }
166
-
167
- /**
168
- * Load the session registry
169
- * @param {string} root - Repository root
170
- * @returns {Object} Registry object
171
- */
172
- function loadRegistry(root) {
173
- const registryPath = path.join(root, REGISTRY_FILE);
174
- try {
175
- if (fs.existsSync(registryPath)) {
176
- return JSON.parse(fs.readFileSync(registryPath, 'utf8'));
177
- }
178
- } catch {
179
- // Corrupted registry, start fresh
180
- }
181
- return { version: 1, sessions: {} };
182
- }
183
-
184
- /**
185
- * Save the session registry
186
- * @param {string} root - Repository root
187
- * @param {Object} registry - Registry object
188
- */
189
- function saveRegistry(root, registry) {
190
- const registryPath = path.join(root, REGISTRY_FILE);
191
- fs.ensureDirSync(path.dirname(registryPath));
192
- fs.writeFileSync(registryPath, JSON.stringify(registry, null, 2));
193
- }
194
-
195
- /**
196
- * Generate a deterministic session ID
197
- * @returns {string}
198
- */
199
- function generateSessionId() {
200
- const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
201
- const suffix = crypto.randomBytes(4).toString('hex');
202
- return `${timestamp}__${suffix}`;
203
- }
204
-
205
- /**
206
- * Start a new session, creating the initial capsule with baseline state
207
- * @param {Object} options - Session options
208
- * @param {string} [options.role] - Agent role (worker, integrator, qa)
209
- * @param {string} [options.specId] - Associated feature spec ID
210
- * @param {string[]} [options.allowedGlobs] - Allowed file patterns
211
- * @param {string[]} [options.forbiddenGlobs] - Forbidden file patterns
212
- * @param {string} [options.intent] - What this session intends to accomplish
213
- * @returns {Object} Created capsule
214
- */
215
- function startSession(options = {}) {
216
- const root = getRepoRoot();
217
- const registry = loadRegistry(root);
218
- const sessionId = generateSessionId();
219
-
220
- const {
221
- role = 'worker',
222
- specId,
223
- allowedGlobs = [],
224
- forbiddenGlobs = [],
225
- intent = '',
226
- } = options;
227
-
228
- // Build scope from spec if available and no explicit globs provided
229
- let scope = {
230
- allowed_globs: allowedGlobs,
231
- forbidden_globs: forbiddenGlobs,
232
- };
233
-
234
- if (specId && allowedGlobs.length === 0) {
235
- try {
236
- const yaml = require('js-yaml');
237
- const specPath = path.join(root, `.caws/specs/${specId}.yaml`);
238
- if (fs.existsSync(specPath)) {
239
- const spec = yaml.load(fs.readFileSync(specPath, 'utf8'));
240
- if (spec.scope) {
241
- scope.allowed_globs = spec.scope.in || [];
242
- scope.forbidden_globs = spec.scope.out || [];
243
- }
244
- }
245
- } catch {
246
- // Non-fatal: scope stays as provided
247
- }
248
- }
249
-
250
- const capsule = {
251
- schema: CAPSULE_SCHEMA_VERSION,
252
- project: getProjectName(root, specId),
253
- skein_id: getSkeinId(root, specId),
254
- session_id: sessionId,
255
- role,
256
- spec_id: specId || null,
257
- scope,
258
- base_state: {
259
- head_rev: getHeadRev(root),
260
- branch: getCurrentBranch(root),
261
- workspace_fingerprint: getWorkspaceFingerprint(root),
262
- },
263
- started_at: new Date().toISOString(),
264
- ended_at: null,
265
- work_summary: {
266
- intent: intent || '',
267
- paths_touched: [],
268
- artifacts_written: [],
269
- commits: [],
270
- },
271
- verification: {
272
- tests_run: [],
273
- determinism_checks: [],
274
- },
275
- known_issues: [],
276
- handoff: {
277
- next_actions: [],
278
- risk_notes: [],
279
- },
280
- };
281
-
282
- // Persist capsule
283
- const sessionsDir = path.join(root, SESSIONS_DIR);
284
- fs.ensureDirSync(sessionsDir);
285
- const capsulePath = path.join(sessionsDir, `${sessionId}.json`);
286
- fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
287
-
288
- // Update registry
289
- registry.sessions[sessionId] = {
290
- path: `${sessionId}.json`,
291
- role,
292
- spec_id: specId || null,
293
- status: 'active',
294
- started_at: capsule.started_at,
295
- ended_at: null,
296
- head_rev: capsule.base_state.head_rev,
297
- branch: capsule.base_state.branch,
298
- };
299
- saveRegistry(root, registry);
300
-
301
- // EVLOG-001: emit session_started event via the sync append path.
302
- // spec_id is optional for this event type; we include it only when
303
- // the session is bound to a spec.
304
- const sessionStartedEvent = {
305
- actor: 'session',
306
- event: 'session_started',
307
- data: {
308
- session_id: sessionId,
309
- role,
310
- branch: capsule.base_state.branch,
311
- head_rev: capsule.base_state.head_rev,
312
- },
313
- };
314
- if (specId) sessionStartedEvent.spec_id = specId;
315
- appendEventSync(sessionStartedEvent, { projectRoot: root, session_id: sessionId });
316
-
317
- return capsule;
318
- }
319
-
320
- /**
321
- * Add a checkpoint to the current (most recent active) session
322
- * @param {Object} data - Checkpoint data
323
- * @param {string} [data.sessionId] - Specific session ID (uses latest active if omitted)
324
- * @param {string[]} [data.pathsTouched] - Files changed
325
- * @param {string[]} [data.artifactsWritten] - Generated artifacts
326
- * @param {Object[]} [data.testsRun] - Test results { name, status, evidence }
327
- * @param {Object[]} [data.determinismChecks] - Determinism checks { name, status, total }
328
- * @param {Object[]} [data.knownIssues] - Issues discovered { type, description }
329
- * @param {string} [data.intent] - Updated intent description
330
- * @returns {Object} Updated capsule
331
- */
332
- function checkpointSession(data = {}) {
333
- const root = getRepoRoot();
334
- const registry = loadRegistry(root);
335
-
336
- // Find session
337
- const sessionId = data.sessionId || findActiveSession(registry);
338
- if (!sessionId) {
339
- throw new Error('No active session found. Start one with: caws session start');
340
- }
341
-
342
- const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
343
- if (!fs.existsSync(capsulePath)) {
344
- throw new Error(`Session capsule not found: ${sessionId}`);
345
- }
346
-
347
- const capsule = JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
348
-
349
- // Merge checkpoint data
350
- if (data.intent) {
351
- capsule.work_summary.intent = data.intent;
352
- }
353
- if (data.pathsTouched) {
354
- const existing = new Set(capsule.work_summary.paths_touched);
355
- for (const p of data.pathsTouched) existing.add(p);
356
- capsule.work_summary.paths_touched = [...existing];
357
- }
358
- if (data.artifactsWritten) {
359
- const existing = new Set(capsule.work_summary.artifacts_written);
360
- for (const a of data.artifactsWritten) existing.add(a);
361
- capsule.work_summary.artifacts_written = [...existing];
362
- }
363
- if (data.testsRun) {
364
- capsule.verification.tests_run.push(...data.testsRun);
365
- }
366
- if (data.determinismChecks) {
367
- capsule.verification.determinism_checks.push(...data.determinismChecks);
368
- }
369
- if (data.knownIssues) {
370
- capsule.known_issues.push(...data.knownIssues);
371
- }
372
-
373
- // Record current commit as a checkpoint
374
- const currentRev = getHeadRev(root);
375
- if (currentRev !== capsule.base_state.head_rev) {
376
- capsule.work_summary.commits.push({
377
- rev: currentRev,
378
- checkpoint_at: new Date().toISOString(),
379
- });
380
- }
381
-
382
- // Write updated capsule
383
- fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
384
-
385
- // Bridge to working state (per-spec)
386
- if (capsule.spec_id && capsule.work_summary.paths_touched.length > 0) {
387
- try { mergeFilesTouched(capsule.spec_id, capsule.work_summary.paths_touched, root); } catch { /* non-fatal */ }
388
- }
389
-
390
- return capsule;
391
- }
392
-
393
- /**
394
- * End a session, finalizing the capsule with handoff information
395
- * @param {Object} data - End session data
396
- * @param {string} [data.sessionId] - Specific session ID (uses latest active if omitted)
397
- * @param {string[]} [data.nextActions] - What the next session should do
398
- * @param {string[]} [data.riskNotes] - Risk notes for handoff
399
- * @returns {Object} Finalized capsule
400
- */
401
- function endSession(data = {}) {
402
- const root = getRepoRoot();
403
- const registry = loadRegistry(root);
404
-
405
- const sessionId = data.sessionId || findActiveSession(registry);
406
- if (!sessionId) {
407
- throw new Error('No active session found.');
408
- }
409
-
410
- const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
411
- if (!fs.existsSync(capsulePath)) {
412
- throw new Error(`Session capsule not found: ${sessionId}`);
413
- }
414
-
415
- const capsule = JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
416
-
417
- // Finalize
418
- capsule.ended_at = new Date().toISOString();
419
-
420
- // Capture final workspace state
421
- const fingerprint = getWorkspaceFingerprint(root);
422
- capsule.work_summary.paths_touched = [
423
- ...new Set([...capsule.work_summary.paths_touched, ...fingerprint.paths_touched]),
424
- ];
425
-
426
- // Record final commit
427
- const finalRev = getHeadRev(root);
428
- if (
429
- finalRev !== capsule.base_state.head_rev &&
430
- !capsule.work_summary.commits.some((c) => c.rev === finalRev)
431
- ) {
432
- capsule.work_summary.commits.push({
433
- rev: finalRev,
434
- checkpoint_at: new Date().toISOString(),
435
- });
436
- }
437
-
438
- // Handoff
439
- if (data.nextActions) {
440
- capsule.handoff.next_actions = data.nextActions;
441
- }
442
- if (data.riskNotes) {
443
- capsule.handoff.risk_notes = data.riskNotes;
444
- }
445
-
446
- // Flag if dirty
447
- if (fingerprint.dirty) {
448
- capsule.known_issues.push({
449
- type: 'warning',
450
- description: `Session ended with ${fingerprint.paths_touched.length} uncommitted file(s).`,
451
- });
452
- }
453
-
454
- // Write finalized capsule
455
- fs.writeFileSync(capsulePath, JSON.stringify(capsule, null, 2));
456
-
457
- // Bridge to working state (per-spec)
458
- if (capsule.spec_id && capsule.work_summary.paths_touched.length > 0) {
459
- try { mergeFilesTouched(capsule.spec_id, capsule.work_summary.paths_touched, root); } catch { /* non-fatal */ }
460
- }
461
-
462
- // Update registry
463
- registry.sessions[sessionId].status = 'completed';
464
- registry.sessions[sessionId].ended_at = capsule.ended_at;
465
- saveRegistry(root, registry);
466
-
467
- // EVLOG-001: emit session_ended event with final files_touched list.
468
- // The renderer uses this to merge file touches into the spec view.
469
- const sessionEndedEvent = {
470
- actor: 'session',
471
- event: 'session_ended',
472
- data: {
473
- session_id: sessionId,
474
- files_touched: capsule.work_summary.paths_touched || [],
475
- outcome: capsule.known_issues.some((i) => i.type === 'error') ? 'error' : 'success',
476
- },
477
- };
478
- if (capsule.spec_id) {
479
- sessionEndedEvent.spec_id = capsule.spec_id;
480
- sessionEndedEvent.data.spec_id = capsule.spec_id;
481
- }
482
- appendEventSync(sessionEndedEvent, { projectRoot: root, session_id: sessionId });
483
-
484
- return capsule;
485
- }
486
-
487
- /**
488
- * List all sessions
489
- * @param {Object} [options] - List options
490
- * @param {string} [options.status] - Filter by status (active, completed)
491
- * @param {number} [options.limit] - Max entries to return
492
- * @returns {Object[]} Session entries
493
- */
494
- function listSessions(options = {}) {
495
- const root = getRepoRoot();
496
- const registry = loadRegistry(root);
497
-
498
- let entries = Object.entries(registry.sessions).map(([id, meta]) => ({
499
- id,
500
- ...meta,
501
- }));
502
-
503
- if (options.status) {
504
- entries = entries.filter((e) => e.status === options.status);
505
- }
506
-
507
- // Sort by started_at descending (most recent first)
508
- entries.sort((a, b) => new Date(b.started_at) - new Date(a.started_at));
509
-
510
- if (options.limit) {
511
- entries = entries.slice(0, options.limit);
512
- }
513
-
514
- return entries;
515
- }
516
-
517
- /**
518
- * Show a specific session's full capsule
519
- * @param {string} sessionId - Session ID (or "latest" for most recent)
520
- * @returns {Object} Full capsule
521
- */
522
- function showSession(sessionId) {
523
- const root = getRepoRoot();
524
-
525
- if (sessionId === 'latest') {
526
- const registry = loadRegistry(root);
527
- const active = findActiveSession(registry);
528
- if (active) {
529
- sessionId = active;
530
- } else {
531
- // Find most recent completed
532
- const entries = Object.entries(registry.sessions).sort(
533
- (a, b) => new Date(b[1].started_at) - new Date(a[1].started_at)
534
- );
535
- if (entries.length === 0) throw new Error('No sessions found.');
536
- sessionId = entries[0][0];
537
- }
538
- }
539
-
540
- const capsulePath = path.join(root, SESSIONS_DIR, `${sessionId}.json`);
541
- if (!fs.existsSync(capsulePath)) {
542
- throw new Error(`Session capsule not found: ${sessionId}`);
543
- }
544
-
545
- return JSON.parse(fs.readFileSync(capsulePath, 'utf8'));
546
- }
547
-
548
- /**
549
- * Briefing output for session start hooks - returns structured text
550
- * @returns {string} Briefing text
551
- */
552
- function getBriefing() {
553
- const root = getRepoRoot();
554
- const registry = loadRegistry(root);
555
-
556
- const lines = [];
557
- lines.push('--- CAWS Session Briefing ---');
558
-
559
- // Git baseline
560
- const headRev = getHeadRev(root);
561
- const branch = getCurrentBranch(root);
562
- const fingerprint = getWorkspaceFingerprint(root);
563
- lines.push(`Git: ${branch} @ ${headRev} (${fingerprint.paths_touched.length} dirty files)`);
564
-
565
- if (fingerprint.dirty) {
566
- lines.push('WARNING: Working tree has uncommitted changes from a prior session.');
567
- }
568
-
569
- // Active sessions
570
- const activeSessions = Object.entries(registry.sessions)
571
- .filter(([, meta]) => meta.status === 'active')
572
- .map(([id, meta]) => ({ id, ...meta }));
573
-
574
- if (activeSessions.length > 0) {
575
- lines.push(`Active sessions: ${activeSessions.length}`);
576
- for (const s of activeSessions) {
577
- lines.push(` - ${s.id} (${s.role}, spec: ${s.spec_id || 'none'})`);
578
- }
579
- }
580
-
581
- // Last completed session handoff
582
- const completedSessions = Object.entries(registry.sessions)
583
- .filter(([, meta]) => meta.status === 'completed')
584
- .sort((a, b) => new Date(b[1].ended_at) - new Date(a[1].ended_at));
585
-
586
- if (completedSessions.length > 0) {
587
- const [lastId] = completedSessions[0];
588
- try {
589
- const capsule = showSession(lastId);
590
- if (capsule.handoff.next_actions.length > 0) {
591
- lines.push('Handoff from prior session:');
592
- for (const action of capsule.handoff.next_actions) {
593
- lines.push(` - ${action}`);
594
- }
595
- }
596
- if (capsule.known_issues.length > 0) {
597
- lines.push('Known issues from prior session:');
598
- for (const issue of capsule.known_issues) {
599
- lines.push(` - [${issue.type}] ${issue.description}`);
600
- }
601
- }
602
- } catch {
603
- // Non-fatal
604
- }
605
- }
606
-
607
- lines.push('---');
608
- lines.push("Run 'caws session start' to begin a tracked session.");
609
- lines.push('--- End CAWS Briefing ---');
610
-
611
- return lines.join('\n');
612
- }
613
-
614
- /**
615
- * Find the most recent active session
616
- * @param {Object} registry - Session registry
617
- * @returns {string|null} Session ID or null
618
- */
619
- function findActiveSession(registry) {
620
- const active = Object.entries(registry.sessions)
621
- .filter(([, meta]) => meta.status === 'active')
622
- .sort((a, b) => new Date(b[1].started_at) - new Date(a[1].started_at));
623
-
624
- return active.length > 0 ? active[0][0] : null;
625
- }
626
-
627
- /**
628
- * Find all active sessions on a specific branch
629
- * @param {string} branch - Branch name to search
630
- * @returns {Object[]} Active sessions on that branch with id and metadata
631
- */
632
- function findActiveSessionsOnBranch(branch) {
633
- const root = getRepoRoot();
634
- const registry = loadRegistry(root);
635
- return Object.entries(registry.sessions)
636
- .filter(([, meta]) => meta.status === 'active' && meta.branch === branch)
637
- .map(([id, meta]) => ({ id, ...meta }));
638
- }
639
-
640
- module.exports = {
641
- startSession,
642
- checkpointSession,
643
- endSession,
644
- listSessions,
645
- showSession,
646
- getBriefing,
647
- loadRegistry,
648
- getRepoRoot,
649
- SESSIONS_DIR,
650
- REGISTRY_FILE,
651
- CAPSULE_SCHEMA_VERSION,
652
- findActiveSessionsOnBranch,
653
- };
@@ -1,33 +0,0 @@
1
- /**
2
- * @fileoverview Sidecar Registry
3
- * Central registry of all bounded governance sidecars.
4
- * @author @darianrosebrook
5
- */
6
-
7
- const { analyzeSpecDrift } = require('./spec-drift');
8
- const { diagnoseQualityGaps } = require('./quality-gaps');
9
- const { draftWaiver } = require('./waiver-draft');
10
- const { summarizeProvenance } = require('./provenance-summary');
11
- const { createSidecarOutput, createNoStateOutput, formatSidecarText } = require('./schema');
12
-
13
- /**
14
- * Registry of available sidecars.
15
- * Each entry has a function and a short description for help text.
16
- */
17
- const SIDECARS = {
18
- drift: { fn: analyzeSpecDrift, description: 'Analyze spec drift vs implementation evidence' },
19
- gaps: { fn: diagnoseQualityGaps, description: 'Diagnose quality gaps preventing phase advancement' },
20
- 'waiver-draft': { fn: draftWaiver, description: 'Generate pre-filled waiver templates from gate failures' },
21
- provenance: { fn: summarizeProvenance, description: 'Summarize work provenance for merge readiness' },
22
- };
23
-
24
- module.exports = {
25
- SIDECARS,
26
- analyzeSpecDrift,
27
- diagnoseQualityGaps,
28
- draftWaiver,
29
- summarizeProvenance,
30
- createSidecarOutput,
31
- createNoStateOutput,
32
- formatSidecarText,
33
- };