@cmetech/otto 1.1.1 → 1.2.5

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 (433) hide show
  1. package/dist/coworker/persona-commands.d.ts +1 -0
  2. package/dist/coworker/persona-commands.js +5 -0
  3. package/dist/coworker/persona-commands.test.d.ts +1 -0
  4. package/dist/coworker/persona-commands.test.js +45 -0
  5. package/dist/resources/.managed-resources-content-hash +1 -1
  6. package/dist/resources/extensions/coworker-artifacts/artifacts-command.js +31 -0
  7. package/dist/resources/extensions/coworker-artifacts/artifacts-singleton.js +17 -0
  8. package/dist/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
  9. package/dist/resources/extensions/coworker-artifacts/index.js +125 -0
  10. package/dist/resources/extensions/coworker-artifacts/list-tool.js +27 -0
  11. package/dist/resources/extensions/coworker-artifacts/open-tool.js +25 -0
  12. package/dist/resources/extensions/coworker-memory/extension-manifest.json +13 -0
  13. package/dist/resources/extensions/coworker-memory/index.js +219 -0
  14. package/dist/resources/extensions/coworker-memory/memorize-tool.js +10 -0
  15. package/dist/resources/extensions/coworker-memory/memory-command.js +157 -0
  16. package/dist/resources/extensions/coworker-memory/memory-singleton.js +55 -0
  17. package/dist/resources/extensions/coworker-memory/recall-tool.js +18 -0
  18. package/dist/resources/extensions/coworker-memory/session-hooks.js +45 -0
  19. package/dist/resources/extensions/coworker-scratchpad/attach-banners.js +53 -0
  20. package/dist/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
  21. package/dist/resources/extensions/coworker-scratchpad/format-age.js +9 -0
  22. package/dist/resources/extensions/coworker-scratchpad/helpers.js +38 -0
  23. package/dist/resources/extensions/coworker-scratchpad/index.js +199 -0
  24. package/dist/resources/extensions/coworker-scratchpad/mime-bundle.js +20 -0
  25. package/dist/resources/extensions/coworker-scratchpad/scratchpad-tool.js +118 -0
  26. package/dist/resources/extensions/coworker-scratchpad/session-sidecar.js +60 -0
  27. package/dist/resources/extensions/coworker-scratchpad/sp-command.js +597 -0
  28. package/dist/resources/extensions/coworker-scratchpad/workspace-pointer.js +41 -0
  29. package/dist/resources/extensions/coworker-scratchpad/workspace-root.js +17 -0
  30. package/dist/resources/extensions/coworker-vault/audit-command.js +35 -0
  31. package/dist/resources/extensions/coworker-vault/connect-command.js +42 -0
  32. package/dist/resources/extensions/coworker-vault/datasource-command.js +50 -0
  33. package/dist/resources/extensions/coworker-vault/extension-manifest.json +12 -0
  34. package/dist/resources/extensions/coworker-vault/index.js +171 -0
  35. package/dist/resources/extensions/coworker-vault/test-helpers.js +86 -0
  36. package/dist/resources/extensions/coworker-vault/vault-singleton.js +24 -0
  37. package/dist/resources/extensions/otto/commands/release-notes/_data.js +83 -0
  38. package/dist/resources/extensions/otto/commands/release-notes/command.js +15 -4
  39. package/dist/resources/extensions/otto/index.js +31 -6
  40. package/dist/resources/extensions/shared/coworker-paths.js +8 -0
  41. package/dist/resources/extensions/slash-commands/{audit.js → audit-codebase.js} +4 -4
  42. package/dist/resources/extensions/slash-commands/extension-manifest.json +1 -1
  43. package/dist/resources/extensions/slash-commands/index.js +2 -2
  44. package/dist/resources/extensions/subagent/index.js +8 -1
  45. package/dist/resources/extensions/subagent/launch.js +37 -5
  46. package/dist/resources/extensions/subagent/run-store.js +1 -0
  47. package/dist/resources/extensions/workflow/bootstrap/register-extension.js +2 -0
  48. package/dist/resources/extensions/workflow/bootstrap/register-hooks.js +10 -0
  49. package/dist/resources/extensions/workflow/health-widget-core.js +1 -1
  50. package/dist/resources/extensions/workflow/persona-status.js +87 -0
  51. package/package.json +26 -10
  52. package/packages/contracts/package.json +1 -1
  53. package/packages/coworker-artifacts/dist/artifact-store.d.ts +25 -0
  54. package/packages/coworker-artifacts/dist/artifact-store.js +187 -0
  55. package/packages/coworker-artifacts/dist/dir-snapshot.d.ts +7 -0
  56. package/packages/coworker-artifacts/dist/dir-snapshot.js +54 -0
  57. package/packages/coworker-artifacts/dist/errors.d.ts +18 -0
  58. package/packages/coworker-artifacts/dist/errors.js +37 -0
  59. package/packages/coworker-artifacts/dist/index.d.ts +7 -0
  60. package/packages/coworker-artifacts/dist/index.js +7 -0
  61. package/packages/coworker-artifacts/dist/readme-renderer.d.ts +5 -0
  62. package/packages/coworker-artifacts/dist/readme-renderer.js +47 -0
  63. package/packages/coworker-artifacts/dist/resolve-uri.d.ts +3 -0
  64. package/packages/coworker-artifacts/dist/resolve-uri.js +29 -0
  65. package/packages/coworker-artifacts/dist/slug.d.ts +4 -0
  66. package/packages/coworker-artifacts/dist/slug.js +32 -0
  67. package/packages/coworker-artifacts/dist/types.d.ts +52 -0
  68. package/packages/coworker-artifacts/dist/types.js +1 -0
  69. package/packages/coworker-artifacts/package.json +20 -0
  70. package/packages/coworker-artifacts/src/artifact-store.test.ts +188 -0
  71. package/packages/coworker-artifacts/src/artifact-store.ts +206 -0
  72. package/packages/coworker-artifacts/src/artifacts-integration.test.ts +109 -0
  73. package/packages/coworker-artifacts/src/dir-snapshot.test.ts +71 -0
  74. package/packages/coworker-artifacts/src/dir-snapshot.ts +52 -0
  75. package/packages/coworker-artifacts/src/errors.test.ts +37 -0
  76. package/packages/coworker-artifacts/src/errors.ts +28 -0
  77. package/packages/coworker-artifacts/src/index.test.ts +22 -0
  78. package/packages/coworker-artifacts/src/index.ts +7 -0
  79. package/packages/coworker-artifacts/src/readme-renderer.test.ts +72 -0
  80. package/packages/coworker-artifacts/src/readme-renderer.ts +56 -0
  81. package/packages/coworker-artifacts/src/resolve-uri.test.ts +46 -0
  82. package/packages/coworker-artifacts/src/resolve-uri.ts +29 -0
  83. package/packages/coworker-artifacts/src/slug.test.ts +47 -0
  84. package/packages/coworker-artifacts/src/slug.ts +31 -0
  85. package/packages/coworker-artifacts/src/types.ts +61 -0
  86. package/packages/coworker-artifacts/tsconfig.json +15 -0
  87. package/packages/coworker-artifacts/tsconfig.publish.json +4 -0
  88. package/packages/coworker-memory/dist/context-injection.d.ts +9 -0
  89. package/packages/coworker-memory/dist/context-injection.js +41 -0
  90. package/packages/coworker-memory/dist/errors.d.ts +25 -0
  91. package/packages/coworker-memory/dist/errors.js +51 -0
  92. package/packages/coworker-memory/dist/index.d.ts +12 -0
  93. package/packages/coworker-memory/dist/index.js +12 -0
  94. package/packages/coworker-memory/dist/layer-a-store.d.ts +16 -0
  95. package/packages/coworker-memory/dist/layer-a-store.js +78 -0
  96. package/packages/coworker-memory/dist/local-sqlite-backend.d.ts +28 -0
  97. package/packages/coworker-memory/dist/local-sqlite-backend.js +167 -0
  98. package/packages/coworker-memory/dist/memory-backend.d.ts +14 -0
  99. package/packages/coworker-memory/dist/memory-backend.js +1 -0
  100. package/packages/coworker-memory/dist/memory-recorder.d.ts +50 -0
  101. package/packages/coworker-memory/dist/memory-recorder.js +69 -0
  102. package/packages/coworker-memory/dist/migrations/001-init.sql +38 -0
  103. package/packages/coworker-memory/dist/migrations/002-artifact-kind.sql +50 -0
  104. package/packages/coworker-memory/dist/paste-detector.d.ts +5 -0
  105. package/packages/coworker-memory/dist/paste-detector.js +14 -0
  106. package/packages/coworker-memory/dist/persona-seed.d.ts +10 -0
  107. package/packages/coworker-memory/dist/persona-seed.js +38 -0
  108. package/packages/coworker-memory/dist/recall-formatter.d.ts +2 -0
  109. package/packages/coworker-memory/dist/recall-formatter.js +14 -0
  110. package/packages/coworker-memory/dist/scope-resolver.d.ts +9 -0
  111. package/packages/coworker-memory/dist/scope-resolver.js +10 -0
  112. package/packages/coworker-memory/dist/types.d.ts +51 -0
  113. package/packages/coworker-memory/dist/types.js +2 -0
  114. package/packages/coworker-memory/dist/workspace-id.d.ts +3 -0
  115. package/packages/coworker-memory/dist/workspace-id.js +54 -0
  116. package/packages/coworker-memory/package.json +35 -0
  117. package/packages/coworker-memory/src/activator-integration.test.ts +141 -0
  118. package/packages/coworker-memory/src/context-injection.test.ts +72 -0
  119. package/packages/coworker-memory/src/context-injection.ts +57 -0
  120. package/packages/coworker-memory/src/errors.test.ts +45 -0
  121. package/packages/coworker-memory/src/errors.ts +42 -0
  122. package/packages/coworker-memory/src/index.test.ts +21 -0
  123. package/packages/coworker-memory/src/index.ts +12 -0
  124. package/packages/coworker-memory/src/layer-a-store.test.ts +85 -0
  125. package/packages/coworker-memory/src/layer-a-store.ts +88 -0
  126. package/packages/coworker-memory/src/local-sqlite-backend.test.ts +110 -0
  127. package/packages/coworker-memory/src/local-sqlite-backend.ts +185 -0
  128. package/packages/coworker-memory/src/memory-backend.ts +10 -0
  129. package/packages/coworker-memory/src/memory-integration.test.ts +89 -0
  130. package/packages/coworker-memory/src/memory-recorder.test.ts +101 -0
  131. package/packages/coworker-memory/src/memory-recorder.ts +95 -0
  132. package/packages/coworker-memory/src/migrations/001-init.sql +38 -0
  133. package/packages/coworker-memory/src/migrations/002-artifact-kind.sql +50 -0
  134. package/packages/coworker-memory/src/paste-detector.test.ts +23 -0
  135. package/packages/coworker-memory/src/paste-detector.ts +18 -0
  136. package/packages/coworker-memory/src/persona-seed.test.ts +57 -0
  137. package/packages/coworker-memory/src/persona-seed.ts +46 -0
  138. package/packages/coworker-memory/src/recall-formatter.test.ts +34 -0
  139. package/packages/coworker-memory/src/recall-formatter.ts +15 -0
  140. package/packages/coworker-memory/src/scope-resolver.test.ts +23 -0
  141. package/packages/coworker-memory/src/scope-resolver.ts +18 -0
  142. package/packages/coworker-memory/src/types.ts +61 -0
  143. package/packages/coworker-memory/src/workspace-id.test.ts +48 -0
  144. package/packages/coworker-memory/src/workspace-id.ts +56 -0
  145. package/packages/coworker-memory/tsconfig.json +15 -0
  146. package/packages/coworker-memory/tsconfig.publish.json +4 -0
  147. package/packages/coworker-persona/dist/commands.d.ts +7 -0
  148. package/packages/coworker-persona/dist/commands.js +35 -0
  149. package/packages/coworker-persona/dist/defaults/manifest.yaml +12 -0
  150. package/packages/coworker-persona/dist/defaults/steering/identity.md +3 -0
  151. package/packages/coworker-persona/dist/index.d.ts +3 -0
  152. package/packages/coworker-persona/dist/index.js +3 -0
  153. package/packages/coworker-persona/dist/manifest.d.ts +24 -0
  154. package/packages/coworker-persona/dist/manifest.js +21 -0
  155. package/packages/coworker-persona/dist/registry.d.ts +22 -0
  156. package/packages/coworker-persona/dist/registry.js +142 -0
  157. package/packages/coworker-persona/package.json +28 -0
  158. package/packages/coworker-persona/scripts/copy-defaults.cjs +17 -0
  159. package/packages/coworker-persona/src/commands.ts +47 -0
  160. package/packages/coworker-persona/src/defaults/manifest.yaml +12 -0
  161. package/packages/coworker-persona/src/defaults/steering/identity.md +3 -0
  162. package/packages/coworker-persona/src/index.ts +3 -0
  163. package/packages/coworker-persona/src/manifest.test.ts +67 -0
  164. package/packages/coworker-persona/src/manifest.ts +49 -0
  165. package/packages/coworker-persona/src/registry.test.ts +89 -0
  166. package/packages/coworker-persona/src/registry.ts +147 -0
  167. package/packages/coworker-persona/tsconfig.json +15 -0
  168. package/packages/coworker-persona/tsconfig.publish.json +4 -0
  169. package/packages/coworker-scratchpad/dist/cell-archive.d.ts +39 -0
  170. package/packages/coworker-scratchpad/dist/cell-archive.js +77 -0
  171. package/packages/coworker-scratchpad/dist/cell-tree.d.ts +14 -0
  172. package/packages/coworker-scratchpad/dist/cell-tree.js +72 -0
  173. package/packages/coworker-scratchpad/dist/child-process-runtime.d.ts +129 -0
  174. package/packages/coworker-scratchpad/dist/child-process-runtime.js +427 -0
  175. package/packages/coworker-scratchpad/dist/collector-registry.d.ts +12 -0
  176. package/packages/coworker-scratchpad/dist/collector-registry.js +29 -0
  177. package/packages/coworker-scratchpad/dist/detect-kind.d.ts +3 -0
  178. package/packages/coworker-scratchpad/dist/detect-kind.js +19 -0
  179. package/packages/coworker-scratchpad/dist/file-collector.d.ts +15 -0
  180. package/packages/coworker-scratchpad/dist/file-collector.js +99 -0
  181. package/packages/coworker-scratchpad/dist/index.d.ts +13 -0
  182. package/packages/coworker-scratchpad/dist/index.js +13 -0
  183. package/packages/coworker-scratchpad/dist/kernel-bindings.d.ts +49 -0
  184. package/packages/coworker-scratchpad/dist/kernel-bindings.js +220 -0
  185. package/packages/coworker-scratchpad/dist/kernel-entry.d.ts +1 -0
  186. package/packages/coworker-scratchpad/dist/kernel-entry.js +355 -0
  187. package/packages/coworker-scratchpad/dist/kernel-protocol.d.ts +171 -0
  188. package/packages/coworker-scratchpad/dist/kernel-protocol.js +48 -0
  189. package/packages/coworker-scratchpad/dist/kernel-spawn.d.ts +3 -0
  190. package/packages/coworker-scratchpad/dist/kernel-spawn.js +54 -0
  191. package/packages/coworker-scratchpad/dist/namespace-codec.d.ts +22 -0
  192. package/packages/coworker-scratchpad/dist/namespace-codec.js +61 -0
  193. package/packages/coworker-scratchpad/dist/scratchpad-lock.d.ts +24 -0
  194. package/packages/coworker-scratchpad/dist/scratchpad-lock.js +86 -0
  195. package/packages/coworker-scratchpad/dist/scratchpad-manager.d.ts +193 -0
  196. package/packages/coworker-scratchpad/dist/scratchpad-manager.js +866 -0
  197. package/packages/coworker-scratchpad/dist/staleness-banner.d.ts +12 -0
  198. package/packages/coworker-scratchpad/dist/staleness-banner.js +27 -0
  199. package/packages/coworker-scratchpad/package.json +31 -0
  200. package/packages/coworker-scratchpad/src/cell-archive.test.ts +150 -0
  201. package/packages/coworker-scratchpad/src/cell-archive.ts +97 -0
  202. package/packages/coworker-scratchpad/src/cell-tree.test.ts +105 -0
  203. package/packages/coworker-scratchpad/src/cell-tree.ts +90 -0
  204. package/packages/coworker-scratchpad/src/child-process-runtime.test.ts +413 -0
  205. package/packages/coworker-scratchpad/src/child-process-runtime.ts +493 -0
  206. package/packages/coworker-scratchpad/src/collector-registry.test.ts +69 -0
  207. package/packages/coworker-scratchpad/src/collector-registry.ts +33 -0
  208. package/packages/coworker-scratchpad/src/detect-kind.test.ts +33 -0
  209. package/packages/coworker-scratchpad/src/detect-kind.ts +22 -0
  210. package/packages/coworker-scratchpad/src/file-collector.test.ts +109 -0
  211. package/packages/coworker-scratchpad/src/file-collector.ts +114 -0
  212. package/packages/coworker-scratchpad/src/index.ts +74 -0
  213. package/packages/coworker-scratchpad/src/kernel-bindings.test.ts +188 -0
  214. package/packages/coworker-scratchpad/src/kernel-bindings.ts +279 -0
  215. package/packages/coworker-scratchpad/src/kernel-entry.test.ts +123 -0
  216. package/packages/coworker-scratchpad/src/kernel-entry.ts +390 -0
  217. package/packages/coworker-scratchpad/src/kernel-protocol.test.ts +105 -0
  218. package/packages/coworker-scratchpad/src/kernel-protocol.ts +230 -0
  219. package/packages/coworker-scratchpad/src/kernel-spawn.test.ts +60 -0
  220. package/packages/coworker-scratchpad/src/kernel-spawn.ts +54 -0
  221. package/packages/coworker-scratchpad/src/namespace-codec.test.ts +102 -0
  222. package/packages/coworker-scratchpad/src/namespace-codec.ts +90 -0
  223. package/packages/coworker-scratchpad/src/scratchpad-lock.test.ts +98 -0
  224. package/packages/coworker-scratchpad/src/scratchpad-lock.ts +102 -0
  225. package/packages/coworker-scratchpad/src/scratchpad-manager.test.ts +1343 -0
  226. package/packages/coworker-scratchpad/src/scratchpad-manager.ts +891 -0
  227. package/packages/coworker-scratchpad/src/staleness-banner.test.ts +53 -0
  228. package/packages/coworker-scratchpad/src/staleness-banner.ts +33 -0
  229. package/packages/coworker-scratchpad/src/vault-integration.test.ts +221 -0
  230. package/packages/coworker-scratchpad/tsconfig.json +15 -0
  231. package/packages/coworker-scratchpad/tsconfig.publish.json +4 -0
  232. package/packages/coworker-types/dist/artifacts.d.ts +31 -0
  233. package/packages/coworker-types/dist/artifacts.js +2 -0
  234. package/packages/coworker-types/dist/contracts.d.ts +32 -0
  235. package/packages/coworker-types/dist/contracts.js +1 -0
  236. package/packages/coworker-types/dist/index.d.ts +5 -0
  237. package/packages/coworker-types/dist/index.js +5 -0
  238. package/packages/coworker-types/dist/memory.d.ts +61 -0
  239. package/packages/coworker-types/dist/memory.js +3 -0
  240. package/packages/coworker-types/dist/scratchpad.d.ts +43 -0
  241. package/packages/coworker-types/dist/scratchpad.js +2 -0
  242. package/packages/coworker-types/dist/vault.d.ts +34 -0
  243. package/packages/coworker-types/dist/vault.js +2 -0
  244. package/packages/coworker-types/package.json +24 -0
  245. package/packages/coworker-types/src/artifacts.test.ts +52 -0
  246. package/packages/coworker-types/src/artifacts.ts +35 -0
  247. package/packages/coworker-types/src/contracts.test.ts +43 -0
  248. package/packages/coworker-types/src/contracts.ts +36 -0
  249. package/packages/coworker-types/src/index.ts +5 -0
  250. package/packages/coworker-types/src/memory.test.ts +50 -0
  251. package/packages/coworker-types/src/memory.ts +79 -0
  252. package/packages/coworker-types/src/scratchpad.test.ts +46 -0
  253. package/packages/coworker-types/src/scratchpad.ts +51 -0
  254. package/packages/coworker-types/src/smoke.test.ts +34 -0
  255. package/packages/coworker-types/src/vault.test.ts +49 -0
  256. package/packages/coworker-types/src/vault.ts +40 -0
  257. package/packages/coworker-types/tsconfig.json +15 -0
  258. package/packages/coworker-types/tsconfig.publish.json +4 -0
  259. package/packages/coworker-utils/dist/audit-log.d.ts +34 -0
  260. package/packages/coworker-utils/dist/audit-log.js +88 -0
  261. package/packages/coworker-utils/dist/index.d.ts +6 -0
  262. package/packages/coworker-utils/dist/index.js +6 -0
  263. package/packages/coworker-utils/dist/lease.d.ts +7 -0
  264. package/packages/coworker-utils/dist/lease.js +67 -0
  265. package/packages/coworker-utils/dist/logger.d.ts +13 -0
  266. package/packages/coworker-utils/dist/logger.js +26 -0
  267. package/packages/coworker-utils/dist/migration-runner.d.ts +7 -0
  268. package/packages/coworker-utils/dist/migration-runner.js +36 -0
  269. package/packages/coworker-utils/dist/ndjson-channel.d.ts +3 -0
  270. package/packages/coworker-utils/dist/ndjson-channel.js +38 -0
  271. package/packages/coworker-utils/dist/secret-scanner.d.ts +10 -0
  272. package/packages/coworker-utils/dist/secret-scanner.js +42 -0
  273. package/packages/coworker-utils/package.json +24 -0
  274. package/packages/coworker-utils/src/audit-log.test.ts +140 -0
  275. package/packages/coworker-utils/src/audit-log.ts +107 -0
  276. package/packages/coworker-utils/src/index.ts +6 -0
  277. package/packages/coworker-utils/src/lease.test.ts +64 -0
  278. package/packages/coworker-utils/src/lease.ts +76 -0
  279. package/packages/coworker-utils/src/logger.test.ts +50 -0
  280. package/packages/coworker-utils/src/logger.ts +45 -0
  281. package/packages/coworker-utils/src/migration-runner.test.ts +65 -0
  282. package/packages/coworker-utils/src/migration-runner.ts +50 -0
  283. package/packages/coworker-utils/src/ndjson-channel.test.ts +76 -0
  284. package/packages/coworker-utils/src/ndjson-channel.ts +41 -0
  285. package/packages/coworker-utils/src/secret-scanner.test.ts +61 -0
  286. package/packages/coworker-utils/src/secret-scanner.ts +56 -0
  287. package/packages/coworker-utils/tsconfig.json +15 -0
  288. package/packages/coworker-utils/tsconfig.publish.json +4 -0
  289. package/packages/coworker-vault/dist/data-vault.d.ts +41 -0
  290. package/packages/coworker-vault/dist/data-vault.js +223 -0
  291. package/packages/coworker-vault/dist/engine-registry.d.ts +34 -0
  292. package/packages/coworker-vault/dist/engine-registry.js +90 -0
  293. package/packages/coworker-vault/dist/engines/jira.yaml +17 -0
  294. package/packages/coworker-vault/dist/errors.d.ts +28 -0
  295. package/packages/coworker-vault/dist/errors.js +57 -0
  296. package/packages/coworker-vault/dist/index.d.ts +6 -0
  297. package/packages/coworker-vault/dist/index.js +6 -0
  298. package/packages/coworker-vault/dist/injector.d.ts +19 -0
  299. package/packages/coworker-vault/dist/injector.js +77 -0
  300. package/packages/coworker-vault/dist/types.d.ts +28 -0
  301. package/packages/coworker-vault/dist/types.js +1 -0
  302. package/packages/coworker-vault/dist/vault-keep.d.ts +4 -0
  303. package/packages/coworker-vault/dist/vault-keep.js +21 -0
  304. package/packages/coworker-vault/package.json +29 -0
  305. package/packages/coworker-vault/src/data-vault.test.ts +199 -0
  306. package/packages/coworker-vault/src/data-vault.ts +257 -0
  307. package/packages/coworker-vault/src/engine-registry.test.ts +120 -0
  308. package/packages/coworker-vault/src/engine-registry.ts +107 -0
  309. package/packages/coworker-vault/src/engines/jira.yaml +17 -0
  310. package/packages/coworker-vault/src/errors.test.ts +58 -0
  311. package/packages/coworker-vault/src/errors.ts +50 -0
  312. package/packages/coworker-vault/src/index.test.ts +24 -0
  313. package/packages/coworker-vault/src/index.ts +6 -0
  314. package/packages/coworker-vault/src/injector.test.ts +109 -0
  315. package/packages/coworker-vault/src/injector.ts +98 -0
  316. package/packages/coworker-vault/src/types.ts +33 -0
  317. package/packages/coworker-vault/src/vault-keep.test.ts +49 -0
  318. package/packages/coworker-vault/src/vault-keep.ts +31 -0
  319. package/packages/coworker-vault/tsconfig.json +15 -0
  320. package/packages/coworker-vault/tsconfig.publish.json +4 -0
  321. package/packages/daemon/package.json +3 -3
  322. package/packages/mcp-server/package.json +3 -3
  323. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  324. package/packages/native/package.json +1 -1
  325. package/packages/native/tsconfig.tsbuildinfo +1 -1
  326. package/packages/pi-agent-core/package.json +1 -1
  327. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  328. package/packages/pi-ai/package.json +1 -1
  329. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  330. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +6 -1
  331. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  332. package/packages/pi-coding-agent/dist/core/extensions/runner.js +22 -3
  333. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  334. package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js +11 -0
  335. package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js.map +1 -1
  336. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts +47 -0
  337. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts.map +1 -0
  338. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js +107 -0
  339. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js.map +1 -0
  340. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts +19 -0
  341. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts.map +1 -0
  342. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js +121 -0
  343. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js.map +1 -0
  344. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  345. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +17 -1
  346. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
  347. package/packages/pi-coding-agent/package.json +2 -2
  348. package/packages/pi-coding-agent/src/core/extensions/runner.ts +22 -3
  349. package/packages/pi-coding-agent/src/core/resolve-config-value.test.ts +11 -0
  350. package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.regression.test.ts +129 -0
  351. package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.ts +117 -0
  352. package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +18 -1
  353. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  354. package/packages/pi-tui/package.json +1 -1
  355. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  356. package/packages/rpc-client/package.json +2 -2
  357. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  358. package/pkg/package.json +1 -1
  359. package/scripts/install.js +6 -5
  360. package/src/resources/extensions/coworker-artifacts/artifacts-command.test.ts +54 -0
  361. package/src/resources/extensions/coworker-artifacts/artifacts-command.ts +43 -0
  362. package/src/resources/extensions/coworker-artifacts/artifacts-singleton.test.ts +25 -0
  363. package/src/resources/extensions/coworker-artifacts/artifacts-singleton.ts +29 -0
  364. package/src/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
  365. package/src/resources/extensions/coworker-artifacts/index.test.ts +46 -0
  366. package/src/resources/extensions/coworker-artifacts/index.ts +154 -0
  367. package/src/resources/extensions/coworker-artifacts/list-tool.test.ts +29 -0
  368. package/src/resources/extensions/coworker-artifacts/list-tool.ts +53 -0
  369. package/src/resources/extensions/coworker-artifacts/open-tool.test.ts +30 -0
  370. package/src/resources/extensions/coworker-artifacts/open-tool.ts +43 -0
  371. package/src/resources/extensions/coworker-memory/extension-manifest.json +13 -0
  372. package/src/resources/extensions/coworker-memory/index.test.ts +137 -0
  373. package/src/resources/extensions/coworker-memory/index.ts +257 -0
  374. package/src/resources/extensions/coworker-memory/memorize-tool.test.ts +41 -0
  375. package/src/resources/extensions/coworker-memory/memorize-tool.ts +20 -0
  376. package/src/resources/extensions/coworker-memory/memory-command.test.ts +134 -0
  377. package/src/resources/extensions/coworker-memory/memory-command.ts +131 -0
  378. package/src/resources/extensions/coworker-memory/memory-singleton.test.ts +41 -0
  379. package/src/resources/extensions/coworker-memory/memory-singleton.ts +89 -0
  380. package/src/resources/extensions/coworker-memory/recall-tool.test.ts +50 -0
  381. package/src/resources/extensions/coworker-memory/recall-tool.ts +35 -0
  382. package/src/resources/extensions/coworker-memory/session-hooks.test.ts +77 -0
  383. package/src/resources/extensions/coworker-memory/session-hooks.ts +61 -0
  384. package/src/resources/extensions/coworker-scratchpad/attach-banners.test.ts +124 -0
  385. package/src/resources/extensions/coworker-scratchpad/attach-banners.ts +67 -0
  386. package/src/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
  387. package/src/resources/extensions/coworker-scratchpad/format-age.test.ts +30 -0
  388. package/src/resources/extensions/coworker-scratchpad/format-age.ts +6 -0
  389. package/src/resources/extensions/coworker-scratchpad/helpers.test.ts +93 -0
  390. package/src/resources/extensions/coworker-scratchpad/helpers.ts +42 -0
  391. package/src/resources/extensions/coworker-scratchpad/index.test.ts +514 -0
  392. package/src/resources/extensions/coworker-scratchpad/index.ts +207 -0
  393. package/src/resources/extensions/coworker-scratchpad/mime-bundle.test.ts +61 -0
  394. package/src/resources/extensions/coworker-scratchpad/mime-bundle.ts +23 -0
  395. package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.test.ts +137 -0
  396. package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.ts +165 -0
  397. package/src/resources/extensions/coworker-scratchpad/session-sidecar.test.ts +133 -0
  398. package/src/resources/extensions/coworker-scratchpad/session-sidecar.ts +68 -0
  399. package/src/resources/extensions/coworker-scratchpad/sp-command.test.ts +836 -0
  400. package/src/resources/extensions/coworker-scratchpad/sp-command.ts +602 -0
  401. package/src/resources/extensions/coworker-scratchpad/workspace-pointer.test.ts +74 -0
  402. package/src/resources/extensions/coworker-scratchpad/workspace-pointer.ts +55 -0
  403. package/src/resources/extensions/coworker-scratchpad/workspace-root.test.ts +51 -0
  404. package/src/resources/extensions/coworker-scratchpad/workspace-root.ts +16 -0
  405. package/src/resources/extensions/coworker-vault/audit-command.test.ts +109 -0
  406. package/src/resources/extensions/coworker-vault/audit-command.ts +56 -0
  407. package/src/resources/extensions/coworker-vault/connect-command.test.ts +103 -0
  408. package/src/resources/extensions/coworker-vault/connect-command.ts +69 -0
  409. package/src/resources/extensions/coworker-vault/datasource-command.test.ts +80 -0
  410. package/src/resources/extensions/coworker-vault/datasource-command.ts +81 -0
  411. package/src/resources/extensions/coworker-vault/extension-manifest.json +12 -0
  412. package/src/resources/extensions/coworker-vault/index.test.ts +82 -0
  413. package/src/resources/extensions/coworker-vault/index.ts +181 -0
  414. package/src/resources/extensions/coworker-vault/test-helpers.ts +120 -0
  415. package/src/resources/extensions/coworker-vault/vault-singleton.test.ts +27 -0
  416. package/src/resources/extensions/coworker-vault/vault-singleton.ts +40 -0
  417. package/src/resources/extensions/otto/commands/release-notes/_data.ts +97 -0
  418. package/src/resources/extensions/otto/commands/release-notes/command.ts +16 -3
  419. package/src/resources/extensions/otto/index.ts +29 -6
  420. package/src/resources/extensions/shared/coworker-paths.test.ts +40 -0
  421. package/src/resources/extensions/shared/coworker-paths.ts +10 -0
  422. package/src/resources/extensions/slash-commands/{audit.ts → audit-codebase.ts} +4 -4
  423. package/src/resources/extensions/slash-commands/extension-manifest.json +1 -1
  424. package/src/resources/extensions/slash-commands/index.ts +2 -2
  425. package/src/resources/extensions/subagent/index.ts +9 -0
  426. package/src/resources/extensions/subagent/launch.test.ts +97 -0
  427. package/src/resources/extensions/subagent/launch.ts +42 -5
  428. package/src/resources/extensions/subagent/run-store.ts +3 -1
  429. package/src/resources/extensions/workflow/bootstrap/register-extension.ts +2 -0
  430. package/src/resources/extensions/workflow/bootstrap/register-hooks.ts +10 -0
  431. package/src/resources/extensions/workflow/health-widget-core.ts +1 -1
  432. package/src/resources/extensions/workflow/persona-status.ts +109 -0
  433. package/src/resources/extensions/workflow/tests/auto-recovery.test.ts +34 -0
@@ -0,0 +1,131 @@
1
+ // src/resources/extensions/coworker-memory/memory-command.ts
2
+ import type { MemoryBundle } from './memory-singleton.js';
3
+ import { runMemorize } from './memorize-tool.js';
4
+ import { runRecall } from './recall-tool.js';
5
+
6
+ export interface MemoryCommandResult {
7
+ message: string;
8
+ }
9
+
10
+ export async function runMemoryCommand(bundle: MemoryBundle, argv: string[]): Promise<MemoryCommandResult> {
11
+ const [sub, ...rest] = argv;
12
+ switch (sub) {
13
+ case 'note': {
14
+ const text = rest.join(' ').trim();
15
+ if (!text) throw new Error('Usage: /memory note <text>');
16
+ await runMemorize(bundle, { text, kind: 'lesson', scope: 'workspace' });
17
+ return { message: `lesson stored in workspace.` };
18
+ }
19
+ case 'status': {
20
+ const status = await bundle.backend.status();
21
+ return {
22
+ message: [
23
+ `scope_mode: ${bundle.scopeMode}`,
24
+ `workspace_wing: ${bundle.workspaceWing}`,
25
+ `drawer_count: ${status.drawer_count}`,
26
+ `layer_b_db_path: ${status.layer_b_db_path}`,
27
+ `schema_version: ${status.schema_version}`,
28
+ ].join('\n'),
29
+ };
30
+ }
31
+ case 'clear': {
32
+ let wing: string | undefined;
33
+ let confirm = false;
34
+ for (let i = 0; i < rest.length; i++) {
35
+ if (rest[i] === '--wing' && rest[i+1]) { wing = rest[++i]; }
36
+ else if (rest[i] === '--confirm') { confirm = true; }
37
+ }
38
+ if (!confirm) throw new Error('Usage: /memory clear --wing <wing> --confirm');
39
+ const out = await bundle.backend.clear({ wing, confirm: true });
40
+ return { message: `deleted: ${out.deleted}` };
41
+ }
42
+ case 'wing':
43
+ case 'room': {
44
+ // Session overrides — caller manages state; this is a placeholder return acknowledging.
45
+ const target = rest.join(' ').trim();
46
+ if (!target) throw new Error(`Usage: /memory ${sub} <name>`);
47
+ return { message: `${sub} override: ${target}` };
48
+ }
49
+ case 'seed': {
50
+ // Caller wires this to applyPersonaSeed via session-hooks; here we just acknowledge the request.
51
+ return { message: 're-seed will run on next session_start; flip workspace.json.memory_seed_applied=false and reattach.' };
52
+ }
53
+ case 'show': {
54
+ let target: 'all' | 'profile' | 'rule' | 'lesson' = 'all';
55
+ let scopeOverride: 'workspace' | 'global' | null = null;
56
+ for (let i = 0; i < rest.length; i++) {
57
+ const t = rest[i];
58
+ if (t === '--scope' && rest[i + 1]) {
59
+ const s = rest[++i];
60
+ if (s !== 'workspace' && s !== 'global') throw new Error(`Usage: /memory show [profile|rules|lessons] [--scope workspace|global]`);
61
+ scopeOverride = s;
62
+ } else if (t === 'profile') { target = 'profile'; }
63
+ else if (t === 'rule' || t === 'rules') { target = 'rule'; }
64
+ else if (t === 'lesson' || t === 'lessons') { target = 'lesson'; }
65
+ else if (t === 'all') { target = 'all'; }
66
+ else { throw new Error(`Usage: /memory show [profile|rules|lessons] [--scope workspace|global]`); }
67
+ }
68
+ const kinds: Array<'profile' | 'rule' | 'lesson'> =
69
+ target === 'all' ? ['profile', 'rule', 'lesson'] : [target];
70
+ // Determine scopes: --scope override wins; otherwise derive from bundle.scopeMode.
71
+ const scopes: Array<{ name: 'workspace' | 'global'; store: typeof bundle.workspaceLayerA }> = [];
72
+ if (scopeOverride === 'workspace') {
73
+ scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
74
+ } else if (scopeOverride === 'global') {
75
+ scopes.push({ name: 'global', store: bundle.globalLayerA });
76
+ } else if (bundle.scopeMode === 'global') {
77
+ scopes.push({ name: 'global', store: bundle.globalLayerA });
78
+ } else if (bundle.scopeMode === 'per-project') {
79
+ scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
80
+ } else {
81
+ // per-project-tagged: workspace first, then global
82
+ scopes.push({ name: 'workspace', store: bundle.workspaceLayerA });
83
+ scopes.push({ name: 'global', store: bundle.globalLayerA });
84
+ }
85
+ const TITLE_FOR = { profile: 'Profile', rule: 'Rules', lesson: 'Lessons' } as const;
86
+ const sections: string[] = [];
87
+ for (const k of kinds) {
88
+ for (const scope of scopes) {
89
+ const body = await scope.store.read(k);
90
+ sections.push(`## ${TITLE_FOR[k]} (${scope.name})\n${body || '(none)'}`);
91
+ }
92
+ }
93
+ return { message: sections.join('\n\n') };
94
+ }
95
+ case 'recall': {
96
+ // First flag/positional split: gather positional tokens for the query;
97
+ // recognize --kind, --room, --wing, --limit, --days_back.
98
+ const flags: { kind?: string; room?: string; wing?: string; limit?: number; days_back?: number } = {};
99
+ const queryTokens: string[] = [];
100
+ for (let i = 0; i < rest.length; i++) {
101
+ const t = rest[i];
102
+ if (t === '--kind' && rest[i + 1]) { flags.kind = rest[++i]; }
103
+ else if (t === '--room' && rest[i + 1]) { flags.room = rest[++i]; }
104
+ else if (t === '--wing' && rest[i + 1]) { flags.wing = rest[++i]; }
105
+ else if (t === '--limit' && rest[i + 1]) {
106
+ const n = parseInt(rest[++i]!, 10);
107
+ if (Number.isFinite(n)) flags.limit = n;
108
+ } else if (t === '--days_back' && rest[i + 1]) {
109
+ const n = parseInt(rest[++i]!, 10);
110
+ if (Number.isFinite(n)) flags.days_back = n;
111
+ } else {
112
+ queryTokens.push(t!);
113
+ }
114
+ }
115
+ const query = queryTokens.join(' ').trim();
116
+ if (!query) throw new Error('Usage: /memory recall <query> [--kind <k>] [--room <r>] [--wing <w>] [--limit N] [--days_back N]');
117
+ // Defer to the same runRecall the LLM tool uses for parity.
118
+ const out = await runRecall(bundle, {
119
+ query,
120
+ kind: flags.kind as never, // runRecall accepts string; backend validates
121
+ room: flags.room,
122
+ wing: flags.wing,
123
+ max_results: flags.limit,
124
+ days_back: flags.days_back,
125
+ });
126
+ return { message: out.markdown };
127
+ }
128
+ default:
129
+ throw new Error(`Unknown /memory subcommand: ${sub}. Try: note, status, clear, wing, room, seed, show, recall.`);
130
+ }
131
+ }
@@ -0,0 +1,41 @@
1
+ // src/resources/extensions/coworker-memory/memory-singleton.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync, mkdirSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { createMemoryBundle } from './memory-singleton.js';
8
+
9
+ describe('memory singleton bundle', () => {
10
+ it('constructs scope-aware bundle with all stores', async () => {
11
+ const home = mkdtempSync(join(tmpdir(), 'mem-home-'));
12
+ const ws = mkdtempSync(join(tmpdir(), 'mem-ws-'));
13
+ mkdirSync(ws, { recursive: true });
14
+ const bundle = await createMemoryBundle({
15
+ globalDir: home, workspaceDir: ws,
16
+ scopeMode: 'per-project-tagged',
17
+ currentScratchpadName: () => null,
18
+ });
19
+ assert.ok(bundle.globalLayerA);
20
+ assert.ok(bundle.workspaceLayerA);
21
+ assert.ok(bundle.backend);
22
+ assert.ok(bundle.recorder);
23
+ assert.equal(bundle.scopeMode, 'per-project-tagged');
24
+ assert.match(bundle.workspaceWing, /-[0-9a-f]{6}$/);
25
+ assert.equal(bundle.writeWing, bundle.workspaceWing);
26
+ assert.deepEqual(bundle.readWings, [bundle.workspaceWing, 'global']);
27
+ await bundle.dispose();
28
+ });
29
+ it('global mode bundle uses wing global', async () => {
30
+ const home = mkdtempSync(join(tmpdir(), 'mem-home-g-'));
31
+ const ws = mkdtempSync(join(tmpdir(), 'mem-ws-g-'));
32
+ const bundle = await createMemoryBundle({
33
+ globalDir: home, workspaceDir: ws,
34
+ scopeMode: 'global',
35
+ currentScratchpadName: () => null,
36
+ });
37
+ assert.equal(bundle.writeWing, 'global');
38
+ assert.deepEqual(bundle.readWings, ['global']);
39
+ await bundle.dispose();
40
+ });
41
+ });
@@ -0,0 +1,89 @@
1
+ // src/resources/extensions/coworker-memory/memory-singleton.ts
2
+ //
3
+ // Factory that constructs the layered memory collaborators (LayerAStore x2,
4
+ // LocalSqliteBackend, MemoryRecorder) plus shared audit + secret scanner for
5
+ // given roots. Mirrors the coworker-vault singleton pattern. Pure beyond what
6
+ // the underlying constructors already perform (dir creation, db open, etc.).
7
+ import { join } from 'node:path';
8
+ import { AuditLog, SecretScanner } from '@otto/coworker-utils';
9
+ import {
10
+ LayerAStore,
11
+ LocalSqliteBackend,
12
+ MemoryRecorder,
13
+ resolveScope,
14
+ resolveWorkspaceId,
15
+ type ScopeMode,
16
+ type Wing,
17
+ type CurrentScratchpadProvider,
18
+ type WorkspaceIdRecord,
19
+ } from '@otto/coworker-memory';
20
+
21
+ export interface MemoryBundleOptions {
22
+ globalDir: string;
23
+ workspaceDir: string;
24
+ scopeMode: ScopeMode;
25
+ currentScratchpadName: CurrentScratchpadProvider;
26
+ }
27
+
28
+ export interface MemoryBundle {
29
+ globalLayerA: LayerAStore;
30
+ workspaceLayerA: LayerAStore;
31
+ backend: LocalSqliteBackend;
32
+ recorder: MemoryRecorder;
33
+ audit: AuditLog;
34
+ scanner: SecretScanner;
35
+ workspaceWing: Wing;
36
+ writeWing: Wing;
37
+ readWings: Wing[];
38
+ scopeMode: ScopeMode;
39
+ workspaceRecord: WorkspaceIdRecord;
40
+ workspaceDir: string;
41
+ dispose(): Promise<void>;
42
+ }
43
+
44
+ export async function createMemoryBundle(opts: MemoryBundleOptions): Promise<MemoryBundle> {
45
+ const audit = new AuditLog({ path: join(opts.globalDir, 'audit.jsonl') });
46
+ const scanner = new SecretScanner();
47
+ const wsRecord = await resolveWorkspaceId(opts.workspaceDir);
48
+ const scope = resolveScope({ mode: opts.scopeMode, workspaceId: wsRecord.id });
49
+ const globalLayerA = new LayerAStore({
50
+ scopeDir: join(opts.globalDir, 'memory'),
51
+ scope: 'global',
52
+ audit,
53
+ scanner,
54
+ });
55
+ const workspaceLayerA = new LayerAStore({
56
+ scopeDir: join(opts.workspaceDir, '.otto', 'memory'),
57
+ scope: 'workspace',
58
+ audit,
59
+ scanner,
60
+ });
61
+ const backend = new LocalSqliteBackend({
62
+ dbPath: join(opts.workspaceDir, '.otto', 'memory', 'layer-b.db'),
63
+ });
64
+ await backend.open();
65
+ const recorder = new MemoryRecorder({
66
+ backend,
67
+ scanner,
68
+ audit,
69
+ writeWing: scope.writeWing,
70
+ currentScratchpadName: opts.currentScratchpadName,
71
+ });
72
+ return {
73
+ globalLayerA,
74
+ workspaceLayerA,
75
+ backend,
76
+ recorder,
77
+ audit,
78
+ scanner,
79
+ workspaceWing: wsRecord.id,
80
+ writeWing: scope.writeWing,
81
+ readWings: scope.readWings,
82
+ scopeMode: opts.scopeMode,
83
+ workspaceRecord: wsRecord,
84
+ workspaceDir: opts.workspaceDir,
85
+ async dispose() {
86
+ await backend.close();
87
+ },
88
+ };
89
+ }
@@ -0,0 +1,50 @@
1
+ // src/resources/extensions/coworker-memory/recall-tool.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync, mkdirSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { createMemoryBundle } from './memory-singleton.js';
8
+ import { runRecall } from './recall-tool.js';
9
+
10
+ async function setup() {
11
+ const home = mkdtempSync(join(tmpdir(), 'rt-home-'));
12
+ const ws = mkdtempSync(join(tmpdir(), 'rt-ws-'));
13
+ mkdirSync(ws, { recursive: true });
14
+ return createMemoryBundle({
15
+ globalDir: home, workspaceDir: ws,
16
+ scopeMode: 'per-project-tagged', currentScratchpadName: () => null,
17
+ });
18
+ }
19
+
20
+ describe('recall tool', () => {
21
+ it('returns results with markdown rendering', async () => {
22
+ const b = await setup();
23
+ await b.recorder.recordTurn({ sessionId: 's', userText: 'customer paste about load balancer', turnId: 't1' });
24
+ const r = await runRecall(b, { query: 'load balancer' });
25
+ assert.equal(r.results.length, 1);
26
+ assert.match(r.markdown, /Memory recall \(1 matches\)/);
27
+ assert.match(r.markdown, /drawer:\/\//);
28
+ await b.dispose();
29
+ });
30
+ it('honors max_results clamp 1..64', async () => {
31
+ const b = await setup();
32
+ for (let i = 0; i < 100; i++) {
33
+ await b.recorder.recordTurn({ sessionId: 's', userText: `apple ${i}`, turnId: `t${i}` });
34
+ }
35
+ const big = await runRecall(b, { query: 'apple', max_results: 200 });
36
+ assert.ok(big.results.length <= 64);
37
+ const small = await runRecall(b, { query: 'apple', max_results: 0 });
38
+ assert.equal(small.results.length, 1);
39
+ await b.dispose();
40
+ });
41
+ it('filters by kind', async () => {
42
+ const b = await setup();
43
+ await b.recorder.recordTurn({ sessionId: 's', userText: 'short alpha', turnId: 't1' });
44
+ await b.recorder.recordPaste({ sessionId: 's', content: 'long alpha paste', turnId: 't2' });
45
+ const onlyPaste = await runRecall(b, { query: 'alpha', kind: 'paste' });
46
+ assert.equal(onlyPaste.results.length, 1);
47
+ assert.equal(onlyPaste.results[0]!.drawer.kind, 'paste');
48
+ await b.dispose();
49
+ });
50
+ });
@@ -0,0 +1,35 @@
1
+ // src/resources/extensions/coworker-memory/recall-tool.ts
2
+ import type { MemoryBundle } from './memory-singleton.js';
3
+ import { formatRecall, type RecallQuery, type RecallResult } from '@otto/coworker-memory';
4
+
5
+ export interface RecallToolArgs {
6
+ query: string;
7
+ kind?: RecallQuery['kind'];
8
+ wing?: string;
9
+ room?: string;
10
+ days_back?: number;
11
+ max_results?: number;
12
+ }
13
+
14
+ export interface RecallToolOutput {
15
+ results: RecallResult[];
16
+ markdown: string;
17
+ }
18
+
19
+ export async function runRecall(bundle: MemoryBundle, args: RecallToolArgs): Promise<RecallToolOutput> {
20
+ const wings = args.wing ? [args.wing, ...bundle.readWings.filter(w => w !== args.wing)] : bundle.readWings;
21
+ const limit = args.max_results === undefined ? 8 : Math.min(Math.max(args.max_results, 1), 64);
22
+ const results = await bundle.backend.recall({
23
+ query: args.query, wing: wings, room: args.room, kind: args.kind,
24
+ days_back: args.days_back, max_results: limit,
25
+ });
26
+ bundle.audit.append({
27
+ _schema: 1, ts: new Date().toISOString(), producer: 'memory', action: 'recall',
28
+ detail: {
29
+ wing_filter: wings, room_filter: args.room ?? null,
30
+ kind_filter: args.kind ?? null, days_back: args.days_back ?? null,
31
+ result_count: results.length,
32
+ },
33
+ });
34
+ return { results, markdown: formatRecall(results) };
35
+ }
@@ -0,0 +1,77 @@
1
+ // src/resources/extensions/coworker-memory/session-hooks.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync, mkdirSync, writeFileSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { createMemoryBundle } from './memory-singleton.js';
8
+ import { onSessionStart, onSessionShutdown } from './session-hooks.js';
9
+
10
+ async function setup() {
11
+ const home = mkdtempSync(join(tmpdir(), 'sh-home-'));
12
+ const ws = mkdtempSync(join(tmpdir(), 'sh-ws-'));
13
+ mkdirSync(ws, { recursive: true });
14
+ return {
15
+ home,
16
+ ws,
17
+ bundle: await createMemoryBundle({
18
+ globalDir: home,
19
+ workspaceDir: ws,
20
+ scopeMode: 'per-project-tagged',
21
+ currentScratchpadName: () => null,
22
+ }),
23
+ };
24
+ }
25
+
26
+ describe('session hooks', () => {
27
+ it('onSessionStart returns Layer A context block', async () => {
28
+ const c = await setup();
29
+ await c.bundle.workspaceLayerA.append({
30
+ kind: 'lesson',
31
+ text: 'do not deploy on friday',
32
+ source: 'user',
33
+ ts: '2026-06-02T00:00:00Z',
34
+ });
35
+ const out = await onSessionStart(c.bundle, { tokenLimit: 3000 });
36
+ assert.match(out.contextBlock, /Memory \(Layer A\)/);
37
+ assert.match(out.contextBlock, /do not deploy on friday/);
38
+ await c.bundle.dispose();
39
+ });
40
+ it('onSessionStart applies persona seed when pending', async () => {
41
+ const c = await setup();
42
+ const personaDir = mkdtempSync(join(tmpdir(), 'persona-'));
43
+ const seedDir = join(personaDir, 'memory-seed');
44
+ mkdirSync(seedDir, { recursive: true });
45
+ writeFileSync(join(seedDir, 'rules.md'), 'Persona rule baseline');
46
+ const out = await onSessionStart(c.bundle, {
47
+ tokenLimit: 3000,
48
+ persona: { id: 'noc-ops', personaDir },
49
+ });
50
+ assert.deepEqual(out.seed.copied, ['rules.md']);
51
+ assert.equal(c.bundle.workspaceRecord.memory_seed_applied, true);
52
+ assert.equal(c.bundle.workspaceRecord.memory_seed_persona, 'noc-ops');
53
+ await c.bundle.dispose();
54
+ });
55
+ it('onSessionStart does not re-apply seed once flag is true', async () => {
56
+ const c = await setup();
57
+ const personaDir = mkdtempSync(join(tmpdir(), 'persona2-'));
58
+ const seedDir = join(personaDir, 'memory-seed');
59
+ mkdirSync(seedDir, { recursive: true });
60
+ writeFileSync(join(seedDir, 'rules.md'), 'Persona rule v1');
61
+ await onSessionStart(c.bundle, { tokenLimit: 3000, persona: { id: 'noc-ops', personaDir } });
62
+ // Now change the seed file but the flag is set.
63
+ writeFileSync(join(seedDir, 'rules.md'), 'Persona rule v2');
64
+ const second = await onSessionStart(c.bundle, {
65
+ tokenLimit: 3000,
66
+ persona: { id: 'noc-ops', personaDir },
67
+ });
68
+ assert.deepEqual(second.seed.copied, []);
69
+ await c.bundle.dispose();
70
+ });
71
+ it('onSessionShutdown closes backend without throwing', async () => {
72
+ const c = await setup();
73
+ await onSessionShutdown(c.bundle);
74
+ // Bundle is now disposed; second close should be safe.
75
+ await c.bundle.dispose();
76
+ });
77
+ });
@@ -0,0 +1,61 @@
1
+ // src/resources/extensions/coworker-memory/session-hooks.ts
2
+ //
3
+ // Session lifecycle seams for the coworker-memory extension.
4
+ // - onSessionStart: applies a one-shot persona seed (if pending) and returns
5
+ // the Layer A context block ready for system-prompt injection.
6
+ // - onSessionShutdown: closes the backend (Layer B). Safe to double-call —
7
+ // LocalSqliteBackend.close() nulls its db handle so a second close is a no-op.
8
+ import { buildLayerAContext, applyPersonaSeed, writeWorkspaceId } from '@otto/coworker-memory';
9
+ import type { MemoryBundle } from './memory-singleton.js';
10
+
11
+ export interface SessionStartOptions {
12
+ tokenLimit?: number;
13
+ persona?: { id: string; personaDir: string };
14
+ }
15
+
16
+ export interface SessionStartResult {
17
+ contextBlock: string;
18
+ seed: { copied: string[]; blocked: string[] };
19
+ }
20
+
21
+ export async function onSessionStart(
22
+ bundle: MemoryBundle,
23
+ opts: SessionStartOptions = {},
24
+ ): Promise<SessionStartResult> {
25
+ let seed = { copied: [] as string[], blocked: [] as string[] };
26
+ if (opts.persona && !bundle.workspaceRecord.memory_seed_applied) {
27
+ seed = await applyPersonaSeed({
28
+ personaId: opts.persona.id,
29
+ personaDir: opts.persona.personaDir,
30
+ store: bundle.workspaceLayerA,
31
+ });
32
+ if (seed.copied.length > 0 || seed.blocked.length > 0) {
33
+ bundle.workspaceRecord.memory_seed_applied = true;
34
+ bundle.workspaceRecord.memory_seed_persona = opts.persona.id;
35
+ await writeWorkspaceId(bundle.workspaceDir, bundle.workspaceRecord);
36
+ bundle.audit.append({
37
+ _schema: 1,
38
+ ts: new Date().toISOString(),
39
+ producer: 'memory',
40
+ action: 'seed-applied',
41
+ detail: {
42
+ persona_id: opts.persona.id,
43
+ files_copied: seed.copied,
44
+ files_blocked: seed.blocked,
45
+ },
46
+ });
47
+ }
48
+ }
49
+ const contextBlock = await buildLayerAContext({
50
+ mode: bundle.scopeMode,
51
+ globalStore: bundle.globalLayerA,
52
+ workspaceStore: bundle.workspaceLayerA,
53
+ tokenLimit: opts.tokenLimit ?? 3000,
54
+ });
55
+ return { contextBlock, seed };
56
+ }
57
+
58
+ export async function onSessionShutdown(bundle: MemoryBundle): Promise<void> {
59
+ // WAL checkpoint happens on backend close; this is the seam.
60
+ await bundle.dispose();
61
+ }
@@ -0,0 +1,124 @@
1
+ import { describe, it, beforeEach, afterEach } from 'node:test';
2
+ import assert from 'node:assert/strict';
3
+ import { mkdtemp, rm } from 'node:fs/promises';
4
+ import { mkdirSync, writeFileSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { showRecoveryNotesBanner, showDivergenceBanner } from './attach-banners.js';
8
+
9
+ interface FakeUi {
10
+ notifications: Array<[string, string]>;
11
+ notify: (msg: string, level: 'info' | 'warning' | 'error') => void;
12
+ }
13
+
14
+ function makeUi(): FakeUi {
15
+ const notifications: FakeUi['notifications'] = [];
16
+ return { notifications, notify: (m, l) => notifications.push([l, m]) };
17
+ }
18
+
19
+ function writeMeta(root: string, name: string, meta: Record<string, unknown>): void {
20
+ mkdirSync(join(root, name), { recursive: true });
21
+ writeFileSync(join(root, name, 'meta.json'), JSON.stringify(meta));
22
+ }
23
+
24
+ let root: string;
25
+
26
+ describe('attach-banners', () => {
27
+ beforeEach(async () => {
28
+ root = await mkdtemp(join(tmpdir(), 'banners-'));
29
+ });
30
+ afterEach(async () => {
31
+ await rm(root, { recursive: true, force: true });
32
+ });
33
+
34
+ it('showRecoveryNotesBanner emits warning when there are unseen notes; returns markSeen=true', () => {
35
+ writeMeta(root, 'p1', {
36
+ recovery_notes: [
37
+ { kind: 'snapshot-failed', message: 'boom', at: '2026-05-31T10:00:00.000Z' },
38
+ { kind: 'cells-since-snapshot', n: 3, at: '2026-05-31T11:00:00.000Z' },
39
+ ],
40
+ recovery_notes_seen_at: null,
41
+ });
42
+ const ui = makeUi();
43
+ const { unseenCount, markSeen } = showRecoveryNotesBanner('p1', root, ui);
44
+ assert.equal(unseenCount, 2);
45
+ assert.equal(markSeen, true);
46
+ assert.equal(ui.notifications.length, 1);
47
+ assert.equal(ui.notifications[0][0], 'warning');
48
+ assert.match(ui.notifications[0][1], /2 unread recovery notes/);
49
+ assert.match(ui.notifications[0][1], /snapshot-failed: boom/);
50
+ assert.match(ui.notifications[0][1], /3 cells since last snapshot/);
51
+ });
52
+
53
+ it('showRecoveryNotesBanner does not notify when all notes are seen; returns markSeen=false', () => {
54
+ writeMeta(root, 'p1', {
55
+ recovery_notes: [
56
+ { kind: 'snapshot-failed', message: 'boom', at: '2026-05-31T10:00:00.000Z' },
57
+ ],
58
+ recovery_notes_seen_at: '2026-05-31T11:00:00.000Z',
59
+ });
60
+ const ui = makeUi();
61
+ const { unseenCount, markSeen } = showRecoveryNotesBanner('p1', root, ui);
62
+ assert.equal(unseenCount, 0);
63
+ assert.equal(markSeen, false);
64
+ assert.equal(ui.notifications.length, 0);
65
+ });
66
+
67
+ it('showRecoveryNotesBanner truncates to 5 with "+ N more" footer', () => {
68
+ const notes = Array.from({ length: 8 }, (_, i) => ({
69
+ kind: 'snapshot-failed' as const,
70
+ message: `err-${i}`,
71
+ at: `2026-05-31T1${i}:00:00.000Z`,
72
+ }));
73
+ writeMeta(root, 'p1', { recovery_notes: notes, recovery_notes_seen_at: null });
74
+ const ui = makeUi();
75
+ const { unseenCount } = showRecoveryNotesBanner('p1', root, ui);
76
+ assert.equal(unseenCount, 8);
77
+ assert.match(ui.notifications[0][1], /\+ 3 more \(run \/sp notes\)/);
78
+ // Should include err-0..err-4 (first 5) but not err-5..err-7
79
+ assert.match(ui.notifications[0][1], /err-0/);
80
+ assert.match(ui.notifications[0][1], /err-4/);
81
+ assert.equal(ui.notifications[0][1].includes('err-5'), false);
82
+ });
83
+
84
+ it('showRecoveryNotesBanner tolerates missing or corrupt meta silently', () => {
85
+ const ui = makeUi();
86
+ // Missing scratchpad dir
87
+ const r1 = showRecoveryNotesBanner('absent', root, ui);
88
+ assert.deepEqual(r1, { unseenCount: 0, markSeen: false });
89
+ // Corrupt meta.json
90
+ mkdirSync(join(root, 'p2'), { recursive: true });
91
+ writeFileSync(join(root, 'p2', 'meta.json'), '{not json');
92
+ const r2 = showRecoveryNotesBanner('p2', root, ui);
93
+ assert.deepEqual(r2, { unseenCount: 0, markSeen: false });
94
+ assert.equal(ui.notifications.length, 0);
95
+ });
96
+
97
+ it('showDivergenceBanner emits info when leaf !== kernel; both set', () => {
98
+ writeMeta(root, 'p1', { cell_leaf_id: 5, kernel_at_cell_id: 8 });
99
+ const ui = makeUi();
100
+ const { diverged } = showDivergenceBanner('p1', root, ui);
101
+ assert.equal(diverged, true);
102
+ assert.equal(ui.notifications.length, 1);
103
+ assert.equal(ui.notifications[0][0], 'info');
104
+ assert.match(ui.notifications[0][1], /kernel state is at cell #8/);
105
+ assert.match(ui.notifications[0][1], /view is at cell #5/);
106
+ assert.match(ui.notifications[0][1], /\/sp tree to inspect/);
107
+ });
108
+
109
+ it('showDivergenceBanner does not notify when leaf===kernel, either is null, or meta missing', () => {
110
+ const ui = makeUi();
111
+ // Equal
112
+ writeMeta(root, 'p1', { cell_leaf_id: 5, kernel_at_cell_id: 5 });
113
+ assert.deepEqual(showDivergenceBanner('p1', root, ui), { diverged: false });
114
+ // Leaf null
115
+ writeMeta(root, 'p2', { cell_leaf_id: null, kernel_at_cell_id: 5 });
116
+ assert.deepEqual(showDivergenceBanner('p2', root, ui), { diverged: false });
117
+ // Kernel null
118
+ writeMeta(root, 'p3', { cell_leaf_id: 5, kernel_at_cell_id: null });
119
+ assert.deepEqual(showDivergenceBanner('p3', root, ui), { diverged: false });
120
+ // Meta missing
121
+ assert.deepEqual(showDivergenceBanner('absent', root, ui), { diverged: false });
122
+ assert.equal(ui.notifications.length, 0);
123
+ });
124
+ });
@@ -0,0 +1,67 @@
1
+ import { existsSync, readFileSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ import type { RecoveryNote } from '@otto/coworker-scratchpad';
4
+
5
+ type RecoveryNoteEntry = RecoveryNote & { at: string };
6
+
7
+ interface UiNotify {
8
+ notify: (msg: string, level: 'info' | 'warning' | 'error') => void;
9
+ }
10
+
11
+ export function showRecoveryNotesBanner(
12
+ name: string,
13
+ rootDir: string,
14
+ ui: UiNotify,
15
+ ): { unseenCount: number; markSeen: boolean } {
16
+ const metaPath = join(rootDir, name, 'meta.json');
17
+ if (!existsSync(metaPath)) return { unseenCount: 0, markSeen: false };
18
+ let meta: Record<string, unknown>;
19
+ try {
20
+ meta = JSON.parse(readFileSync(metaPath, 'utf8')) as Record<string, unknown>;
21
+ } catch {
22
+ return { unseenCount: 0, markSeen: false };
23
+ }
24
+ const notes = Array.isArray(meta.recovery_notes) ? (meta.recovery_notes as RecoveryNoteEntry[]) : [];
25
+ if (notes.length === 0) return { unseenCount: 0, markSeen: false };
26
+ const seenAt = typeof meta.recovery_notes_seen_at === 'string' ? meta.recovery_notes_seen_at : null;
27
+ const unseen = notes.filter((n) => seenAt === null || n.at > seenAt);
28
+ if (unseen.length === 0) return { unseenCount: 0, markSeen: false };
29
+ const head = unseen.slice(0, 5).map(formatNoteLine).join('\n');
30
+ const tail = unseen.length > 5 ? `\n+ ${unseen.length - 5} more (run /sp notes)` : '';
31
+ ui.notify(`⚠ ${unseen.length} unread recovery notes:\n${head}${tail}`, 'warning');
32
+ return { unseenCount: unseen.length, markSeen: true };
33
+ }
34
+
35
+ export function showDivergenceBanner(
36
+ name: string,
37
+ rootDir: string,
38
+ ui: UiNotify,
39
+ ): { diverged: boolean } {
40
+ const metaPath = join(rootDir, name, 'meta.json');
41
+ if (!existsSync(metaPath)) return { diverged: false };
42
+ let meta: Record<string, unknown>;
43
+ try {
44
+ meta = JSON.parse(readFileSync(metaPath, 'utf8')) as Record<string, unknown>;
45
+ } catch {
46
+ return { diverged: false };
47
+ }
48
+ const leaf = typeof meta.cell_leaf_id === 'number' ? meta.cell_leaf_id : null;
49
+ const kernel = typeof meta.kernel_at_cell_id === 'number' ? meta.kernel_at_cell_id : null;
50
+ if (leaf === null || kernel === null || leaf === kernel) return { diverged: false };
51
+ ui.notify(
52
+ `ℹ kernel state is at cell #${kernel}; view is at cell #${leaf} (run /sp tree to inspect)`,
53
+ 'info',
54
+ );
55
+ return { diverged: true };
56
+ }
57
+
58
+ export function formatNoteLine(n: RecoveryNoteEntry): string {
59
+ const ts = n.at.slice(0, 19);
60
+ switch (n.kind) {
61
+ case 'snapshot-failed': return ` • [${ts}] snapshot-failed: ${n.message}`;
62
+ case 'cells-since-snapshot': return ` • [${ts}] ${n.n} cells since last snapshot`;
63
+ case 'namespace-corrupt': return ` • [${ts}] namespace-corrupt: ${n.message}`;
64
+ case 'namespace-absent': return ` • [${ts}] namespace-absent`;
65
+ default: return ` • [${ts}] ${(n as { kind: string }).kind}`;
66
+ }
67
+ }