@paths.design/caws-cli 10.1.0 → 11.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (419) hide show
  1. package/README.md +125 -374
  2. package/dist/index.js +43 -756
  3. package/dist/shell/binding/resolve-binding.d.ts +4 -0
  4. package/dist/shell/binding/resolve-binding.d.ts.map +1 -0
  5. package/dist/shell/binding/resolve-binding.js +228 -0
  6. package/dist/shell/binding/resolve-binding.js.map +1 -0
  7. package/dist/shell/binding/types.d.ts +42 -0
  8. package/dist/shell/binding/types.d.ts.map +1 -0
  9. package/dist/shell/binding/types.js +21 -0
  10. package/dist/shell/binding/types.js.map +1 -0
  11. package/dist/shell/commands/claim.d.ts +14 -0
  12. package/dist/shell/commands/claim.d.ts.map +1 -0
  13. package/dist/shell/commands/claim.js +197 -0
  14. package/dist/shell/commands/claim.js.map +1 -0
  15. package/dist/shell/commands/doctor.d.ts +13 -0
  16. package/dist/shell/commands/doctor.d.ts.map +1 -0
  17. package/dist/shell/commands/doctor.js +97 -0
  18. package/dist/shell/commands/doctor.js.map +1 -0
  19. package/dist/shell/commands/evidence.d.ts +28 -0
  20. package/dist/shell/commands/evidence.d.ts.map +1 -0
  21. package/dist/shell/commands/evidence.js +166 -0
  22. package/dist/shell/commands/evidence.js.map +1 -0
  23. package/dist/shell/commands/gates.d.ts +19 -0
  24. package/dist/shell/commands/gates.d.ts.map +1 -0
  25. package/dist/shell/commands/gates.js +181 -0
  26. package/dist/shell/commands/gates.js.map +1 -0
  27. package/dist/shell/commands/init.d.ts +8 -0
  28. package/dist/shell/commands/init.d.ts.map +1 -0
  29. package/dist/shell/commands/init.js +64 -0
  30. package/dist/shell/commands/init.js.map +1 -0
  31. package/dist/shell/commands/scope.d.ts +11 -0
  32. package/dist/shell/commands/scope.d.ts.map +1 -0
  33. package/dist/shell/commands/scope.js +92 -0
  34. package/dist/shell/commands/scope.js.map +1 -0
  35. package/dist/shell/commands/status.d.ts +15 -0
  36. package/dist/shell/commands/status.d.ts.map +1 -0
  37. package/dist/shell/commands/status.js +106 -0
  38. package/dist/shell/commands/status.js.map +1 -0
  39. package/dist/shell/commands/waiver.d.ts +38 -0
  40. package/dist/shell/commands/waiver.d.ts.map +1 -0
  41. package/dist/shell/commands/waiver.js +240 -0
  42. package/dist/shell/commands/waiver.js.map +1 -0
  43. package/dist/shell/gates/disposition.d.ts +23 -0
  44. package/dist/shell/gates/disposition.d.ts.map +1 -0
  45. package/dist/shell/gates/disposition.js +87 -0
  46. package/dist/shell/gates/disposition.js.map +1 -0
  47. package/dist/shell/gates/gate-result-contract.d.ts +39 -0
  48. package/dist/shell/gates/gate-result-contract.d.ts.map +1 -0
  49. package/dist/shell/gates/gate-result-contract.js +150 -0
  50. package/dist/shell/gates/gate-result-contract.js.map +1 -0
  51. package/dist/shell/gates/quality-gates-adapter.d.ts +55 -0
  52. package/dist/shell/gates/quality-gates-adapter.d.ts.map +1 -0
  53. package/dist/shell/gates/quality-gates-adapter.js +161 -0
  54. package/dist/shell/gates/quality-gates-adapter.js.map +1 -0
  55. package/dist/shell/gates/waiver-filter.d.ts +58 -0
  56. package/dist/shell/gates/waiver-filter.d.ts.map +1 -0
  57. package/dist/shell/gates/waiver-filter.js +119 -0
  58. package/dist/shell/gates/waiver-filter.js.map +1 -0
  59. package/dist/shell/index.d.ts +50 -0
  60. package/dist/shell/index.d.ts.map +1 -0
  61. package/dist/shell/index.js +73 -0
  62. package/dist/shell/index.js.map +1 -0
  63. package/dist/shell/register.d.ts +11 -0
  64. package/dist/shell/register.d.ts.map +1 -0
  65. package/dist/shell/register.js +274 -0
  66. package/dist/shell/register.js.map +1 -0
  67. package/dist/shell/render/claim.d.ts +22 -0
  68. package/dist/shell/render/claim.d.ts.map +1 -0
  69. package/dist/shell/render/claim.js +75 -0
  70. package/dist/shell/render/claim.js.map +1 -0
  71. package/dist/shell/render/decision.d.ts +15 -0
  72. package/dist/shell/render/decision.d.ts.map +1 -0
  73. package/dist/shell/render/decision.js +66 -0
  74. package/dist/shell/render/decision.js.map +1 -0
  75. package/dist/shell/render/diagnostic.d.ts +19 -0
  76. package/dist/shell/render/diagnostic.d.ts.map +1 -0
  77. package/dist/shell/render/diagnostic.js +76 -0
  78. package/dist/shell/render/diagnostic.js.map +1 -0
  79. package/dist/shell/render/finding.d.ts +15 -0
  80. package/dist/shell/render/finding.d.ts.map +1 -0
  81. package/dist/shell/render/finding.js +57 -0
  82. package/dist/shell/render/finding.js.map +1 -0
  83. package/dist/shell/render/gates.d.ts +3 -0
  84. package/dist/shell/render/gates.d.ts.map +1 -0
  85. package/dist/shell/render/gates.js +56 -0
  86. package/dist/shell/render/gates.js.map +1 -0
  87. package/dist/shell/render/init.d.ts +11 -0
  88. package/dist/shell/render/init.d.ts.map +1 -0
  89. package/dist/shell/render/init.js +32 -0
  90. package/dist/shell/render/init.js.map +1 -0
  91. package/dist/shell/render/status.d.ts +26 -0
  92. package/dist/shell/render/status.d.ts.map +1 -0
  93. package/dist/shell/render/status.js +143 -0
  94. package/dist/shell/render/status.js.map +1 -0
  95. package/dist/shell/render/waiver.d.ts +21 -0
  96. package/dist/shell/render/waiver.d.ts.map +1 -0
  97. package/dist/shell/render/waiver.js +94 -0
  98. package/dist/shell/render/waiver.js.map +1 -0
  99. package/dist/shell/rules.d.ts +37 -0
  100. package/dist/shell/rules.d.ts.map +1 -0
  101. package/dist/shell/rules.js +51 -0
  102. package/dist/shell/rules.js.map +1 -0
  103. package/dist/shell/session/actor.d.ts +14 -0
  104. package/dist/shell/session/actor.d.ts.map +1 -0
  105. package/dist/shell/session/actor.js +34 -0
  106. package/dist/shell/session/actor.js.map +1 -0
  107. package/dist/shell/session/resolve-session.d.ts +5 -0
  108. package/dist/shell/session/resolve-session.d.ts.map +1 -0
  109. package/dist/shell/session/resolve-session.js +239 -0
  110. package/dist/shell/session/resolve-session.js.map +1 -0
  111. package/dist/shell/session/types.d.ts +56 -0
  112. package/dist/shell/session/types.d.ts.map +1 -0
  113. package/dist/shell/session/types.js +15 -0
  114. package/dist/shell/session/types.js.map +1 -0
  115. package/dist/store/agents-store.d.ts +3 -0
  116. package/dist/store/agents-store.d.ts.map +1 -0
  117. package/dist/store/agents-store.js +63 -0
  118. package/dist/store/agents-store.js.map +1 -0
  119. package/dist/store/apply-patch.d.ts +16 -0
  120. package/dist/store/apply-patch.d.ts.map +1 -0
  121. package/dist/store/apply-patch.js +191 -0
  122. package/dist/store/apply-patch.js.map +1 -0
  123. package/dist/store/atomic-write.d.ts +16 -0
  124. package/dist/store/atomic-write.d.ts.map +1 -0
  125. package/dist/store/atomic-write.js +132 -0
  126. package/dist/store/atomic-write.js.map +1 -0
  127. package/dist/store/doctor-snapshot.d.ts +20 -0
  128. package/dist/store/doctor-snapshot.d.ts.map +1 -0
  129. package/dist/store/doctor-snapshot.js +176 -0
  130. package/dist/store/doctor-snapshot.js.map +1 -0
  131. package/dist/store/events-store.d.ts +33 -0
  132. package/dist/store/events-store.d.ts.map +1 -0
  133. package/dist/store/events-store.js +297 -0
  134. package/dist/store/events-store.js.map +1 -0
  135. package/dist/store/index.d.ts +21 -0
  136. package/dist/store/index.d.ts.map +1 -0
  137. package/dist/store/index.js +47 -0
  138. package/dist/store/index.js.map +1 -0
  139. package/dist/store/init-store.d.ts +21 -0
  140. package/dist/store/init-store.d.ts.map +1 -0
  141. package/dist/store/init-store.js +295 -0
  142. package/dist/store/init-store.js.map +1 -0
  143. package/dist/store/json-store.d.ts +3 -0
  144. package/dist/store/json-store.d.ts.map +1 -0
  145. package/dist/store/json-store.js +65 -0
  146. package/dist/store/json-store.js.map +1 -0
  147. package/dist/store/policy-store.d.ts +3 -0
  148. package/dist/store/policy-store.d.ts.map +1 -0
  149. package/dist/store/policy-store.js +65 -0
  150. package/dist/store/policy-store.js.map +1 -0
  151. package/dist/store/repo-root.d.ts +46 -0
  152. package/dist/store/repo-root.d.ts.map +1 -0
  153. package/dist/store/repo-root.js +145 -0
  154. package/dist/store/repo-root.js.map +1 -0
  155. package/dist/store/rules.d.ts +53 -0
  156. package/dist/store/rules.d.ts.map +1 -0
  157. package/dist/store/rules.js +78 -0
  158. package/dist/store/rules.js.map +1 -0
  159. package/dist/store/specs-store.d.ts +3 -0
  160. package/dist/store/specs-store.d.ts.map +1 -0
  161. package/dist/store/specs-store.js +131 -0
  162. package/dist/store/specs-store.js.map +1 -0
  163. package/dist/store/types.d.ts +84 -0
  164. package/dist/store/types.d.ts.map +1 -0
  165. package/dist/store/types.js +14 -0
  166. package/dist/store/types.js.map +1 -0
  167. package/dist/store/waivers-store.d.ts +25 -0
  168. package/dist/store/waivers-store.d.ts.map +1 -0
  169. package/dist/store/waivers-store.js +232 -0
  170. package/dist/store/waivers-store.js.map +1 -0
  171. package/dist/store/worktrees-store.d.ts +3 -0
  172. package/dist/store/worktrees-store.d.ts.map +1 -0
  173. package/dist/store/worktrees-store.js +62 -0
  174. package/dist/store/worktrees-store.js.map +1 -0
  175. package/dist/store/yaml-store.d.ts +9 -0
  176. package/dist/store/yaml-store.d.ts.map +1 -0
  177. package/dist/store/yaml-store.js +121 -0
  178. package/dist/store/yaml-store.js.map +1 -0
  179. package/package.json +15 -13
  180. package/dist/budget-derivation.js +0 -751
  181. package/dist/cicd-optimizer.js +0 -504
  182. package/dist/commands/archive.js +0 -500
  183. package/dist/commands/burnup.js +0 -198
  184. package/dist/commands/diagnose.js +0 -525
  185. package/dist/commands/evaluate.js +0 -314
  186. package/dist/commands/gates.js +0 -149
  187. package/dist/commands/init.js +0 -857
  188. package/dist/commands/iterate.js +0 -417
  189. package/dist/commands/mode.js +0 -269
  190. package/dist/commands/parallel.js +0 -242
  191. package/dist/commands/plan.js +0 -438
  192. package/dist/commands/provenance.js +0 -1143
  193. package/dist/commands/quality-monitor.js +0 -284
  194. package/dist/commands/scope.js +0 -264
  195. package/dist/commands/session.js +0 -312
  196. package/dist/commands/sidecar.js +0 -74
  197. package/dist/commands/specs.js +0 -1448
  198. package/dist/commands/status.js +0 -1151
  199. package/dist/commands/templates.js +0 -237
  200. package/dist/commands/tool.js +0 -136
  201. package/dist/commands/tutorial.js +0 -480
  202. package/dist/commands/validate.js +0 -357
  203. package/dist/commands/verify-acs.js +0 -443
  204. package/dist/commands/waivers.js +0 -599
  205. package/dist/commands/workflow.js +0 -243
  206. package/dist/commands/worktree.js +0 -386
  207. package/dist/config/lite-scope.js +0 -158
  208. package/dist/config/modes.js +0 -347
  209. package/dist/constants/spec-types.js +0 -65
  210. package/dist/gates/budget-limit.js +0 -121
  211. package/dist/gates/feedback.js +0 -260
  212. package/dist/gates/format.js +0 -179
  213. package/dist/gates/god-object.js +0 -117
  214. package/dist/gates/pipeline.js +0 -167
  215. package/dist/gates/scope-boundary.js +0 -93
  216. package/dist/gates/spec-completeness.js +0 -109
  217. package/dist/gates/todo-detection.js +0 -205
  218. package/dist/generators/jest-config-generator.js +0 -242
  219. package/dist/generators/working-spec.js +0 -237
  220. package/dist/minimal-cli.js +0 -88
  221. package/dist/parallel/parallel-manager.js +0 -433
  222. package/dist/policy/PolicyManager.js +0 -465
  223. package/dist/scaffold/claude-hooks.js +0 -443
  224. package/dist/scaffold/cursor-hooks.js +0 -177
  225. package/dist/scaffold/git-hooks.js +0 -928
  226. package/dist/scaffold/index.js +0 -794
  227. package/dist/session/session-manager.js +0 -653
  228. package/dist/sidecars/index.js +0 -33
  229. package/dist/sidecars/listeners.js +0 -40
  230. package/dist/sidecars/provenance-summary.js +0 -238
  231. package/dist/sidecars/quality-gaps.js +0 -258
  232. package/dist/sidecars/schema.js +0 -149
  233. package/dist/sidecars/spec-drift.js +0 -151
  234. package/dist/sidecars/waiver-draft.js +0 -176
  235. package/dist/spec/SpecFileManager.js +0 -419
  236. package/dist/templates/.caws/schemas/policy.schema.json +0 -112
  237. package/dist/templates/.caws/schemas/scope.schema.json +0 -52
  238. package/dist/templates/.caws/schemas/waivers.schema.json +0 -106
  239. package/dist/templates/.caws/schemas/working-spec.schema.json +0 -340
  240. package/dist/templates/.caws/schemas/worktrees.schema.json +0 -38
  241. package/dist/templates/.caws/templates/working-spec.template.yml +0 -80
  242. package/dist/templates/.caws/tools/README.md +0 -18
  243. package/dist/templates/.caws/tools/scope-guard.js +0 -203
  244. package/dist/templates/.caws/tools-allow.json +0 -331
  245. package/dist/templates/.caws/waivers.yml +0 -19
  246. package/dist/templates/.claude/README.md +0 -190
  247. package/dist/templates/.claude/hooks/audit.sh +0 -121
  248. package/dist/templates/.claude/hooks/block-dangerous.sh +0 -203
  249. package/dist/templates/.claude/hooks/classify_command.py +0 -592
  250. package/dist/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  251. package/dist/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  252. package/dist/templates/.claude/hooks/naming-check.sh +0 -100
  253. package/dist/templates/.claude/hooks/protected-paths.sh +0 -39
  254. package/dist/templates/.claude/hooks/quality-check.sh +0 -81
  255. package/dist/templates/.claude/hooks/scan-secrets.sh +0 -85
  256. package/dist/templates/.claude/hooks/scope-guard.sh +0 -381
  257. package/dist/templates/.claude/hooks/session-caws-status.sh +0 -117
  258. package/dist/templates/.claude/hooks/session-log.sh +0 -634
  259. package/dist/templates/.claude/hooks/simplification-guard.sh +0 -92
  260. package/dist/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  261. package/dist/templates/.claude/hooks/test_classify_command.py +0 -370
  262. package/dist/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  263. package/dist/templates/.claude/hooks/validate-spec.sh +0 -76
  264. package/dist/templates/.claude/hooks/worktree-guard.sh +0 -220
  265. package/dist/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  266. package/dist/templates/.claude/rules/git-safety.md +0 -26
  267. package/dist/templates/.claude/rules/worktree-isolation.md +0 -83
  268. package/dist/templates/.claude/settings.json +0 -141
  269. package/dist/templates/.cursor/README.md +0 -299
  270. package/dist/templates/.cursor/hooks/audit.sh +0 -55
  271. package/dist/templates/.cursor/hooks/block-dangerous.sh +0 -84
  272. package/dist/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  273. package/dist/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  274. package/dist/templates/.cursor/hooks/format.sh +0 -38
  275. package/dist/templates/.cursor/hooks/naming-check.sh +0 -64
  276. package/dist/templates/.cursor/hooks/scan-secrets.sh +0 -51
  277. package/dist/templates/.cursor/hooks/scope-guard.sh +0 -52
  278. package/dist/templates/.cursor/hooks/session-log.sh +0 -924
  279. package/dist/templates/.cursor/hooks/validate-spec.sh +0 -83
  280. package/dist/templates/.cursor/hooks.json +0 -76
  281. package/dist/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  282. package/dist/templates/.cursor/rules/01-working-style.mdc +0 -50
  283. package/dist/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  284. package/dist/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  285. package/dist/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  286. package/dist/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  287. package/dist/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  288. package/dist/templates/.cursor/rules/07-process-ops.mdc +0 -20
  289. package/dist/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  290. package/dist/templates/.cursor/rules/09-docstrings.mdc +0 -89
  291. package/dist/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  292. package/dist/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  293. package/dist/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  294. package/dist/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  295. package/dist/templates/.cursor/rules/README.md +0 -148
  296. package/dist/templates/.github/copilot-instructions.md +0 -82
  297. package/dist/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  298. package/dist/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  299. package/dist/templates/.junie/guidelines.md +0 -73
  300. package/dist/templates/.vscode/launch.json +0 -17
  301. package/dist/templates/.vscode/settings.json +0 -95
  302. package/dist/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  303. package/dist/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  304. package/dist/templates/CLAUDE.md +0 -174
  305. package/dist/templates/COMMIT_CONVENTIONS.md +0 -86
  306. package/dist/templates/OIDC_SETUP.md +0 -300
  307. package/dist/templates/agents.md +0 -145
  308. package/dist/templates/codemod/README.md +0 -1
  309. package/dist/templates/codemod/test.js +0 -93
  310. package/dist/templates/docs/README.md +0 -151
  311. package/dist/templates/scripts/new_feature.sh +0 -80
  312. package/dist/templates/scripts/quality-gates/check-god-objects.js +0 -146
  313. package/dist/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  314. package/dist/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
  315. package/dist/test-analysis.js +0 -786
  316. package/dist/tool-interface.js +0 -314
  317. package/dist/tool-loader.js +0 -303
  318. package/dist/tool-validator.js +0 -393
  319. package/dist/utils/agent-session.js +0 -202
  320. package/dist/utils/async-utils.js +0 -188
  321. package/dist/utils/command-wrapper.js +0 -200
  322. package/dist/utils/event-log.js +0 -584
  323. package/dist/utils/event-renderer.js +0 -521
  324. package/dist/utils/finalization.js +0 -230
  325. package/dist/utils/git-lock.js +0 -119
  326. package/dist/utils/gitignore-updater.js +0 -158
  327. package/dist/utils/ide-detection.js +0 -133
  328. package/dist/utils/lifecycle-events.js +0 -94
  329. package/dist/utils/project-analysis.js +0 -367
  330. package/dist/utils/promise-utils.js +0 -72
  331. package/dist/utils/quality-gates-errors.js +0 -520
  332. package/dist/utils/quality-gates-utils.js +0 -387
  333. package/dist/utils/schema-validator.js +0 -50
  334. package/dist/utils/spec-resolver.js +0 -711
  335. package/dist/utils/typescript-detector.js +0 -369
  336. package/dist/utils/working-state.js +0 -530
  337. package/dist/utils/yaml-validation.js +0 -156
  338. package/dist/validation/spec-validation.js +0 -921
  339. package/dist/waivers-manager.js +0 -732
  340. package/dist/worktree/worktree-manager.js +0 -1374
  341. package/templates/.caws/schemas/policy.schema.json +0 -112
  342. package/templates/.caws/schemas/scope.schema.json +0 -52
  343. package/templates/.caws/schemas/waivers.schema.json +0 -106
  344. package/templates/.caws/schemas/working-spec.schema.json +0 -340
  345. package/templates/.caws/schemas/worktrees.schema.json +0 -38
  346. package/templates/.caws/templates/working-spec.template.yml +0 -80
  347. package/templates/.caws/tools/README.md +0 -18
  348. package/templates/.caws/tools/scope-guard.js +0 -203
  349. package/templates/.caws/tools-allow.json +0 -331
  350. package/templates/.caws/waivers.yml +0 -19
  351. package/templates/.claude/README.md +0 -190
  352. package/templates/.claude/hooks/audit.sh +0 -121
  353. package/templates/.claude/hooks/block-dangerous.sh +0 -203
  354. package/templates/.claude/hooks/classify_command.py +0 -592
  355. package/templates/.claude/hooks/doc-frontmatter-check.sh +0 -173
  356. package/templates/.claude/hooks/lite-sprawl-check.sh +0 -145
  357. package/templates/.claude/hooks/naming-check.sh +0 -100
  358. package/templates/.claude/hooks/protected-paths.sh +0 -39
  359. package/templates/.claude/hooks/quality-check.sh +0 -81
  360. package/templates/.claude/hooks/scan-secrets.sh +0 -85
  361. package/templates/.claude/hooks/scope-guard.sh +0 -381
  362. package/templates/.claude/hooks/session-caws-status.sh +0 -117
  363. package/templates/.claude/hooks/session-log.sh +0 -634
  364. package/templates/.claude/hooks/simplification-guard.sh +0 -92
  365. package/templates/.claude/hooks/stop-worktree-check.sh +0 -46
  366. package/templates/.claude/hooks/test_classify_command.py +0 -370
  367. package/templates/.claude/hooks/test_wrapper_smoke.sh +0 -96
  368. package/templates/.claude/hooks/validate-spec.sh +0 -76
  369. package/templates/.claude/hooks/worktree-guard.sh +0 -220
  370. package/templates/.claude/hooks/worktree-write-guard.sh +0 -190
  371. package/templates/.claude/rules/git-safety.md +0 -26
  372. package/templates/.claude/rules/worktree-isolation.md +0 -83
  373. package/templates/.claude/settings.json +0 -141
  374. package/templates/.cursor/README.md +0 -299
  375. package/templates/.cursor/hooks/audit.sh +0 -55
  376. package/templates/.cursor/hooks/block-dangerous.sh +0 -84
  377. package/templates/.cursor/hooks/caws-quality-check.sh +0 -52
  378. package/templates/.cursor/hooks/caws-scope-guard.sh +0 -130
  379. package/templates/.cursor/hooks/format.sh +0 -38
  380. package/templates/.cursor/hooks/naming-check.sh +0 -64
  381. package/templates/.cursor/hooks/scan-secrets.sh +0 -51
  382. package/templates/.cursor/hooks/scope-guard.sh +0 -52
  383. package/templates/.cursor/hooks/session-log.sh +0 -924
  384. package/templates/.cursor/hooks/validate-spec.sh +0 -83
  385. package/templates/.cursor/hooks.json +0 -76
  386. package/templates/.cursor/rules/00-claims-verification.mdc +0 -144
  387. package/templates/.cursor/rules/01-working-style.mdc +0 -50
  388. package/templates/.cursor/rules/02-quality-gates.mdc +0 -368
  389. package/templates/.cursor/rules/03-naming-and-refactor.mdc +0 -33
  390. package/templates/.cursor/rules/04-logging-language-style.mdc +0 -23
  391. package/templates/.cursor/rules/05-safe-defaults-guards.mdc +0 -23
  392. package/templates/.cursor/rules/06-typescript-conventions.mdc +0 -36
  393. package/templates/.cursor/rules/07-process-ops.mdc +0 -20
  394. package/templates/.cursor/rules/08-solid-and-architecture.mdc +0 -16
  395. package/templates/.cursor/rules/09-docstrings.mdc +0 -89
  396. package/templates/.cursor/rules/10-documentation-quality-standards.mdc +0 -385
  397. package/templates/.cursor/rules/11-scope-management-waivers.mdc +0 -381
  398. package/templates/.cursor/rules/12-implementation-completeness.mdc +0 -516
  399. package/templates/.cursor/rules/13-language-agnostic-standards.mdc +0 -578
  400. package/templates/.cursor/rules/README.md +0 -148
  401. package/templates/.github/copilot-instructions.md +0 -82
  402. package/templates/.idea/runConfigurations/CAWS_Evaluate.xml +0 -5
  403. package/templates/.idea/runConfigurations/CAWS_Validate.xml +0 -5
  404. package/templates/.junie/guidelines.md +0 -73
  405. package/templates/.vscode/launch.json +0 -17
  406. package/templates/.vscode/settings.json +0 -95
  407. package/templates/.windsurf/rules/caws-quality-standards.md +0 -54
  408. package/templates/.windsurf/workflows/caws-guided-development.md +0 -92
  409. package/templates/CLAUDE.md +0 -174
  410. package/templates/COMMIT_CONVENTIONS.md +0 -86
  411. package/templates/OIDC_SETUP.md +0 -300
  412. package/templates/agents.md +0 -145
  413. package/templates/codemod/README.md +0 -1
  414. package/templates/codemod/test.js +0 -93
  415. package/templates/docs/README.md +0 -151
  416. package/templates/scripts/new_feature.sh +0 -80
  417. package/templates/scripts/quality-gates/check-god-objects.js +0 -146
  418. package/templates/scripts/quality-gates/run-quality-gates.js +0 -50
  419. package/templates/scripts/v3/analysis/todo_analyzer.py +0 -1997
@@ -1,634 +0,0 @@
1
- #!/bin/bash
2
- # Session Logger for Claude Code → ChatGPT Context Transfer
3
- #
4
- # On Stop/PreCompact: reads the full transcript from ~/.claude/ and generates:
5
- # session.txt — lightweight index (header + turn list + exploration + audit)
6
- # turn-001.txt — per-turn narrative (user message + reasoning + key tool output)
7
- # turn-001.json — per-turn structured data (reasoning + tools + edits + results)
8
- #
9
- # Output: ./tmp/<session-id>/
10
- #
11
- # Wired into: SessionStart (metadata), Stop (generate), PreCompact (safety net)
12
-
13
- set -euo pipefail
14
-
15
- INPUT=$(cat)
16
-
17
- # --- Parse common fields ---
18
- SESSION_ID=$(echo "$INPUT" | jq -r '.session_id // "unknown"')
19
- HOOK_EVENT=$(echo "$INPUT" | jq -r '.hook_event_name // "unknown"')
20
- CWD=$(echo "$INPUT" | jq -r '.cwd // "."')
21
- TRANSCRIPT_PATH=$(echo "$INPUT" | jq -r '.transcript_path // ""')
22
- TIMESTAMP=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
23
-
24
- # --- Log directory ---
25
- LOG_DIR="${CWD}/tmp/${SESSION_ID}"
26
- mkdir -p "$LOG_DIR"
27
-
28
- SESSION_MD="$LOG_DIR/session.txt"
29
- META_FILE="$LOG_DIR/.meta.json"
30
-
31
- # ============================================================
32
- # Helper: resolve transcript path
33
- # ============================================================
34
- resolve_transcript() {
35
- if [ -n "$TRANSCRIPT_PATH" ] && [ -f "$TRANSCRIPT_PATH" ]; then
36
- echo "$TRANSCRIPT_PATH"
37
- return
38
- fi
39
- local slug
40
- slug=$(echo "$CWD" | sed 's|/|-|g; s|^-||')
41
- local candidate="$HOME/.claude/projects/${slug}/${SESSION_ID}.jsonl"
42
- if [ -f "$candidate" ]; then
43
- echo "$candidate"
44
- return
45
- fi
46
- candidate="$HOME/.claude/projects/-${slug}/${SESSION_ID}.jsonl"
47
- if [ -f "$candidate" ]; then
48
- echo "$candidate"
49
- return
50
- fi
51
- echo ""
52
- }
53
-
54
- # ============================================================
55
- # Helper: make path relative to project
56
- # ============================================================
57
- rel_path() {
58
- echo "$1" | sed "s|${CWD}/||"
59
- }
60
-
61
- # ============================================================
62
- # Generate per-turn files + session.md index from transcript
63
- # ============================================================
64
- generate_session_output() {
65
- local transcript="$1"
66
- local branch head_sha dirty_count
67
- branch=$(cd "$CWD" 2>/dev/null && git rev-parse --abbrev-ref HEAD 2>/dev/null || echo "unknown")
68
- head_sha=$(cd "$CWD" 2>/dev/null && git rev-parse --short HEAD 2>/dev/null || echo "unknown")
69
- dirty_count=$(cd "$CWD" 2>/dev/null && git status --porcelain 2>/dev/null | wc -l | tr -d ' ' || echo "0")
70
-
71
- # --- Read metadata if available ---
72
- local started_at model start_sha
73
- if [ -f "$META_FILE" ]; then
74
- started_at=$(jq -r '.local_time // "unknown"' "$META_FILE")
75
- model=$(jq -r '.model // "unknown"' "$META_FILE")
76
- start_sha=$(jq -r '.head_sha // ""' "$META_FILE")
77
- else
78
- started_at="(resumed session)"
79
- model="unknown"
80
- start_sha=""
81
- fi
82
-
83
- if [ -z "$transcript" ] || [ ! -f "$transcript" ]; then
84
- cat > "$SESSION_MD" << MDEOF
85
- # Session Log: $(basename "$CWD")
86
-
87
- | Field | Value |
88
- |-------|-------|
89
- | Session ID | \`${SESSION_ID}\` |
90
- | Started | ${started_at} |
91
- | Model | ${model} |
92
- | Branch | \`${branch}\` @ \`${head_sha}\` |
93
-
94
- ---
95
-
96
- _No transcript found. Narrative extraction unavailable._
97
- MDEOF
98
- return
99
- fi
100
-
101
- # --- Generate per-turn files via python ---
102
- # jq emits each content block as a separate chronological event.
103
- # Python accumulates into turns and writes sequential timeline per turn.
104
- jq -c '
105
- if .type == "user" then
106
- if (.message.content | type) == "string" then
107
- {ev: "user_text", text: .message.content}
108
- elif (.message.content | type) == "array" then
109
- .message.content[]? |
110
- if .type == "tool_result" then
111
- {ev: "tool_result", id: .tool_use_id, content: ((.content // "") | tostring), is_error: (.is_error // false)}
112
- else
113
- empty
114
- end
115
- else
116
- empty
117
- end
118
- elif .type == "assistant" then
119
- .message.content[]? |
120
- if .type == "text" then
121
- {ev: "text", text: .text}
122
- elif .type == "tool_use" then
123
- {ev: "tool_use", name, id,
124
- file: (.input.file_path // null),
125
- command: (.input.command // null),
126
- description: (.input.description // null),
127
- pattern: (.input.pattern // null),
128
- prompt: (.input.prompt // null),
129
- subagent_type: (.input.subagent_type // null)}
130
- else
131
- empty
132
- end
133
- else
134
- empty
135
- end
136
- ' "$transcript" 2>/dev/null > "$LOG_DIR/.events.jsonl"
137
-
138
- # Write python script to temp file (can't pipe + heredoc simultaneously)
139
- local pyscript
140
- pyscript=$(mktemp "${TMPDIR:-/tmp}/session-log-XXXX.py")
141
- trap "rm -f '$pyscript'" RETURN
142
- cat > "$pyscript" << 'PYEOF'
143
- import json, sys, os
144
-
145
- log_dir = sys.argv[1]
146
- cwd = sys.argv[2]
147
- session_id = sys.argv[3]
148
- started_at = sys.argv[4]
149
- model = sys.argv[5]
150
- branch = sys.argv[6]
151
- head_sha = sys.argv[7]
152
- dirty_count = sys.argv[8]
153
- start_sha = sys.argv[9]
154
-
155
- def rel(path):
156
- if path and path.startswith(cwd + "/"):
157
- return path[len(cwd) + 1:]
158
- return path or ""
159
-
160
- def decode_structured_text_payload(raw):
161
- """Decode JSON-escaped text payloads (e.g., Agent/Task tool outputs)."""
162
- if not isinstance(raw, str):
163
- return raw
164
- payload = raw.strip()
165
- if not payload or payload[0] not in "[{":
166
- return raw
167
- try:
168
- parsed = json.loads(payload)
169
- except Exception:
170
- return raw
171
-
172
- text_blocks = []
173
- if isinstance(parsed, dict):
174
- parsed = [parsed]
175
- if isinstance(parsed, list):
176
- for item in parsed:
177
- if not isinstance(item, dict):
178
- continue
179
- text = item.get("text")
180
- if isinstance(text, str) and text.strip():
181
- text_blocks.append(text)
182
-
183
- if text_blocks:
184
- return "\n\n".join(text_blocks)
185
- return raw
186
-
187
- # ---- Accumulate turns as chronological event timelines ----
188
- turns = []
189
- # Each turn: {user, timeline: [{kind, ...}, ...], edits, reads, searches, commands}
190
- current = {"user": None, "timeline": [], "edits": [], "reads": [], "searches": [], "commands": []}
191
-
192
- def new_turn(user_text):
193
- return {
194
- "user": user_text if user_text else None,
195
- "timeline": [], "edits": [], "reads": [], "searches": [], "commands": [],
196
- }
197
-
198
- # Track pending tool_use IDs to match with results
199
- pending_tools = {} # id -> {name, ...}
200
-
201
- NOISE_PREFIXES = ("<local-command", "<command-name", "<local-command-stdout",
202
- "<local-command-caveat", "This session is being continued")
203
-
204
- # Keywords that make a tool result "notable" (worth showing inline)
205
- NOTABLE_KW = ["error", "fail", "refusal", "mismatch", "passed", "assert",
206
- "traceback", "exception", "pytest", "PASSED", "FAILED", "TypedRefusal"]
207
-
208
- for line in sys.stdin:
209
- try:
210
- entry = json.loads(line)
211
- except json.JSONDecodeError:
212
- continue
213
-
214
- ev = entry.get("ev")
215
-
216
- if ev == "user_text":
217
- text = entry["text"]
218
- if any(text.startswith(p) for p in NOISE_PREFIXES):
219
- continue
220
- if not text.strip():
221
- continue
222
- if current["user"] or current["timeline"]:
223
- turns.append(current)
224
- current = new_turn(text)
225
-
226
- elif ev == "text":
227
- text = entry.get("text", "")
228
- if len(text) > 20:
229
- current["timeline"].append({"kind": "reasoning", "text": text})
230
-
231
- elif ev == "tool_use":
232
- name = entry.get("name", "")
233
- tid = entry.get("id", "")
234
- tool_entry = {"kind": "tool_call", "name": name, "id": tid}
235
-
236
- if name in ("Write", "Edit"):
237
- f = rel(entry.get("file"))
238
- tool_entry["file"] = f
239
- if f and f not in current["edits"]:
240
- current["edits"].append(f)
241
- elif name == "Read":
242
- f = rel(entry.get("file"))
243
- tool_entry["file"] = f
244
- if f and f not in current["reads"]:
245
- current["reads"].append(f)
246
- elif name in ("Grep", "Glob"):
247
- pat = entry.get("pattern", "")
248
- tool_entry["pattern"] = pat
249
- if pat:
250
- current["searches"].append(pat)
251
- elif name == "Bash":
252
- cmd = entry.get("command", "")
253
- desc = entry.get("description", "")
254
- tool_entry["command"] = cmd
255
- tool_entry["description"] = desc or ""
256
- if cmd:
257
- current["commands"].append({"cmd": cmd, "desc": desc or ""})
258
- elif name == "Task":
259
- tool_entry["prompt"] = entry.get("prompt", "")
260
- tool_entry["subagent_type"] = entry.get("subagent_type", "")
261
-
262
- current["timeline"].append(tool_entry)
263
- pending_tools[tid] = tool_entry
264
-
265
- elif ev == "tool_result":
266
- tid = entry.get("id", "")
267
- content = entry.get("content", "")
268
- is_error = entry.get("is_error", False)
269
- tool_info = pending_tools.get(tid, {})
270
- name = tool_info.get("name", "unknown")
271
-
272
- # Always capture tool results for Bash, Task, Agent.
273
- # For Read/Write/Edit, only capture if notable (errors, test output, etc.)
274
- # to avoid dumping entire file contents into turn logs.
275
- always_capture = name in ("Bash", "Task", "Agent")
276
- notable = is_error
277
- if not notable and content:
278
- content_lower = content.lower()
279
- notable = any(kw.lower() in content_lower for kw in NOTABLE_KW)
280
-
281
- if (always_capture or notable) and content:
282
- # Cap file-content tools (full file reads/writes blow out turn files)
283
- display = content
284
- if name in ("Read", "Write", "Edit") and len(content) > 2000:
285
- display = content[:2000] + "\n...(file content truncated)"
286
- elif name in ("Task", "Agent"):
287
- display = decode_structured_text_payload(content)
288
- elif name == "Bash" and len(content) > 5000:
289
- display = content[:5000] + "\n...(output truncated at 5000 chars)"
290
- # Graft result onto the original tool_call entry (not a separate timeline item)
291
- if tool_info:
292
- tool_info["output"] = display
293
- tool_info["is_error"] = is_error
294
- else:
295
- # Orphan result (no matching call) — append standalone
296
- current["timeline"].append({
297
- "kind": "tool_output",
298
- "name": name,
299
- "content": display,
300
- "is_error": is_error,
301
- })
302
-
303
- if current["user"] or current["timeline"]:
304
- turns.append(current)
305
-
306
- # ---- Write per-turn files ----
307
- turn_index = []
308
-
309
- for i, turn in enumerate(turns):
310
- num = i + 1
311
- padded = f"{num:03d}"
312
-
313
- # --- Build per-turn markdown: chronological timeline ---
314
- md_lines = [f"# Turn {num}", ""]
315
-
316
- if turn["user"]:
317
- md_lines.extend([f"> ---user---\n{turn['user']}\n---\/user---", ""])
318
-
319
- for event in turn["timeline"]:
320
- kind = event["kind"]
321
-
322
- if kind == "reasoning":
323
- text = event["text"]
324
- md_lines.append(text)
325
- md_lines.extend(["", "---", ""])
326
-
327
- elif kind == "tool_call":
328
- name = event.get("name", "")
329
- if name in ("Read", "Glob"):
330
- f = event.get("file") or event.get("pattern", "")
331
- md_lines.append(f"`{name}` {f}")
332
- elif name in ("Write", "Edit"):
333
- md_lines.append(f"`{name}` {event.get('file', '')}")
334
- elif name == "Bash":
335
- cmd = event.get("command", "")
336
- desc = event.get("description", "")
337
- header = f"`Bash` _{desc}_" if desc else "`Bash`"
338
- if len(cmd) > 120:
339
- md_lines.extend([header, "```", cmd, "```"])
340
- else:
341
- md_lines.append(f"{header} `{cmd}`" if cmd else header)
342
- elif name in ("Grep",):
343
- md_lines.append(f"`Grep` {event.get('pattern', '')}")
344
- elif name == "Task":
345
- sa = event.get("subagent_type", "subagent")
346
- prompt = event.get("prompt", "")
347
- header = f"`Task` ({sa})" if sa else "`Task` (subagent)"
348
- if prompt:
349
- # Show the dispatch prompt so readers know what the subagent was asked
350
- short_prompt = prompt[:500]
351
- if len(prompt) > 500:
352
- short_prompt += "..."
353
- md_lines.extend([header, "", f"> {short_prompt}", ""])
354
- else:
355
- md_lines.append(f"`{name}`")
356
- md_lines.append("")
357
-
358
- # If tool result was grafted onto this call, render it inline
359
- if "output" in event:
360
- output = event["output"]
361
- is_error = event.get("is_error", False)
362
- label = "error" if is_error else "output"
363
- md_lines.extend([
364
- f"**{name}** ({label}):",
365
- "```",
366
- output,
367
- "```",
368
- "",
369
- ])
370
-
371
- elif kind == "tool_output":
372
- # Orphan result (no matching call found) — render standalone
373
- content = event.get("content", "")
374
- name = event.get("name", "")
375
- is_error = event.get("is_error", False)
376
- label = "error" if is_error else "output"
377
- md_lines.extend([
378
- f"**{name}** ({label}):",
379
- "```",
380
- content,
381
- "```",
382
- "",
383
- ])
384
-
385
- # Write turn markdown
386
- with open(os.path.join(log_dir, f"turn-{padded}.txt"), "w") as f:
387
- f.write("\n".join(md_lines))
388
-
389
- # --- Build per-turn JSON: chronological timeline ---
390
- tool_summary = {}
391
- for event in turn["timeline"]:
392
- if event["kind"] == "tool_call":
393
- n = event.get("name", "")
394
- tool_summary[n] = tool_summary.get(n, 0) + 1
395
-
396
- def group_by_ext(paths):
397
- groups = {}
398
- for p in paths:
399
- ext = os.path.splitext(p)[1] or "(no ext)"
400
- groups.setdefault(ext, []).append(p)
401
- return groups
402
-
403
- turn_json = {
404
- "turn": num,
405
- "user": turn["user"],
406
- "timeline": turn["timeline"],
407
- "tool_summary": tool_summary,
408
- "files_edited": group_by_ext(turn["edits"]),
409
- "files_read": group_by_ext(turn["reads"]),
410
- "searches": turn["searches"],
411
- "commands": [c["cmd"] for c in turn["commands"]],
412
- }
413
-
414
- with open(os.path.join(log_dir, f"turn-{padded}.json"), "w") as f:
415
- json.dump(turn_json, f, indent=2)
416
-
417
- # Index entry
418
- user_preview = (turn["user"] or "(no user message)")[:120]
419
- reasoning_count = sum(1 for e in turn["timeline"] if e["kind"] == "reasoning")
420
- tool_count = sum(1 for e in turn["timeline"] if e["kind"] == "tool_call")
421
- turn_index.append({
422
- "num": num,
423
- "padded": padded,
424
- "user_preview": user_preview,
425
- "reasoning_count": reasoning_count,
426
- "tool_count": tool_count,
427
- "edits": turn["edits"],
428
- })
429
-
430
- # ---- Write session.md index ----
431
- with open(os.path.join(log_dir, "session.txt"), "w") as f:
432
- f.write(f"# Session Log: {os.path.basename(cwd)}\n\n")
433
- f.write("| Field | Value |\n")
434
- f.write("|-------|-------|\n")
435
- f.write(f"| Session ID | `{session_id}` |\n")
436
- f.write(f"| Started | {started_at} |\n")
437
- f.write(f"| Model | {model} |\n")
438
- f.write(f"| Branch | `{branch}` @ `{head_sha}` |\n")
439
- f.write(f"| Turns | {len(turn_index)} |\n")
440
- f.write("\n---\n\n")
441
-
442
- f.write("## Turns\n\n")
443
- for t in turn_index:
444
- edits_str = ", ".join(f"`{e}`" for e in t["edits"][:3])
445
- if len(t["edits"]) > 3:
446
- edits_str += f" +{len(t['edits'])-3} more"
447
- summary = f"{t['reasoning_count']} msgs, {t['tool_count']} tools"
448
- if edits_str:
449
- summary += f" | {edits_str}"
450
- f.write(f"- **[Turn {t['num']}](turn-{t['padded']}.md)** — {t['user_preview']}\n")
451
- f.write(f" _{summary}_\n")
452
-
453
- f.write("\n---\n\n")
454
-
455
- # Exploration summary (deduplicated across all turns)
456
- all_reads = []
457
- all_searches = []
458
- all_edits = []
459
- all_commands = []
460
- for turn in turns:
461
- all_reads.extend(turn["reads"])
462
- all_searches.extend(turn["searches"])
463
- all_edits.extend(turn["edits"])
464
- all_commands.extend(turn["commands"])
465
-
466
- f.write("## Exploration\n")
467
- f.write("_Files read and searches performed (deduplicated)._\n\n")
468
- for r in sorted(set(all_reads)):
469
- f.write(f"- READ `{r}`\n")
470
- for s in sorted(set(all_searches)):
471
- f.write(f"- SEARCH `{s}`\n")
472
- f.write("\n")
473
-
474
- f.write("## Audit\n")
475
- f.write("_Edits, commands, git activity._\n\n")
476
- for e in sorted(set(all_edits)):
477
- f.write(f"- EDIT `{e}`\n")
478
- for cmd in all_commands:
479
- short = cmd["cmd"][:120]
480
- # Only log meaningful commands
481
- meaningful = any(kw in short for kw in [
482
- "pytest", "cargo test", "ruff", "mypy", "npm test",
483
- "git log", "git diff", "git status", "git add", "git commit",
484
- "git merge", "caws ", "pip install", "make", "cargo build"
485
- ])
486
- if meaningful:
487
- if cmd["desc"]:
488
- f.write(f"- BASH `{short}` — {cmd['desc']}\n")
489
- else:
490
- f.write(f"- BASH `{short}`\n")
491
- f.write("\n")
492
-
493
- f.write("## Session Snapshot\n\n")
494
- f.write("| Field | Value |\n")
495
- f.write("|-------|-------|\n")
496
- f.write(f"| Branch | `{branch}` @ `{head_sha}` |\n")
497
- f.write(f"| Dirty files | {dirty_count} |\n")
498
- f.write(f"| Total turns | {len(turn_index)} |\n")
499
-
500
- PYEOF
501
-
502
- # Run the python script with events as input
503
- python3 "$pyscript" "$LOG_DIR" "$CWD" "$SESSION_ID" "$started_at" "$model" "$branch" "$head_sha" "$dirty_count" "$start_sha" < "$LOG_DIR/.events.jsonl"
504
- rm -f "$LOG_DIR/.events.jsonl"
505
- }
506
-
507
- # ============================================================
508
- # EVENT: SessionStart — save metadata
509
- # ============================================================
510
- handle_session_start() {
511
- local model source branch head_sha dirty_count full_time
512
- model=$(echo "$INPUT" | jq -r '.model // "unknown"')
513
- source=$(echo "$INPUT" | jq -r '.source // "unknown"')
514
- branch=$(cd "$CWD" 2>/dev/null && git rev-parse --abbrev-ref HEAD 2>/dev/null || echo "unknown")
515
- head_sha=$(cd "$CWD" 2>/dev/null && git rev-parse --short HEAD 2>/dev/null || echo "unknown")
516
- dirty_count=$(cd "$CWD" 2>/dev/null && git status --porcelain 2>/dev/null | wc -l | tr -d ' ' || echo "0")
517
- full_time=$(date +"%Y-%m-%d %H:%M:%S %Z")
518
-
519
- jq -cn \
520
- --arg sid "$SESSION_ID" \
521
- --arg ts "$TIMESTAMP" \
522
- --arg lt "$full_time" \
523
- --arg model "$model" \
524
- --arg source "$source" \
525
- --arg branch "$branch" \
526
- --arg head "$head_sha" \
527
- --arg dirty "$dirty_count" \
528
- --arg project "$(basename "$CWD")" \
529
- --arg transcript "$TRANSCRIPT_PATH" \
530
- '{session_id: $sid, started_at: $ts, local_time: $lt, model: $model, source: $source, branch: $branch, head_sha: $head, dirty_files: $dirty, project: $project, transcript_path: $transcript}' \
531
- > "$META_FILE"
532
-
533
- # Generate initial output (may be empty if transcript not ready)
534
- generate_session_output "$(resolve_transcript)"
535
- }
536
-
537
- # ============================================================
538
- # EVENT: Stop — regenerate from transcript
539
- # ============================================================
540
- handle_stop() {
541
- generate_session_output "$(resolve_transcript)"
542
- }
543
-
544
- # ============================================================
545
- # EVENT: PreCompact — safety net before context eviction
546
- # ============================================================
547
- handle_pre_compact() {
548
- generate_session_output "$(resolve_transcript)"
549
- }
550
-
551
- # ============================================================
552
- # Agent registry heartbeat — register this agent with CAWS
553
- # ============================================================
554
- AGENTS_REGISTRY="${CWD}/.caws/agents.json"
555
-
556
- heartbeat_agent() {
557
- [ "$SESSION_ID" = "unknown" ] && return
558
- [ ! -d "${CWD}/.caws" ] && return
559
-
560
- local model_val
561
- model_val=$(echo "$INPUT" | jq -r '.model // "unknown"' 2>/dev/null)
562
-
563
- local registry
564
- if [ -f "$AGENTS_REGISTRY" ]; then
565
- registry=$(cat "$AGENTS_REGISTRY" 2>/dev/null || echo '{"version":1,"agents":{}}')
566
- else
567
- registry='{"version":1,"agents":{}}'
568
- fi
569
-
570
- registry=$(echo "$registry" | python3 -c "
571
- import json, sys
572
- from datetime import datetime, timedelta, timezone
573
-
574
- TTL = timedelta(minutes=30)
575
- now = datetime.now(timezone.utc)
576
- sid = '$SESSION_ID'
577
- model = '$model_val'
578
-
579
- data = json.load(sys.stdin)
580
- agents = data.get('agents', {})
581
-
582
- pruned = {}
583
- for k, entry in agents.items():
584
- try:
585
- last = datetime.fromisoformat(entry['lastSeen'].replace('Z', '+00:00'))
586
- if now - last < TTL:
587
- pruned[k] = entry
588
- except (KeyError, ValueError):
589
- pass
590
-
591
- existing = pruned.get(sid, {})
592
- pruned[sid] = {
593
- 'sessionId': sid,
594
- 'platform': 'claude-code',
595
- 'model': model if model != 'unknown' else existing.get('model'),
596
- 'specId': existing.get('specId'),
597
- 'ttl': 1800000,
598
- 'firstSeen': existing.get('firstSeen', now.strftime('%Y-%m-%dT%H:%M:%SZ')),
599
- 'lastSeen': now.strftime('%Y-%m-%dT%H:%M:%SZ'),
600
- }
601
-
602
- data['agents'] = pruned
603
- json.dump(data, sys.stdout, indent=2)
604
- " 2>/dev/null)
605
-
606
- [ -n "$registry" ] && echo "$registry" > "$AGENTS_REGISTRY"
607
- }
608
-
609
- remove_agent() {
610
- [ "$SESSION_ID" = "unknown" ] && return
611
- [ ! -f "$AGENTS_REGISTRY" ] && return
612
-
613
- python3 -c "
614
- import json
615
- sid = '$SESSION_ID'
616
- with open('$AGENTS_REGISTRY', 'r') as f:
617
- data = json.load(f)
618
- data.get('agents', {}).pop(sid, None)
619
- with open('$AGENTS_REGISTRY', 'w') as f:
620
- json.dump(data, f, indent=2)
621
- " 2>/dev/null || true
622
- }
623
-
624
- # ============================================================
625
- # DISPATCH
626
- # ============================================================
627
- case "$HOOK_EVENT" in
628
- SessionStart) handle_session_start; heartbeat_agent ;;
629
- Stop) handle_stop; remove_agent ;;
630
- PreCompact) handle_pre_compact; heartbeat_agent ;;
631
- *) ;; # Other events: no-op
632
- esac
633
-
634
- exit 0