@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,514 @@
1
+ import { describe, it, beforeEach, afterEach } from 'node:test';
2
+ import assert from 'node:assert/strict';
3
+ import { mkdtemp, mkdir, rm } from 'node:fs/promises';
4
+ import { mkdtempSync, rmSync, mkdirSync, writeFileSync, existsSync } from 'node:fs';
5
+ import { execSync } from 'node:child_process';
6
+ import { tmpdir } from 'node:os';
7
+ import { join } from 'node:path';
8
+ import coworkerScratchpadExtension, { tryRestoreCurrentName } from './index.js';
9
+ import { sessionSidecarPath, writeSessionSidecar } from './session-sidecar.js';
10
+ import { workspaceHash, workspacePointerPath, writeWorkspacePointer } from './workspace-pointer.js';
11
+ import { detectWorkspaceRoot } from './workspace-root.js';
12
+ import { readFileSync } from 'node:fs';
13
+ import { makeFakeApi, fireSessionStart } from '../coworker-vault/test-helpers.js';
14
+
15
+ // Minimal pi.ExtensionAPI stub — captures registrations and lets us fire session_start/session_shutdown.
16
+ interface StubPi {
17
+ commands: Map<string, { description: string; handler: (args: string, ctx: any) => Promise<void> }>;
18
+ tools: Map<string, { name: string; execute: (id: string, params: any, signal: any, onUpdate: any, ctx: any) => Promise<{ details: any }> }>;
19
+ hooks: Map<string, Array<(event: any, ctx: any) => Promise<void>>>;
20
+ registerCommand(name: string, opts: any): void;
21
+ registerTool(opts: any): void;
22
+ on(event: string, fn: (event: any, ctx: any) => Promise<void>): void;
23
+ fire(event: string, payload: any, ctx: any): Promise<void>;
24
+ }
25
+ function makePi(): StubPi {
26
+ const commands = new Map();
27
+ const tools = new Map();
28
+ const hooks = new Map();
29
+ return {
30
+ commands, tools, hooks,
31
+ registerCommand(name, opts) { commands.set(name, opts); },
32
+ registerTool(opts) { tools.set(opts.name, opts); },
33
+ on(event, fn) { if (!hooks.has(event)) hooks.set(event, []); hooks.get(event)!.push(fn); },
34
+ async fire(event, payload, ctx) {
35
+ for (const fn of hooks.get(event) ?? []) await fn(payload, ctx);
36
+ },
37
+ };
38
+ }
39
+
40
+ describe('coworker-scratchpad extension (live kernel)', () => {
41
+ let workspace: string;
42
+ let scratchpadRoot: string;
43
+
44
+ beforeEach(async () => {
45
+ workspace = await mkdtemp(join(tmpdir(), 'spext-ws-'));
46
+ await mkdir(join(workspace, '.otto', 'inputs'), { recursive: true });
47
+ scratchpadRoot = await mkdtemp(join(tmpdir(), 'spext-root-'));
48
+ process.env.OTTO_SCRATCHPAD_ROOT = scratchpadRoot; // see index.ts: honors this env var for tests
49
+ });
50
+ afterEach(async () => {
51
+ delete process.env.OTTO_SCRATCHPAD_ROOT;
52
+ await rm(workspace, { recursive: true, force: true });
53
+ await rm(scratchpadRoot, { recursive: true, force: true });
54
+ });
55
+
56
+ it('registers /sp and scratchpad after session_start; survives exec + dispose', async () => {
57
+ const pi = makePi();
58
+ coworkerScratchpadExtension(pi as any);
59
+
60
+ // session_start fires after registration; the index.ts handler captures ctx for the manager.
61
+ await pi.fire('session_start', {}, {
62
+ cwd: workspace,
63
+ sessionManager: { getSessionFile: () => undefined },
64
+ hasUI: false,
65
+ ui: { notify: () => {} },
66
+ });
67
+
68
+ assert.ok(pi.commands.has('sp'), 'sp slash command registered');
69
+ assert.ok(pi.tools.has('cw_scratchpad'), 'cw_scratchpad tool registered');
70
+
71
+ // Run a cell via the scratchpad tool.
72
+ const execResult = await pi.tools.get('cw_scratchpad')!.execute(
73
+ '', { action: 'exec', code: 'globalThis.x = 42; return globalThis.x;' }, undefined, undefined, {},
74
+ );
75
+ const exec = execResult.details as { ok: boolean; cell_id: number; mime: Record<string, unknown> };
76
+ assert.equal(exec.ok, true);
77
+ assert.equal(exec.cell_id, 1);
78
+ assert.deepEqual(exec.mime, { 'application/json': 42 });
79
+ assert.ok(existsSync(join(scratchpadRoot, 'default', 'kernel.db')), 'kernel.db created');
80
+ assert.ok(existsSync(join(scratchpadRoot, 'default', 'cells.jsonl')), 'cells.jsonl created');
81
+
82
+ // View shows the cell.
83
+ const viewResult = await pi.tools.get('cw_scratchpad')!.execute('', { action: 'view' }, undefined, undefined, {});
84
+ const view = viewResult.details as { total_cells: number; cells: Array<{ id: number; ok: boolean; value: unknown }> };
85
+ assert.equal(view.total_cells, 1);
86
+ assert.equal(view.cells[0].id, 1);
87
+ assert.equal(view.cells[0].ok, true);
88
+ assert.equal(view.cells[0].value, 42);
89
+
90
+ // session_shutdown disposes the manager (kernel exits cleanly).
91
+ await pi.fire('session_shutdown', {}, {});
92
+ // After shutdown a re-exec would re-spawn; we don't assert that here (covered by manager tests).
93
+ });
94
+ });
95
+
96
+ describe('coworker-scratchpad extension (session affinity — 1g)', () => {
97
+ let workspace: string;
98
+ let scratchpadRoot: string;
99
+
100
+ beforeEach(async () => {
101
+ workspace = await mkdtemp(join(tmpdir(), 'spext-ws-'));
102
+ await mkdir(join(workspace, '.otto', 'inputs'), { recursive: true });
103
+ scratchpadRoot = await mkdtemp(join(tmpdir(), 'spext-root-'));
104
+ process.env.OTTO_SCRATCHPAD_ROOT = scratchpadRoot;
105
+ });
106
+ afterEach(async () => {
107
+ delete process.env.OTTO_SCRATCHPAD_ROOT;
108
+ await rm(workspace, { recursive: true, force: true });
109
+ await rm(scratchpadRoot, { recursive: true, force: true });
110
+ });
111
+
112
+ function makeSessionCtx(sessionFile: string | undefined): {
113
+ cwd: string;
114
+ sessionManager: { getSessionFile: () => string | undefined };
115
+ hasUI: boolean;
116
+ ui: { notify: (m: string, l: string) => void; confirm: (a: string, b: string) => Promise<boolean> };
117
+ notifications: Array<[string, string]>;
118
+ } {
119
+ const notifications: Array<[string, string]> = [];
120
+ return {
121
+ cwd: workspace,
122
+ sessionManager: { getSessionFile: () => sessionFile },
123
+ hasUI: false,
124
+ ui: {
125
+ notify: (m, l) => notifications.push([l, m]),
126
+ confirm: async () => true,
127
+ },
128
+ notifications,
129
+ };
130
+ }
131
+
132
+ it('session_start restores currentName from a valid sidecar', async () => {
133
+ // Pre-create the scratchpad on disk so the meta.json existence check passes.
134
+ const { mkdir, writeFile } = await import('node:fs/promises');
135
+ await mkdir(join(scratchpadRoot, 'p1'), { recursive: true });
136
+ await writeFile(join(scratchpadRoot, 'p1', 'meta.json'), '{}');
137
+ // Pre-write a sidecar for sessionId=sess-A.
138
+ await mkdir(join(scratchpadRoot, '_sessions'), { recursive: true });
139
+ await writeFile(
140
+ join(scratchpadRoot, '_sessions', 'sidecar_sess-A.json'),
141
+ JSON.stringify({ schema_version: 1, session_id: 'sess-A', current_name: 'p1', attached_at: 't' }),
142
+ );
143
+
144
+ const pi = makePi();
145
+ coworkerScratchpadExtension(pi as any);
146
+ const ctx = makeSessionCtx('/tmp/sess-A.jsonl');
147
+ await pi.fire('session_start', {}, ctx);
148
+
149
+ assert.ok(ctx.notifications.some(([l, m]) => l === 'info' && /restored/.test(m)));
150
+ });
151
+
152
+ it('session_start clears a broken sidecar silently when no workspace pointer applies', async () => {
153
+ const { mkdir, writeFile } = await import('node:fs/promises');
154
+ await mkdir(join(scratchpadRoot, '_sessions'), { recursive: true });
155
+ const sidecarPath = join(scratchpadRoot, '_sessions', 'sidecar_sess-B.json');
156
+ await writeFile(
157
+ sidecarPath,
158
+ JSON.stringify({ schema_version: 1, session_id: 'sess-B', current_name: 'p-missing', attached_at: 't' }),
159
+ );
160
+
161
+ const pi = makePi();
162
+ coworkerScratchpadExtension(pi as any);
163
+ const ctx = makeSessionCtx('/tmp/sess-B.jsonl');
164
+ await pi.fire('session_start', {}, ctx);
165
+
166
+ assert.ok(!existsSync(sidecarPath), 'broken sidecar deleted');
167
+ // With Task A: no "not restored" noise — silent clean start when no fallback applies.
168
+ assert.equal(ctx.notifications.length, 0, 'no notifications when neither restore source applies');
169
+ });
170
+
171
+ it('session_start with no sidecar is a silent no-op', async () => {
172
+ const pi = makePi();
173
+ coworkerScratchpadExtension(pi as any);
174
+ const ctx = makeSessionCtx('/tmp/sess-C.jsonl');
175
+ await pi.fire('session_start', {}, ctx);
176
+ assert.equal(ctx.notifications.length, 0);
177
+ });
178
+ });
179
+
180
+ describe('coworker-scratchpad restore precedence (Task A)', () => {
181
+ let root: string;
182
+ let ws: string;
183
+ beforeEach(() => {
184
+ root = mkdtempSync(join(tmpdir(), 'cws-root-'));
185
+ ws = mkdtempSync(join(tmpdir(), 'cws-ws-'));
186
+ execSync('git init -q', { cwd: ws });
187
+ });
188
+ afterEach(() => {
189
+ rmSync(root, { recursive: true, force: true });
190
+ rmSync(ws, { recursive: true, force: true });
191
+ });
192
+
193
+ function makeScratchpad(name: string): void {
194
+ mkdirSync(join(root, name), { recursive: true });
195
+ writeFileSync(join(root, name, 'meta.json'), JSON.stringify({ name, schema_version: 3 }));
196
+ }
197
+
198
+ it('(a) sidecar wins when its scratchpad exists', () => {
199
+ makeScratchpad('alpha');
200
+ makeScratchpad('beta');
201
+ const sessionId = 'sess-A';
202
+ writeSessionSidecar(sessionSidecarPath(root, sessionId), {
203
+ schema_version: 1,
204
+ session_id: sessionId,
205
+ current_name: 'alpha',
206
+ attached_at: new Date().toISOString(),
207
+ });
208
+ const wsRoot = detectWorkspaceRoot(ws);
209
+ writeWorkspacePointer(workspacePointerPath(root, workspaceHash(wsRoot)), {
210
+ schema_version: 1,
211
+ workspace_hash: workspaceHash(wsRoot),
212
+ workspace_root: wsRoot,
213
+ last_session_id: 'sess-OTHER',
214
+ last_current_name: 'beta',
215
+ last_attached_at: new Date().toISOString(),
216
+ });
217
+
218
+ const result = tryRestoreCurrentName(root, sessionId, ws, Date.now());
219
+ assert.equal(result.name, 'alpha');
220
+ assert.match(result.notice!, /alpha.*restored/);
221
+ });
222
+
223
+ it('(b) sidecar falls through to workspace pointer when its scratchpad is gone', () => {
224
+ makeScratchpad('beta'); // alpha intentionally absent
225
+ const sessionId = 'sess-A';
226
+ const sidecarPath = sessionSidecarPath(root, sessionId);
227
+ writeSessionSidecar(sidecarPath, {
228
+ schema_version: 1,
229
+ session_id: sessionId,
230
+ current_name: 'alpha-gone',
231
+ attached_at: new Date().toISOString(),
232
+ });
233
+ const wsRoot = detectWorkspaceRoot(ws);
234
+ writeWorkspacePointer(workspacePointerPath(root, workspaceHash(wsRoot)), {
235
+ schema_version: 1,
236
+ workspace_hash: workspaceHash(wsRoot),
237
+ workspace_root: wsRoot,
238
+ last_session_id: 'sess-OTHER',
239
+ last_current_name: 'beta',
240
+ last_attached_at: new Date().toISOString(),
241
+ });
242
+
243
+ const result = tryRestoreCurrentName(root, sessionId, ws, Date.now());
244
+ assert.equal(result.name, 'beta');
245
+ assert.match(result.notice!, /beta.*from workspace/);
246
+ assert.equal(existsSync(sidecarPath), false, 'broken sidecar should be deleted');
247
+ });
248
+
249
+ it('(c) pointer-only path (no sidecar)', () => {
250
+ makeScratchpad('beta');
251
+ const wsRoot = detectWorkspaceRoot(ws);
252
+ writeWorkspacePointer(workspacePointerPath(root, workspaceHash(wsRoot)), {
253
+ schema_version: 1,
254
+ workspace_hash: workspaceHash(wsRoot),
255
+ workspace_root: wsRoot,
256
+ last_session_id: 'sess-OTHER',
257
+ last_current_name: 'beta',
258
+ last_attached_at: new Date().toISOString(),
259
+ });
260
+ const result = tryRestoreCurrentName(root, 'sess-FRESH', ws, Date.now());
261
+ assert.equal(result.name, 'beta');
262
+ assert.match(result.notice!, /beta.*from workspace/);
263
+ });
264
+
265
+ it('(d) no restore when neither sidecar nor fresh pointer exist', () => {
266
+ const result = tryRestoreCurrentName(root, 'sess-FRESH', ws, Date.now());
267
+ assert.equal(result.name, null);
268
+ assert.equal(result.notice, null);
269
+ });
270
+
271
+ it('(e) stale workspace pointer (> 7d old) does not restore', () => {
272
+ makeScratchpad('beta');
273
+ const wsRoot = detectWorkspaceRoot(ws);
274
+ const eightDaysAgo = new Date(Date.now() - 8 * 24 * 60 * 60 * 1000).toISOString();
275
+ writeWorkspacePointer(workspacePointerPath(root, workspaceHash(wsRoot)), {
276
+ schema_version: 1, workspace_hash: workspaceHash(wsRoot), workspace_root: wsRoot,
277
+ last_session_id: 'sess-OTHER', last_current_name: 'beta',
278
+ last_attached_at: eightDaysAgo,
279
+ });
280
+ const result = tryRestoreCurrentName(root, 'sess-FRESH', ws, Date.now());
281
+ assert.equal(result.name, null);
282
+ assert.equal(result.notice, null);
283
+ });
284
+ });
285
+
286
+ // Phase 3.1 Task 4: closure-shape lock for the scratchpad onDataLoad hop.
287
+ // The activator wires `new ScratchpadManager({ onDataLoad })` to a closure that
288
+ // reads getMemoryRecorder() lazily and forwards (drawer, scratchpadName) into
289
+ // recorder.recordFileLoad(...). These tests replicate that closure inline so
290
+ // the contract is verified independently of the activator wiring (which is
291
+ // exercised by the Task 5 integration test).
292
+ describe('coworker-scratchpad onDataLoad closure shape (Phase 3.1 Task 4)', () => {
293
+ type DrawerLite = {
294
+ kind: 'data_load';
295
+ collector: string;
296
+ uri: string;
297
+ bytes: number | null;
298
+ rows_loaded: number | null;
299
+ loaded_at: string;
300
+ schema: null;
301
+ };
302
+ type RecorderLike = {
303
+ recordFileLoad: (args: {
304
+ scratchpadName: string; collector: string; uri: string;
305
+ bytes: number; rows_loaded?: number; schema?: object; turnId: string;
306
+ }) => Promise<unknown>;
307
+ };
308
+
309
+ function makeClosure(getRecorder: () => RecorderLike | null) {
310
+ return (drawer: DrawerLite, scratchpadName: string): void => {
311
+ const recorder = getRecorder();
312
+ if (!recorder) return;
313
+ void recorder.recordFileLoad({
314
+ scratchpadName,
315
+ collector: drawer.collector,
316
+ uri: drawer.uri,
317
+ bytes: drawer.bytes ?? 0,
318
+ rows_loaded: drawer.rows_loaded ?? undefined,
319
+ schema: drawer.schema ?? undefined,
320
+ turnId: '',
321
+ }).catch(() => { /* silent: file loads are frequent; failures visible in /audit */ });
322
+ };
323
+ }
324
+
325
+ function makeDrawer(overrides: Partial<DrawerLite> = {}): DrawerLite {
326
+ return {
327
+ kind: 'data_load',
328
+ collector: 'file',
329
+ uri: 'file:///tmp/data.csv',
330
+ bytes: 1234,
331
+ rows_loaded: 10,
332
+ loaded_at: new Date().toISOString(),
333
+ schema: null,
334
+ ...overrides,
335
+ };
336
+ }
337
+
338
+ it('does not throw and does not call recordFileLoad when recorder is null', () => {
339
+ let called = 0;
340
+ const onDataLoad = makeClosure(() => null);
341
+ // No assertion on recorder.recordFileLoad — recorder is null. Just ensure no throw.
342
+ assert.doesNotThrow(() => onDataLoad(makeDrawer(), 'p1'));
343
+ assert.equal(called, 0);
344
+ });
345
+
346
+ it('calls recordFileLoad with translated args when recorder is present', async () => {
347
+ const calls: Array<Parameters<RecorderLike['recordFileLoad']>[0]> = [];
348
+ const recorder: RecorderLike = {
349
+ recordFileLoad: async (args) => { calls.push(args); return { id: 'd1' }; },
350
+ };
351
+ const onDataLoad = makeClosure(() => recorder);
352
+ const drawer = makeDrawer({ collector: 'http', uri: 'https://x/y.json', bytes: 555, rows_loaded: 7 });
353
+ onDataLoad(drawer, 'p1');
354
+ // Allow the floating promise to resolve.
355
+ await new Promise((r) => setImmediate(r));
356
+ assert.equal(calls.length, 1);
357
+ assert.equal(calls[0].scratchpadName, 'p1');
358
+ assert.equal(calls[0].collector, 'http');
359
+ assert.equal(calls[0].uri, 'https://x/y.json');
360
+ assert.equal(calls[0].bytes, 555);
361
+ assert.equal(calls[0].rows_loaded, 7);
362
+ assert.equal(calls[0].schema, undefined);
363
+ assert.equal(calls[0].turnId, '');
364
+ });
365
+
366
+ it('swallows recordFileLoad rejection silently (no unhandled rejection)', async () => {
367
+ const recorder: RecorderLike = {
368
+ recordFileLoad: async () => { throw new Error('backend offline'); },
369
+ };
370
+ const onDataLoad = makeClosure(() => recorder);
371
+ // Track unhandled rejections to assert none escape.
372
+ const seen: unknown[] = [];
373
+ const handler = (r: unknown): void => { seen.push(r); };
374
+ process.on('unhandledRejection', handler);
375
+ try {
376
+ assert.doesNotThrow(() => onDataLoad(makeDrawer(), 'p1'));
377
+ // Allow microtasks + tick for unhandled rejection detection.
378
+ await new Promise((r) => setImmediate(r));
379
+ await new Promise((r) => setImmediate(r));
380
+ assert.equal(seen.length, 0, 'no unhandled rejection should escape');
381
+ } finally {
382
+ process.off('unhandledRejection', handler);
383
+ }
384
+ });
385
+
386
+ it('maps null bytes to 0 and null rows_loaded to undefined', async () => {
387
+ const calls: Array<Parameters<RecorderLike['recordFileLoad']>[0]> = [];
388
+ const recorder: RecorderLike = {
389
+ recordFileLoad: async (args) => { calls.push(args); return { id: 'd1' }; },
390
+ };
391
+ const onDataLoad = makeClosure(() => recorder);
392
+ const drawer = makeDrawer({ bytes: null, rows_loaded: null });
393
+ onDataLoad(drawer, 'p2');
394
+ await new Promise((r) => setImmediate(r));
395
+ assert.equal(calls.length, 1);
396
+ assert.equal(calls[0].bytes, 0);
397
+ assert.equal(calls[0].rows_loaded, undefined);
398
+ assert.equal(calls[0].scratchpadName, 'p2');
399
+ });
400
+ });
401
+
402
+ describe('scratchpad activator — onArtifactCreate closure (Phase 4 Task 12)', () => {
403
+ it('closure with null recorder does not throw', () => {
404
+ const getRec = (): null => null;
405
+ const drawer = {
406
+ kind: 'artifact' as const, slug: 'rca-1', artifact_kind: 'report',
407
+ uri: 'artifact://rca-1', primary_path: '/x/report.md', created_at: 't',
408
+ };
409
+ const onArtifactCreate = (d: typeof drawer, name: string): void => {
410
+ const rec = getRec();
411
+ if (!rec) return;
412
+ };
413
+ assert.doesNotThrow(() => onArtifactCreate(drawer, 'p1'));
414
+ });
415
+ it('closure with recorder calls recordArtifact with translated args', async () => {
416
+ const calls: Array<{ scratchpadName: string; slug: string; kind: string; uri: string }> = [];
417
+ const recorder = {
418
+ recordArtifact: async (args: { scratchpadName: string; slug: string; kind: string; uri: string; turnId: string }) => {
419
+ calls.push({ scratchpadName: args.scratchpadName, slug: args.slug, kind: args.kind, uri: args.uri });
420
+ },
421
+ };
422
+ const drawer = {
423
+ kind: 'artifact' as const, slug: 'rca-1', artifact_kind: 'report',
424
+ uri: 'artifact://rca-1', primary_path: '/x/report.md', created_at: 't',
425
+ };
426
+ const onArtifactCreate = (d: typeof drawer, name: string): void => {
427
+ void recorder.recordArtifact({
428
+ scratchpadName: name, slug: d.slug, kind: d.artifact_kind, uri: d.uri, turnId: '',
429
+ }).catch(() => {});
430
+ };
431
+ onArtifactCreate(drawer, 'p1');
432
+ await new Promise(r => setImmediate(r));
433
+ assert.equal(calls.length, 1);
434
+ assert.equal(calls[0]!.scratchpadName, 'p1');
435
+ assert.equal(calls[0]!.slug, 'rca-1');
436
+ assert.equal(calls[0]!.kind, 'report');
437
+ assert.equal(calls[0]!.uri, 'artifact://rca-1');
438
+ });
439
+ });
440
+
441
+ describe('coworker-scratchpad activator — OTTO_SUBAGENT_SCRATCHPAD force-attach (Phase 4.5)', () => {
442
+ const ORIGINAL_SCRATCH = process.env.OTTO_SCRATCHPAD_ROOT;
443
+ const ORIGINAL_SUB = process.env.OTTO_SUBAGENT_SCRATCHPAD;
444
+
445
+ function cleanup(): void {
446
+ if (ORIGINAL_SCRATCH !== undefined) process.env.OTTO_SCRATCHPAD_ROOT = ORIGINAL_SCRATCH;
447
+ else delete process.env.OTTO_SCRATCHPAD_ROOT;
448
+ if (ORIGINAL_SUB !== undefined) process.env.OTTO_SUBAGENT_SCRATCHPAD = ORIGINAL_SUB;
449
+ else delete process.env.OTTO_SUBAGENT_SCRATCHPAD;
450
+ }
451
+
452
+ it('env var set + scratchpad missing → creates dir + meta.json + attaches', async () => {
453
+ const root = mkdtempSync(join(tmpdir(), 'sub-att-'));
454
+ process.env.OTTO_SCRATCHPAD_ROOT = root;
455
+ process.env.OTTO_SUBAGENT_SCRATCHPAD = 'subagent-rca-analyst-abc123';
456
+ try {
457
+ const api = makeFakeApi();
458
+ coworkerScratchpadExtension(api.api);
459
+ await fireSessionStart(api, { cwd: mkdtempSync(join(tmpdir(), 'sub-ws-')) });
460
+ assert.ok(existsSync(join(root, 'subagent-rca-analyst-abc123', 'meta.json')),
461
+ 'expected scratchpad dir + meta.json created');
462
+ const notice = api.notifyCalls.find((c) => /subagent dispatch/.test(c.message));
463
+ assert.ok(notice, 'expected subagent-dispatch attach notice');
464
+ } finally { cleanup(); }
465
+ });
466
+
467
+ it('env var set + scratchpad already exists → attaches without recreating', async () => {
468
+ const root = mkdtempSync(join(tmpdir(), 'sub-att2-'));
469
+ const name = 'subagent-rca-analyst-def456';
470
+ mkdirSync(join(root, name), { recursive: true, mode: 0o700 });
471
+ writeFileSync(
472
+ join(root, name, 'meta.json'),
473
+ JSON.stringify({ name, schema_version: 1, sentinel: true }),
474
+ { mode: 0o600 },
475
+ );
476
+ process.env.OTTO_SCRATCHPAD_ROOT = root;
477
+ process.env.OTTO_SUBAGENT_SCRATCHPAD = name;
478
+ try {
479
+ const api = makeFakeApi();
480
+ coworkerScratchpadExtension(api.api);
481
+ await fireSessionStart(api, { cwd: mkdtempSync(join(tmpdir(), 'sub-ws-')) });
482
+ const meta = JSON.parse(readFileSync(join(root, name, 'meta.json'), 'utf8'));
483
+ assert.equal(meta.sentinel, true, 'existing meta.json must NOT be overwritten');
484
+ } finally { cleanup(); }
485
+ });
486
+
487
+ it('env var unset → existing restore logic runs unchanged (regression)', async () => {
488
+ const root = mkdtempSync(join(tmpdir(), 'sub-att3-'));
489
+ process.env.OTTO_SCRATCHPAD_ROOT = root;
490
+ delete process.env.OTTO_SUBAGENT_SCRATCHPAD;
491
+ try {
492
+ const api = makeFakeApi();
493
+ coworkerScratchpadExtension(api.api);
494
+ await fireSessionStart(api, { cwd: mkdtempSync(join(tmpdir(), 'sub-ws-')) });
495
+ const subagentNotice = api.notifyCalls.find((c) => /subagent dispatch/.test(c.message));
496
+ assert.equal(subagentNotice, undefined);
497
+ } finally { cleanup(); }
498
+ });
499
+
500
+ it('env var set to invalid name → warn + fall through; no force-attach', async () => {
501
+ const root = mkdtempSync(join(tmpdir(), 'sub-att4-'));
502
+ process.env.OTTO_SCRATCHPAD_ROOT = root;
503
+ process.env.OTTO_SUBAGENT_SCRATCHPAD = 'NOT-a-valid_subagent_name';
504
+ try {
505
+ const api = makeFakeApi();
506
+ coworkerScratchpadExtension(api.api);
507
+ await fireSessionStart(api, { cwd: mkdtempSync(join(tmpdir(), 'sub-ws-')) });
508
+ const warn = api.notifyCalls.find((c) => c.level === 'warning' && /subagent scratchpad/.test(c.message));
509
+ assert.ok(warn, 'expected warning about invalid subagent scratchpad name');
510
+ assert.equal(existsSync(join(root, 'NOT-a-valid_subagent_name')), false,
511
+ 'invalid name must NOT result in a created dir');
512
+ } finally { cleanup(); }
513
+ });
514
+ });