@paths.design/caws-cli 10.2.0 → 11.0.0

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