@cmetech/otto 1.1.1 → 1.2.4

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 (423) 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-paths.js +8 -0
  7. package/dist/resources/extensions/coworker-artifacts/artifacts-command.js +31 -0
  8. package/dist/resources/extensions/coworker-artifacts/artifacts-singleton.js +17 -0
  9. package/dist/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
  10. package/dist/resources/extensions/coworker-artifacts/index.js +125 -0
  11. package/dist/resources/extensions/coworker-artifacts/list-tool.js +27 -0
  12. package/dist/resources/extensions/coworker-artifacts/open-tool.js +25 -0
  13. package/dist/resources/extensions/coworker-memory/extension-manifest.json +13 -0
  14. package/dist/resources/extensions/coworker-memory/index.js +219 -0
  15. package/dist/resources/extensions/coworker-memory/memorize-tool.js +10 -0
  16. package/dist/resources/extensions/coworker-memory/memory-command.js +157 -0
  17. package/dist/resources/extensions/coworker-memory/memory-singleton.js +55 -0
  18. package/dist/resources/extensions/coworker-memory/recall-tool.js +18 -0
  19. package/dist/resources/extensions/coworker-memory/session-hooks.js +45 -0
  20. package/dist/resources/extensions/coworker-scratchpad/attach-banners.js +53 -0
  21. package/dist/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
  22. package/dist/resources/extensions/coworker-scratchpad/format-age.js +9 -0
  23. package/dist/resources/extensions/coworker-scratchpad/helpers.js +38 -0
  24. package/dist/resources/extensions/coworker-scratchpad/index.js +199 -0
  25. package/dist/resources/extensions/coworker-scratchpad/mime-bundle.js +20 -0
  26. package/dist/resources/extensions/coworker-scratchpad/scratchpad-tool.js +118 -0
  27. package/dist/resources/extensions/coworker-scratchpad/session-sidecar.js +60 -0
  28. package/dist/resources/extensions/coworker-scratchpad/sp-command.js +597 -0
  29. package/dist/resources/extensions/coworker-scratchpad/workspace-pointer.js +41 -0
  30. package/dist/resources/extensions/coworker-scratchpad/workspace-root.js +17 -0
  31. package/dist/resources/extensions/coworker-vault/audit-command.js +35 -0
  32. package/dist/resources/extensions/coworker-vault/connect-command.js +42 -0
  33. package/dist/resources/extensions/coworker-vault/datasource-command.js +50 -0
  34. package/dist/resources/extensions/coworker-vault/extension-manifest.json +12 -0
  35. package/dist/resources/extensions/coworker-vault/index.js +171 -0
  36. package/dist/resources/extensions/coworker-vault/test-helpers.js +86 -0
  37. package/dist/resources/extensions/coworker-vault/vault-singleton.js +24 -0
  38. package/dist/resources/extensions/otto/commands/release-notes/_data.js +71 -0
  39. package/dist/resources/extensions/otto/commands/release-notes/command.js +15 -4
  40. package/dist/resources/extensions/subagent/index.js +8 -1
  41. package/dist/resources/extensions/subagent/launch.js +37 -5
  42. package/dist/resources/extensions/subagent/run-store.js +1 -0
  43. package/dist/resources/extensions/workflow/bootstrap/register-extension.js +2 -0
  44. package/dist/resources/extensions/workflow/bootstrap/register-hooks.js +10 -0
  45. package/dist/resources/extensions/workflow/persona-status.js +87 -0
  46. package/package.json +25 -10
  47. package/packages/contracts/package.json +1 -1
  48. package/packages/coworker-artifacts/dist/artifact-store.d.ts +25 -0
  49. package/packages/coworker-artifacts/dist/artifact-store.js +187 -0
  50. package/packages/coworker-artifacts/dist/dir-snapshot.d.ts +7 -0
  51. package/packages/coworker-artifacts/dist/dir-snapshot.js +54 -0
  52. package/packages/coworker-artifacts/dist/errors.d.ts +18 -0
  53. package/packages/coworker-artifacts/dist/errors.js +37 -0
  54. package/packages/coworker-artifacts/dist/index.d.ts +7 -0
  55. package/packages/coworker-artifacts/dist/index.js +7 -0
  56. package/packages/coworker-artifacts/dist/readme-renderer.d.ts +5 -0
  57. package/packages/coworker-artifacts/dist/readme-renderer.js +47 -0
  58. package/packages/coworker-artifacts/dist/resolve-uri.d.ts +3 -0
  59. package/packages/coworker-artifacts/dist/resolve-uri.js +29 -0
  60. package/packages/coworker-artifacts/dist/slug.d.ts +4 -0
  61. package/packages/coworker-artifacts/dist/slug.js +32 -0
  62. package/packages/coworker-artifacts/dist/types.d.ts +52 -0
  63. package/packages/coworker-artifacts/dist/types.js +1 -0
  64. package/packages/coworker-artifacts/package.json +20 -0
  65. package/packages/coworker-artifacts/src/artifact-store.test.ts +188 -0
  66. package/packages/coworker-artifacts/src/artifact-store.ts +206 -0
  67. package/packages/coworker-artifacts/src/artifacts-integration.test.ts +109 -0
  68. package/packages/coworker-artifacts/src/dir-snapshot.test.ts +71 -0
  69. package/packages/coworker-artifacts/src/dir-snapshot.ts +52 -0
  70. package/packages/coworker-artifacts/src/errors.test.ts +37 -0
  71. package/packages/coworker-artifacts/src/errors.ts +28 -0
  72. package/packages/coworker-artifacts/src/index.test.ts +22 -0
  73. package/packages/coworker-artifacts/src/index.ts +7 -0
  74. package/packages/coworker-artifacts/src/readme-renderer.test.ts +72 -0
  75. package/packages/coworker-artifacts/src/readme-renderer.ts +56 -0
  76. package/packages/coworker-artifacts/src/resolve-uri.test.ts +46 -0
  77. package/packages/coworker-artifacts/src/resolve-uri.ts +29 -0
  78. package/packages/coworker-artifacts/src/slug.test.ts +47 -0
  79. package/packages/coworker-artifacts/src/slug.ts +31 -0
  80. package/packages/coworker-artifacts/src/types.ts +61 -0
  81. package/packages/coworker-artifacts/tsconfig.json +15 -0
  82. package/packages/coworker-artifacts/tsconfig.publish.json +4 -0
  83. package/packages/coworker-memory/dist/context-injection.d.ts +9 -0
  84. package/packages/coworker-memory/dist/context-injection.js +41 -0
  85. package/packages/coworker-memory/dist/errors.d.ts +25 -0
  86. package/packages/coworker-memory/dist/errors.js +51 -0
  87. package/packages/coworker-memory/dist/index.d.ts +12 -0
  88. package/packages/coworker-memory/dist/index.js +12 -0
  89. package/packages/coworker-memory/dist/layer-a-store.d.ts +16 -0
  90. package/packages/coworker-memory/dist/layer-a-store.js +78 -0
  91. package/packages/coworker-memory/dist/local-sqlite-backend.d.ts +28 -0
  92. package/packages/coworker-memory/dist/local-sqlite-backend.js +167 -0
  93. package/packages/coworker-memory/dist/memory-backend.d.ts +14 -0
  94. package/packages/coworker-memory/dist/memory-backend.js +1 -0
  95. package/packages/coworker-memory/dist/memory-recorder.d.ts +50 -0
  96. package/packages/coworker-memory/dist/memory-recorder.js +69 -0
  97. package/packages/coworker-memory/dist/migrations/001-init.sql +38 -0
  98. package/packages/coworker-memory/dist/migrations/002-artifact-kind.sql +50 -0
  99. package/packages/coworker-memory/dist/paste-detector.d.ts +5 -0
  100. package/packages/coworker-memory/dist/paste-detector.js +14 -0
  101. package/packages/coworker-memory/dist/persona-seed.d.ts +10 -0
  102. package/packages/coworker-memory/dist/persona-seed.js +38 -0
  103. package/packages/coworker-memory/dist/recall-formatter.d.ts +2 -0
  104. package/packages/coworker-memory/dist/recall-formatter.js +14 -0
  105. package/packages/coworker-memory/dist/scope-resolver.d.ts +9 -0
  106. package/packages/coworker-memory/dist/scope-resolver.js +10 -0
  107. package/packages/coworker-memory/dist/types.d.ts +51 -0
  108. package/packages/coworker-memory/dist/types.js +2 -0
  109. package/packages/coworker-memory/dist/workspace-id.d.ts +3 -0
  110. package/packages/coworker-memory/dist/workspace-id.js +54 -0
  111. package/packages/coworker-memory/package.json +35 -0
  112. package/packages/coworker-memory/src/activator-integration.test.ts +141 -0
  113. package/packages/coworker-memory/src/context-injection.test.ts +72 -0
  114. package/packages/coworker-memory/src/context-injection.ts +57 -0
  115. package/packages/coworker-memory/src/errors.test.ts +45 -0
  116. package/packages/coworker-memory/src/errors.ts +42 -0
  117. package/packages/coworker-memory/src/index.test.ts +21 -0
  118. package/packages/coworker-memory/src/index.ts +12 -0
  119. package/packages/coworker-memory/src/layer-a-store.test.ts +85 -0
  120. package/packages/coworker-memory/src/layer-a-store.ts +88 -0
  121. package/packages/coworker-memory/src/local-sqlite-backend.test.ts +110 -0
  122. package/packages/coworker-memory/src/local-sqlite-backend.ts +185 -0
  123. package/packages/coworker-memory/src/memory-backend.ts +10 -0
  124. package/packages/coworker-memory/src/memory-integration.test.ts +89 -0
  125. package/packages/coworker-memory/src/memory-recorder.test.ts +101 -0
  126. package/packages/coworker-memory/src/memory-recorder.ts +95 -0
  127. package/packages/coworker-memory/src/migrations/001-init.sql +38 -0
  128. package/packages/coworker-memory/src/migrations/002-artifact-kind.sql +50 -0
  129. package/packages/coworker-memory/src/paste-detector.test.ts +23 -0
  130. package/packages/coworker-memory/src/paste-detector.ts +18 -0
  131. package/packages/coworker-memory/src/persona-seed.test.ts +57 -0
  132. package/packages/coworker-memory/src/persona-seed.ts +46 -0
  133. package/packages/coworker-memory/src/recall-formatter.test.ts +34 -0
  134. package/packages/coworker-memory/src/recall-formatter.ts +15 -0
  135. package/packages/coworker-memory/src/scope-resolver.test.ts +23 -0
  136. package/packages/coworker-memory/src/scope-resolver.ts +18 -0
  137. package/packages/coworker-memory/src/types.ts +61 -0
  138. package/packages/coworker-memory/src/workspace-id.test.ts +48 -0
  139. package/packages/coworker-memory/src/workspace-id.ts +56 -0
  140. package/packages/coworker-memory/tsconfig.json +15 -0
  141. package/packages/coworker-memory/tsconfig.publish.json +4 -0
  142. package/packages/coworker-persona/dist/commands.d.ts +7 -0
  143. package/packages/coworker-persona/dist/commands.js +35 -0
  144. package/packages/coworker-persona/dist/defaults/manifest.yaml +12 -0
  145. package/packages/coworker-persona/dist/defaults/steering/identity.md +3 -0
  146. package/packages/coworker-persona/dist/index.d.ts +3 -0
  147. package/packages/coworker-persona/dist/index.js +3 -0
  148. package/packages/coworker-persona/dist/manifest.d.ts +24 -0
  149. package/packages/coworker-persona/dist/manifest.js +21 -0
  150. package/packages/coworker-persona/dist/registry.d.ts +22 -0
  151. package/packages/coworker-persona/dist/registry.js +142 -0
  152. package/packages/coworker-persona/package.json +28 -0
  153. package/packages/coworker-persona/scripts/copy-defaults.cjs +17 -0
  154. package/packages/coworker-persona/src/commands.ts +47 -0
  155. package/packages/coworker-persona/src/defaults/manifest.yaml +12 -0
  156. package/packages/coworker-persona/src/defaults/steering/identity.md +3 -0
  157. package/packages/coworker-persona/src/index.ts +3 -0
  158. package/packages/coworker-persona/src/manifest.test.ts +67 -0
  159. package/packages/coworker-persona/src/manifest.ts +49 -0
  160. package/packages/coworker-persona/src/registry.test.ts +89 -0
  161. package/packages/coworker-persona/src/registry.ts +147 -0
  162. package/packages/coworker-persona/tsconfig.json +15 -0
  163. package/packages/coworker-persona/tsconfig.publish.json +4 -0
  164. package/packages/coworker-scratchpad/dist/cell-archive.d.ts +39 -0
  165. package/packages/coworker-scratchpad/dist/cell-archive.js +77 -0
  166. package/packages/coworker-scratchpad/dist/cell-tree.d.ts +14 -0
  167. package/packages/coworker-scratchpad/dist/cell-tree.js +72 -0
  168. package/packages/coworker-scratchpad/dist/child-process-runtime.d.ts +129 -0
  169. package/packages/coworker-scratchpad/dist/child-process-runtime.js +427 -0
  170. package/packages/coworker-scratchpad/dist/collector-registry.d.ts +12 -0
  171. package/packages/coworker-scratchpad/dist/collector-registry.js +29 -0
  172. package/packages/coworker-scratchpad/dist/detect-kind.d.ts +3 -0
  173. package/packages/coworker-scratchpad/dist/detect-kind.js +19 -0
  174. package/packages/coworker-scratchpad/dist/file-collector.d.ts +15 -0
  175. package/packages/coworker-scratchpad/dist/file-collector.js +99 -0
  176. package/packages/coworker-scratchpad/dist/index.d.ts +13 -0
  177. package/packages/coworker-scratchpad/dist/index.js +13 -0
  178. package/packages/coworker-scratchpad/dist/kernel-bindings.d.ts +49 -0
  179. package/packages/coworker-scratchpad/dist/kernel-bindings.js +220 -0
  180. package/packages/coworker-scratchpad/dist/kernel-entry.d.ts +1 -0
  181. package/packages/coworker-scratchpad/dist/kernel-entry.js +355 -0
  182. package/packages/coworker-scratchpad/dist/kernel-protocol.d.ts +171 -0
  183. package/packages/coworker-scratchpad/dist/kernel-protocol.js +48 -0
  184. package/packages/coworker-scratchpad/dist/kernel-spawn.d.ts +3 -0
  185. package/packages/coworker-scratchpad/dist/kernel-spawn.js +54 -0
  186. package/packages/coworker-scratchpad/dist/namespace-codec.d.ts +22 -0
  187. package/packages/coworker-scratchpad/dist/namespace-codec.js +61 -0
  188. package/packages/coworker-scratchpad/dist/scratchpad-lock.d.ts +24 -0
  189. package/packages/coworker-scratchpad/dist/scratchpad-lock.js +86 -0
  190. package/packages/coworker-scratchpad/dist/scratchpad-manager.d.ts +193 -0
  191. package/packages/coworker-scratchpad/dist/scratchpad-manager.js +866 -0
  192. package/packages/coworker-scratchpad/dist/staleness-banner.d.ts +12 -0
  193. package/packages/coworker-scratchpad/dist/staleness-banner.js +27 -0
  194. package/packages/coworker-scratchpad/package.json +31 -0
  195. package/packages/coworker-scratchpad/src/cell-archive.test.ts +150 -0
  196. package/packages/coworker-scratchpad/src/cell-archive.ts +97 -0
  197. package/packages/coworker-scratchpad/src/cell-tree.test.ts +105 -0
  198. package/packages/coworker-scratchpad/src/cell-tree.ts +90 -0
  199. package/packages/coworker-scratchpad/src/child-process-runtime.test.ts +413 -0
  200. package/packages/coworker-scratchpad/src/child-process-runtime.ts +493 -0
  201. package/packages/coworker-scratchpad/src/collector-registry.test.ts +69 -0
  202. package/packages/coworker-scratchpad/src/collector-registry.ts +33 -0
  203. package/packages/coworker-scratchpad/src/detect-kind.test.ts +33 -0
  204. package/packages/coworker-scratchpad/src/detect-kind.ts +22 -0
  205. package/packages/coworker-scratchpad/src/file-collector.test.ts +109 -0
  206. package/packages/coworker-scratchpad/src/file-collector.ts +114 -0
  207. package/packages/coworker-scratchpad/src/index.ts +74 -0
  208. package/packages/coworker-scratchpad/src/kernel-bindings.test.ts +188 -0
  209. package/packages/coworker-scratchpad/src/kernel-bindings.ts +279 -0
  210. package/packages/coworker-scratchpad/src/kernel-entry.test.ts +123 -0
  211. package/packages/coworker-scratchpad/src/kernel-entry.ts +390 -0
  212. package/packages/coworker-scratchpad/src/kernel-protocol.test.ts +105 -0
  213. package/packages/coworker-scratchpad/src/kernel-protocol.ts +230 -0
  214. package/packages/coworker-scratchpad/src/kernel-spawn.test.ts +60 -0
  215. package/packages/coworker-scratchpad/src/kernel-spawn.ts +54 -0
  216. package/packages/coworker-scratchpad/src/namespace-codec.test.ts +102 -0
  217. package/packages/coworker-scratchpad/src/namespace-codec.ts +90 -0
  218. package/packages/coworker-scratchpad/src/scratchpad-lock.test.ts +98 -0
  219. package/packages/coworker-scratchpad/src/scratchpad-lock.ts +102 -0
  220. package/packages/coworker-scratchpad/src/scratchpad-manager.test.ts +1343 -0
  221. package/packages/coworker-scratchpad/src/scratchpad-manager.ts +891 -0
  222. package/packages/coworker-scratchpad/src/staleness-banner.test.ts +53 -0
  223. package/packages/coworker-scratchpad/src/staleness-banner.ts +33 -0
  224. package/packages/coworker-scratchpad/src/vault-integration.test.ts +221 -0
  225. package/packages/coworker-scratchpad/tsconfig.json +15 -0
  226. package/packages/coworker-scratchpad/tsconfig.publish.json +4 -0
  227. package/packages/coworker-types/dist/artifacts.d.ts +31 -0
  228. package/packages/coworker-types/dist/artifacts.js +2 -0
  229. package/packages/coworker-types/dist/contracts.d.ts +32 -0
  230. package/packages/coworker-types/dist/contracts.js +1 -0
  231. package/packages/coworker-types/dist/index.d.ts +5 -0
  232. package/packages/coworker-types/dist/index.js +5 -0
  233. package/packages/coworker-types/dist/memory.d.ts +61 -0
  234. package/packages/coworker-types/dist/memory.js +3 -0
  235. package/packages/coworker-types/dist/scratchpad.d.ts +43 -0
  236. package/packages/coworker-types/dist/scratchpad.js +2 -0
  237. package/packages/coworker-types/dist/vault.d.ts +34 -0
  238. package/packages/coworker-types/dist/vault.js +2 -0
  239. package/packages/coworker-types/package.json +24 -0
  240. package/packages/coworker-types/src/artifacts.test.ts +52 -0
  241. package/packages/coworker-types/src/artifacts.ts +35 -0
  242. package/packages/coworker-types/src/contracts.test.ts +43 -0
  243. package/packages/coworker-types/src/contracts.ts +36 -0
  244. package/packages/coworker-types/src/index.ts +5 -0
  245. package/packages/coworker-types/src/memory.test.ts +50 -0
  246. package/packages/coworker-types/src/memory.ts +79 -0
  247. package/packages/coworker-types/src/scratchpad.test.ts +46 -0
  248. package/packages/coworker-types/src/scratchpad.ts +51 -0
  249. package/packages/coworker-types/src/smoke.test.ts +34 -0
  250. package/packages/coworker-types/src/vault.test.ts +49 -0
  251. package/packages/coworker-types/src/vault.ts +40 -0
  252. package/packages/coworker-types/tsconfig.json +15 -0
  253. package/packages/coworker-types/tsconfig.publish.json +4 -0
  254. package/packages/coworker-utils/dist/audit-log.d.ts +34 -0
  255. package/packages/coworker-utils/dist/audit-log.js +88 -0
  256. package/packages/coworker-utils/dist/index.d.ts +6 -0
  257. package/packages/coworker-utils/dist/index.js +6 -0
  258. package/packages/coworker-utils/dist/lease.d.ts +7 -0
  259. package/packages/coworker-utils/dist/lease.js +67 -0
  260. package/packages/coworker-utils/dist/logger.d.ts +13 -0
  261. package/packages/coworker-utils/dist/logger.js +26 -0
  262. package/packages/coworker-utils/dist/migration-runner.d.ts +7 -0
  263. package/packages/coworker-utils/dist/migration-runner.js +36 -0
  264. package/packages/coworker-utils/dist/ndjson-channel.d.ts +3 -0
  265. package/packages/coworker-utils/dist/ndjson-channel.js +38 -0
  266. package/packages/coworker-utils/dist/secret-scanner.d.ts +10 -0
  267. package/packages/coworker-utils/dist/secret-scanner.js +42 -0
  268. package/packages/coworker-utils/package.json +24 -0
  269. package/packages/coworker-utils/src/audit-log.test.ts +140 -0
  270. package/packages/coworker-utils/src/audit-log.ts +107 -0
  271. package/packages/coworker-utils/src/index.ts +6 -0
  272. package/packages/coworker-utils/src/lease.test.ts +64 -0
  273. package/packages/coworker-utils/src/lease.ts +76 -0
  274. package/packages/coworker-utils/src/logger.test.ts +50 -0
  275. package/packages/coworker-utils/src/logger.ts +45 -0
  276. package/packages/coworker-utils/src/migration-runner.test.ts +65 -0
  277. package/packages/coworker-utils/src/migration-runner.ts +50 -0
  278. package/packages/coworker-utils/src/ndjson-channel.test.ts +76 -0
  279. package/packages/coworker-utils/src/ndjson-channel.ts +41 -0
  280. package/packages/coworker-utils/src/secret-scanner.test.ts +61 -0
  281. package/packages/coworker-utils/src/secret-scanner.ts +56 -0
  282. package/packages/coworker-utils/tsconfig.json +15 -0
  283. package/packages/coworker-utils/tsconfig.publish.json +4 -0
  284. package/packages/coworker-vault/dist/data-vault.d.ts +41 -0
  285. package/packages/coworker-vault/dist/data-vault.js +223 -0
  286. package/packages/coworker-vault/dist/engine-registry.d.ts +34 -0
  287. package/packages/coworker-vault/dist/engine-registry.js +90 -0
  288. package/packages/coworker-vault/dist/engines/jira.yaml +17 -0
  289. package/packages/coworker-vault/dist/errors.d.ts +28 -0
  290. package/packages/coworker-vault/dist/errors.js +57 -0
  291. package/packages/coworker-vault/dist/index.d.ts +6 -0
  292. package/packages/coworker-vault/dist/index.js +6 -0
  293. package/packages/coworker-vault/dist/injector.d.ts +19 -0
  294. package/packages/coworker-vault/dist/injector.js +77 -0
  295. package/packages/coworker-vault/dist/types.d.ts +28 -0
  296. package/packages/coworker-vault/dist/types.js +1 -0
  297. package/packages/coworker-vault/dist/vault-keep.d.ts +4 -0
  298. package/packages/coworker-vault/dist/vault-keep.js +21 -0
  299. package/packages/coworker-vault/package.json +29 -0
  300. package/packages/coworker-vault/src/data-vault.test.ts +199 -0
  301. package/packages/coworker-vault/src/data-vault.ts +257 -0
  302. package/packages/coworker-vault/src/engine-registry.test.ts +120 -0
  303. package/packages/coworker-vault/src/engine-registry.ts +107 -0
  304. package/packages/coworker-vault/src/engines/jira.yaml +17 -0
  305. package/packages/coworker-vault/src/errors.test.ts +58 -0
  306. package/packages/coworker-vault/src/errors.ts +50 -0
  307. package/packages/coworker-vault/src/index.test.ts +24 -0
  308. package/packages/coworker-vault/src/index.ts +6 -0
  309. package/packages/coworker-vault/src/injector.test.ts +109 -0
  310. package/packages/coworker-vault/src/injector.ts +98 -0
  311. package/packages/coworker-vault/src/types.ts +33 -0
  312. package/packages/coworker-vault/src/vault-keep.test.ts +49 -0
  313. package/packages/coworker-vault/src/vault-keep.ts +31 -0
  314. package/packages/coworker-vault/tsconfig.json +15 -0
  315. package/packages/coworker-vault/tsconfig.publish.json +4 -0
  316. package/packages/daemon/package.json +3 -3
  317. package/packages/mcp-server/package.json +3 -3
  318. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  319. package/packages/native/package.json +1 -1
  320. package/packages/native/tsconfig.tsbuildinfo +1 -1
  321. package/packages/pi-agent-core/package.json +1 -1
  322. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  323. package/packages/pi-ai/package.json +1 -1
  324. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  325. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +6 -1
  326. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  327. package/packages/pi-coding-agent/dist/core/extensions/runner.js +22 -3
  328. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  329. package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js +11 -0
  330. package/packages/pi-coding-agent/dist/core/resolve-config-value.test.js.map +1 -1
  331. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts +47 -0
  332. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.d.ts.map +1 -0
  333. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js +107 -0
  334. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.js.map +1 -0
  335. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts +19 -0
  336. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.d.ts.map +1 -0
  337. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js +121 -0
  338. package/packages/pi-coding-agent/dist/modes/rpc/raw-stdout.regression.test.js.map +1 -0
  339. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  340. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +17 -1
  341. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
  342. package/packages/pi-coding-agent/package.json +2 -2
  343. package/packages/pi-coding-agent/src/core/extensions/runner.ts +22 -3
  344. package/packages/pi-coding-agent/src/core/resolve-config-value.test.ts +11 -0
  345. package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.regression.test.ts +129 -0
  346. package/packages/pi-coding-agent/src/modes/rpc/raw-stdout.ts +117 -0
  347. package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +18 -1
  348. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  349. package/packages/pi-tui/package.json +1 -1
  350. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  351. package/packages/rpc-client/package.json +2 -2
  352. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  353. package/pkg/package.json +1 -1
  354. package/scripts/install.js +6 -5
  355. package/src/resources/extensions/_coworker-paths.test.ts +40 -0
  356. package/src/resources/extensions/_coworker-paths.ts +10 -0
  357. package/src/resources/extensions/coworker-artifacts/artifacts-command.test.ts +54 -0
  358. package/src/resources/extensions/coworker-artifacts/artifacts-command.ts +43 -0
  359. package/src/resources/extensions/coworker-artifacts/artifacts-singleton.test.ts +25 -0
  360. package/src/resources/extensions/coworker-artifacts/artifacts-singleton.ts +29 -0
  361. package/src/resources/extensions/coworker-artifacts/extension-manifest.json +13 -0
  362. package/src/resources/extensions/coworker-artifacts/index.test.ts +46 -0
  363. package/src/resources/extensions/coworker-artifacts/index.ts +154 -0
  364. package/src/resources/extensions/coworker-artifacts/list-tool.test.ts +29 -0
  365. package/src/resources/extensions/coworker-artifacts/list-tool.ts +53 -0
  366. package/src/resources/extensions/coworker-artifacts/open-tool.test.ts +30 -0
  367. package/src/resources/extensions/coworker-artifacts/open-tool.ts +43 -0
  368. package/src/resources/extensions/coworker-memory/extension-manifest.json +13 -0
  369. package/src/resources/extensions/coworker-memory/index.test.ts +137 -0
  370. package/src/resources/extensions/coworker-memory/index.ts +257 -0
  371. package/src/resources/extensions/coworker-memory/memorize-tool.test.ts +41 -0
  372. package/src/resources/extensions/coworker-memory/memorize-tool.ts +20 -0
  373. package/src/resources/extensions/coworker-memory/memory-command.test.ts +134 -0
  374. package/src/resources/extensions/coworker-memory/memory-command.ts +131 -0
  375. package/src/resources/extensions/coworker-memory/memory-singleton.test.ts +41 -0
  376. package/src/resources/extensions/coworker-memory/memory-singleton.ts +89 -0
  377. package/src/resources/extensions/coworker-memory/recall-tool.test.ts +50 -0
  378. package/src/resources/extensions/coworker-memory/recall-tool.ts +35 -0
  379. package/src/resources/extensions/coworker-memory/session-hooks.test.ts +77 -0
  380. package/src/resources/extensions/coworker-memory/session-hooks.ts +61 -0
  381. package/src/resources/extensions/coworker-scratchpad/attach-banners.test.ts +124 -0
  382. package/src/resources/extensions/coworker-scratchpad/attach-banners.ts +67 -0
  383. package/src/resources/extensions/coworker-scratchpad/extension-manifest.json +13 -0
  384. package/src/resources/extensions/coworker-scratchpad/format-age.test.ts +30 -0
  385. package/src/resources/extensions/coworker-scratchpad/format-age.ts +6 -0
  386. package/src/resources/extensions/coworker-scratchpad/helpers.test.ts +93 -0
  387. package/src/resources/extensions/coworker-scratchpad/helpers.ts +42 -0
  388. package/src/resources/extensions/coworker-scratchpad/index.test.ts +514 -0
  389. package/src/resources/extensions/coworker-scratchpad/index.ts +207 -0
  390. package/src/resources/extensions/coworker-scratchpad/mime-bundle.test.ts +61 -0
  391. package/src/resources/extensions/coworker-scratchpad/mime-bundle.ts +23 -0
  392. package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.test.ts +137 -0
  393. package/src/resources/extensions/coworker-scratchpad/scratchpad-tool.ts +165 -0
  394. package/src/resources/extensions/coworker-scratchpad/session-sidecar.test.ts +133 -0
  395. package/src/resources/extensions/coworker-scratchpad/session-sidecar.ts +68 -0
  396. package/src/resources/extensions/coworker-scratchpad/sp-command.test.ts +836 -0
  397. package/src/resources/extensions/coworker-scratchpad/sp-command.ts +602 -0
  398. package/src/resources/extensions/coworker-scratchpad/workspace-pointer.test.ts +74 -0
  399. package/src/resources/extensions/coworker-scratchpad/workspace-pointer.ts +55 -0
  400. package/src/resources/extensions/coworker-scratchpad/workspace-root.test.ts +51 -0
  401. package/src/resources/extensions/coworker-scratchpad/workspace-root.ts +16 -0
  402. package/src/resources/extensions/coworker-vault/audit-command.test.ts +109 -0
  403. package/src/resources/extensions/coworker-vault/audit-command.ts +56 -0
  404. package/src/resources/extensions/coworker-vault/connect-command.test.ts +103 -0
  405. package/src/resources/extensions/coworker-vault/connect-command.ts +69 -0
  406. package/src/resources/extensions/coworker-vault/datasource-command.test.ts +80 -0
  407. package/src/resources/extensions/coworker-vault/datasource-command.ts +81 -0
  408. package/src/resources/extensions/coworker-vault/extension-manifest.json +12 -0
  409. package/src/resources/extensions/coworker-vault/index.test.ts +82 -0
  410. package/src/resources/extensions/coworker-vault/index.ts +181 -0
  411. package/src/resources/extensions/coworker-vault/test-helpers.ts +120 -0
  412. package/src/resources/extensions/coworker-vault/vault-singleton.test.ts +27 -0
  413. package/src/resources/extensions/coworker-vault/vault-singleton.ts +40 -0
  414. package/src/resources/extensions/otto/commands/release-notes/_data.ts +85 -0
  415. package/src/resources/extensions/otto/commands/release-notes/command.ts +16 -3
  416. package/src/resources/extensions/subagent/index.ts +9 -0
  417. package/src/resources/extensions/subagent/launch.test.ts +97 -0
  418. package/src/resources/extensions/subagent/launch.ts +42 -5
  419. package/src/resources/extensions/subagent/run-store.ts +3 -1
  420. package/src/resources/extensions/workflow/bootstrap/register-extension.ts +2 -0
  421. package/src/resources/extensions/workflow/bootstrap/register-hooks.ts +10 -0
  422. package/src/resources/extensions/workflow/persona-status.ts +109 -0
  423. package/src/resources/extensions/workflow/tests/auto-recovery.test.ts +34 -0
@@ -0,0 +1,55 @@
1
+ import { createHash } from 'node:crypto';
2
+ import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync } from 'node:fs';
3
+ import { dirname, join } from 'node:path';
4
+
5
+ export interface WorkspacePointer {
6
+ schema_version: 1;
7
+ workspace_hash: string;
8
+ workspace_root: string;
9
+ last_session_id: string;
10
+ last_current_name: string;
11
+ last_attached_at: string;
12
+ }
13
+
14
+ export const WORKSPACE_POINTER_STALE_MS = 7 * 24 * 60 * 60 * 1000;
15
+
16
+ export function workspaceHash(workspaceRoot: string): string {
17
+ return createHash('sha256').update(workspaceRoot).digest('hex').slice(0, 16);
18
+ }
19
+
20
+ export function workspacePointerPath(rootDir: string, hash: string): string {
21
+ return join(rootDir, '_workspaces', `${hash}.json`);
22
+ }
23
+
24
+ export function readWorkspacePointer(path: string): WorkspacePointer | null {
25
+ if (!existsSync(path)) return null;
26
+ try {
27
+ const parsed = JSON.parse(readFileSync(path, 'utf8')) as Partial<WorkspacePointer>;
28
+ if (
29
+ parsed.schema_version === 1 &&
30
+ typeof parsed.workspace_hash === 'string' &&
31
+ typeof parsed.workspace_root === 'string' &&
32
+ typeof parsed.last_session_id === 'string' &&
33
+ typeof parsed.last_current_name === 'string' &&
34
+ typeof parsed.last_attached_at === 'string'
35
+ ) {
36
+ return parsed as WorkspacePointer;
37
+ }
38
+ return null;
39
+ } catch {
40
+ return null;
41
+ }
42
+ }
43
+
44
+ export function writeWorkspacePointer(path: string, payload: WorkspacePointer): void {
45
+ mkdirSync(dirname(path), { recursive: true });
46
+ const tmp = `${path}.tmp`;
47
+ writeFileSync(tmp, JSON.stringify(payload, null, 2));
48
+ renameSync(tmp, path);
49
+ }
50
+
51
+ export function isPointerFresh(pointer: WorkspacePointer, now: number): boolean {
52
+ const attachedAt = Date.parse(pointer.last_attached_at);
53
+ if (Number.isNaN(attachedAt)) return false;
54
+ return (now - attachedAt) < WORKSPACE_POINTER_STALE_MS;
55
+ }
@@ -0,0 +1,51 @@
1
+ import { describe, it } from 'node:test';
2
+ import assert from 'node:assert';
3
+ import { mkdtempSync, mkdirSync, rmSync } from 'node:fs';
4
+ import { execSync } from 'node:child_process';
5
+ import { join } from 'node:path';
6
+ import { tmpdir } from 'node:os';
7
+ import { detectWorkspaceRoot } from './workspace-root.js';
8
+
9
+ describe('detectWorkspaceRoot', () => {
10
+ it('returns git toplevel when invoked inside a git repo', () => {
11
+ const dir = mkdtempSync(join(tmpdir(), 'wsr-git-'));
12
+ try {
13
+ execSync('git init -q', { cwd: dir });
14
+ const sub = join(dir, 'a', 'b');
15
+ mkdirSync(sub, { recursive: true });
16
+ const root = detectWorkspaceRoot(sub);
17
+ // On macOS, /tmp resolves through /private — compare resolved paths.
18
+ assert.equal(
19
+ execSync('pwd -P', { cwd: dir, encoding: 'utf8' }).trim(),
20
+ execSync('pwd -P', { cwd: root, encoding: 'utf8' }).trim(),
21
+ );
22
+ } finally {
23
+ rmSync(dir, { recursive: true, force: true });
24
+ }
25
+ });
26
+
27
+ it('returns cwd when not in a git repo', () => {
28
+ const dir = mkdtempSync(join(tmpdir(), 'wsr-nogit-'));
29
+ try {
30
+ assert.equal(detectWorkspaceRoot(dir), dir);
31
+ } finally {
32
+ rmSync(dir, { recursive: true, force: true });
33
+ }
34
+ });
35
+
36
+ it('returns cwd when git command fails (mocked via PATH override)', () => {
37
+ const dir = mkdtempSync(join(tmpdir(), 'wsr-pathfail-'));
38
+ try {
39
+ // Run with PATH=/nonexistent so git isn't findable; detection should fall back to cwd.
40
+ const origPath = process.env.PATH;
41
+ process.env.PATH = '/nonexistent';
42
+ try {
43
+ assert.equal(detectWorkspaceRoot(dir), dir);
44
+ } finally {
45
+ process.env.PATH = origPath;
46
+ }
47
+ } finally {
48
+ rmSync(dir, { recursive: true, force: true });
49
+ }
50
+ });
51
+ });
@@ -0,0 +1,16 @@
1
+ import { execSync } from 'node:child_process';
2
+
3
+ export function detectWorkspaceRoot(cwd: string): string {
4
+ try {
5
+ const out = execSync('git rev-parse --show-toplevel', {
6
+ cwd,
7
+ stdio: ['ignore', 'pipe', 'ignore'],
8
+ encoding: 'utf8',
9
+ timeout: 500,
10
+ }).trim();
11
+ if (out) return out;
12
+ } catch {
13
+ /* not a git repo, git not installed, or timeout — fall through */
14
+ }
15
+ return cwd;
16
+ }
@@ -0,0 +1,109 @@
1
+ // src/resources/extensions/coworker-vault/audit-command.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import type { AuditRecord } from '@otto/coworker-utils';
8
+ import { createVaultBundle, type VaultBundle } from './vault-singleton.js';
9
+ import { runAudit } from './audit-command.js';
10
+
11
+ async function freshBundle(): Promise<VaultBundle> {
12
+ const root = mkdtempSync(join(tmpdir(), 'audit-cmd-'));
13
+ return createVaultBundle({ globalDir: join(root, 'global') });
14
+ }
15
+
16
+ function record(overrides: Partial<AuditRecord> & { ts: string }): AuditRecord {
17
+ return {
18
+ _schema: 1,
19
+ producer: 'vault',
20
+ action: 'set',
21
+ detail: {},
22
+ ...overrides,
23
+ };
24
+ }
25
+
26
+ describe('/audit', () => {
27
+ it('returns last 50 records by default, newest first', async () => {
28
+ const bundle = await freshBundle();
29
+ const base = Date.now();
30
+ for (let i = 0; i < 60; i++) {
31
+ bundle.audit.append(
32
+ record({ ts: new Date(base + i * 1000).toISOString() }),
33
+ );
34
+ }
35
+ const out = await runAudit(bundle, {});
36
+ assert.equal(out.length, 50);
37
+ // newest first → ts strictly decreasing
38
+ for (let i = 1; i < out.length; i++) {
39
+ assert.ok(out[i - 1]!.ts > out[i]!.ts, `expected ${out[i - 1]!.ts} > ${out[i]!.ts}`);
40
+ }
41
+ // Newest record overall must be first
42
+ const expectedNewest = new Date(base + 59 * 1000).toISOString();
43
+ assert.equal(out[0]!.ts, expectedNewest);
44
+ });
45
+
46
+ it('filters by producer', async () => {
47
+ const bundle = await freshBundle();
48
+ const now = Date.now();
49
+ bundle.audit.append(
50
+ record({
51
+ ts: new Date(now).toISOString(),
52
+ producer: 'vault',
53
+ action: 'set',
54
+ }),
55
+ );
56
+ bundle.audit.append(
57
+ record({
58
+ ts: new Date(now + 1000).toISOString(),
59
+ producer: 'secret-scanner',
60
+ action: 'redact',
61
+ }),
62
+ );
63
+ const out = await runAudit(bundle, { producer: 'secret-scanner' });
64
+ assert.equal(out.length, 1);
65
+ assert.equal(out[0]!.producer, 'secret-scanner');
66
+ assert.equal(out[0]!.action, 'redact');
67
+ });
68
+
69
+ it('filters by engine via detail.engine', async () => {
70
+ const bundle = await freshBundle();
71
+ const now = Date.now();
72
+ bundle.audit.append(
73
+ record({
74
+ ts: new Date(now).toISOString(),
75
+ producer: 'vault',
76
+ action: 'set',
77
+ detail: { engine: 'jira' },
78
+ }),
79
+ );
80
+ bundle.audit.append(
81
+ record({
82
+ ts: new Date(now + 1000).toISOString(),
83
+ producer: 'vault',
84
+ action: 'set',
85
+ detail: { engine: 'datadog' },
86
+ }),
87
+ );
88
+ const out = await runAudit(bundle, { engine: 'jira' });
89
+ assert.equal(out.length, 1);
90
+ assert.equal((out[0]!.detail as { engine?: string }).engine, 'jira');
91
+ });
92
+
93
+ it('--since filter accepts duration tokens (1h, 24h, 7d)', async () => {
94
+ const bundle = await freshBundle();
95
+ const now = Date.now();
96
+ const twoHoursAgo = new Date(now - 2 * 60 * 60 * 1000).toISOString();
97
+ const thirtySecondsAgo = new Date(now - 30 * 1000).toISOString();
98
+ bundle.audit.append(
99
+ record({ ts: twoHoursAgo, action: 'old' }),
100
+ );
101
+ bundle.audit.append(
102
+ record({ ts: thirtySecondsAgo, action: 'recent' }),
103
+ );
104
+ const out = await runAudit(bundle, { since: '1h' });
105
+ assert.equal(out.length, 1);
106
+ assert.equal(out[0]!.action, 'recent');
107
+ assert.equal(out[0]!.ts, thirtySecondsAgo);
108
+ });
109
+ });
@@ -0,0 +1,56 @@
1
+ // src/resources/extensions/coworker-vault/audit-command.ts
2
+ //
3
+ // Programmatic reader for the audit log. Resolves duration tokens like
4
+ // '1h', '24h', '7d' into ISO-8601 lower bounds, delegates filtering to
5
+ // AuditLog.read, and caps the result at `limit` (default 50). Records are
6
+ // returned newest-first because AuditLog yields in descending ts order.
7
+ import type { AuditRecord } from '@otto/coworker-utils';
8
+ import type { VaultBundle } from './vault-singleton.js';
9
+
10
+ export interface AuditQuery {
11
+ since?: string; // '1h' | '24h' | '7d' | ISO-8601
12
+ producer?: string;
13
+ engine?: string;
14
+ action?: string;
15
+ severity?: 'info' | 'warn';
16
+ limit?: number; // default 50
17
+ }
18
+
19
+ const DURATION_RE = /^(\d+)([smhd])$/;
20
+ const MS: Record<string, number> = {
21
+ s: 1000,
22
+ m: 60_000,
23
+ h: 3_600_000,
24
+ d: 86_400_000,
25
+ };
26
+
27
+ function resolveSince(token: string | undefined): string | undefined {
28
+ if (!token) return undefined;
29
+ const m = DURATION_RE.exec(token);
30
+ if (m) {
31
+ const n = parseInt(m[1]!, 10);
32
+ const unit = m[2]!;
33
+ return new Date(Date.now() - n * MS[unit]!).toISOString();
34
+ }
35
+ return token; // treat as ISO-8601
36
+ }
37
+
38
+ export async function runAudit(
39
+ bundle: VaultBundle,
40
+ q: AuditQuery,
41
+ ): Promise<AuditRecord[]> {
42
+ const since = resolveSince(q.since);
43
+ const limit = q.limit ?? 50;
44
+ const out: AuditRecord[] = [];
45
+ for await (const r of bundle.audit.read({
46
+ since,
47
+ producer: q.producer,
48
+ action: q.action,
49
+ severity: q.severity,
50
+ engineId: q.engine,
51
+ })) {
52
+ out.push(r);
53
+ if (out.length >= limit) break;
54
+ }
55
+ return out;
56
+ }
@@ -0,0 +1,103 @@
1
+ // src/resources/extensions/coworker-vault/connect-command.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { createVaultBundle } from './vault-singleton.js';
8
+ import { runConnect } from './connect-command.js';
9
+ import { VAULT_KEEP } from '@otto/coworker-vault';
10
+
11
+ function answers(map: Record<string, string>): (field: string) => Promise<string> {
12
+ return async (field: string) => map[field] ?? '';
13
+ }
14
+
15
+ async function freshBundle() {
16
+ const root = mkdtempSync(join(tmpdir(), 'connect-cmd-'));
17
+ return createVaultBundle({ globalDir: join(root, 'global') });
18
+ }
19
+
20
+ describe('/connect', () => {
21
+ it('creates a new entry from field prompts', async () => {
22
+ const bundle = await freshBundle();
23
+ await runConnect(bundle, {
24
+ engineId: 'jira',
25
+ entryName: 'prod',
26
+ forceWorkspace: false,
27
+ promptProvider: answers({
28
+ url: 'https://x',
29
+ email: 'a@b',
30
+ token: 'tok',
31
+ }),
32
+ });
33
+ const got = await bundle.vault.get({ engine: 'jira', name: 'prod' });
34
+ assert.deepEqual(got.fields, { url: 'https://x', email: 'a@b', token: 'tok' });
35
+ });
36
+
37
+ it('edits an existing entry; sentinel preserves the stored secret', async () => {
38
+ const bundle = await freshBundle();
39
+ await bundle.vault.set(
40
+ { engine: 'jira', name: 'prod' },
41
+ { url: 'u', email: 'e', token: 'OLD' },
42
+ );
43
+ await runConnect(bundle, {
44
+ engineId: 'jira',
45
+ entryName: 'prod',
46
+ forceWorkspace: false,
47
+ promptProvider: answers({
48
+ url: 'NEW_URL',
49
+ email: 'NEW_EMAIL',
50
+ token: VAULT_KEEP,
51
+ }),
52
+ });
53
+ const got = await bundle.vault.get({ engine: 'jira', name: 'prod' });
54
+ assert.deepEqual(got.fields, { url: 'NEW_URL', email: 'NEW_EMAIL', token: 'OLD' });
55
+ });
56
+
57
+ it('rejects unknown engine', async () => {
58
+ const bundle = await freshBundle();
59
+ await assert.rejects(
60
+ runConnect(bundle, {
61
+ engineId: 'nope',
62
+ entryName: 'prod',
63
+ forceWorkspace: false,
64
+ promptProvider: answers({}),
65
+ }),
66
+ /Unknown engine/,
67
+ );
68
+ });
69
+
70
+ it('rejects sentinel in create-mode secret field', async () => {
71
+ const bundle = await freshBundle();
72
+ await assert.rejects(
73
+ runConnect(bundle, {
74
+ engineId: 'jira',
75
+ entryName: 'prod',
76
+ forceWorkspace: false,
77
+ promptProvider: answers({
78
+ url: 'https://x',
79
+ email: 'a@b',
80
+ token: VAULT_KEEP,
81
+ }),
82
+ }),
83
+ /VAULT_KEEP is reserved/,
84
+ );
85
+ });
86
+
87
+ it('errors when a required field is empty', async () => {
88
+ const bundle = await freshBundle();
89
+ await assert.rejects(
90
+ runConnect(bundle, {
91
+ engineId: 'jira',
92
+ entryName: 'prod',
93
+ forceWorkspace: false,
94
+ promptProvider: answers({
95
+ url: '',
96
+ email: 'a@b',
97
+ token: 'tok',
98
+ }),
99
+ }),
100
+ /required/i,
101
+ );
102
+ });
103
+ });
@@ -0,0 +1,69 @@
1
+ // src/resources/extensions/coworker-vault/connect-command.ts
2
+ //
3
+ // Programmatic /connect wizard. Walks an engine's fields via a `promptProvider`
4
+ // callback, handles VAULT_KEEP sentinel for edits, validates required fields,
5
+ // and writes through LocalDataVault.set. No TUI dependencies — the TUI binding
6
+ // (clack/prompts) is wired separately and passes a clack-based provider.
7
+ import {
8
+ mergeWithSentinel,
9
+ assertNoSentinelInCreate,
10
+ VAULT_KEEP,
11
+ VaultEntryNotFound,
12
+ } from '@otto/coworker-vault';
13
+ import type { VaultBundle } from './vault-singleton.js';
14
+
15
+ export interface PromptFieldOptions {
16
+ label: string;
17
+ secret: boolean;
18
+ required: boolean;
19
+ defaultValue?: string;
20
+ }
21
+
22
+ export interface ConnectOptions {
23
+ engineId: string;
24
+ entryName: string;
25
+ forceWorkspace: boolean;
26
+ promptProvider: (field: string, opts: PromptFieldOptions) => Promise<string>;
27
+ }
28
+
29
+ export async function runConnect(bundle: VaultBundle, opts: ConnectOptions): Promise<void> {
30
+ const engine = bundle.registry.require(opts.engineId);
31
+
32
+ let existing: Record<string, string> | undefined;
33
+ try {
34
+ const got = await bundle.vault.get({ engine: opts.engineId, name: opts.entryName });
35
+ existing = got.fields;
36
+ } catch (err) {
37
+ if (!(err instanceof VaultEntryNotFound)) throw err;
38
+ }
39
+
40
+ const submitted: Record<string, string> = {};
41
+ for (const f of engine.fields) {
42
+ const defaultValue = existing && f.secret
43
+ ? VAULT_KEEP
44
+ : (existing?.[f.name] ?? f.default ?? '');
45
+ const value = await opts.promptProvider(f.name, {
46
+ label: f.label,
47
+ secret: f.secret,
48
+ required: f.required,
49
+ defaultValue,
50
+ });
51
+ if (f.required && value.trim() === '') {
52
+ throw new Error(`Field "${f.name}" is required.`);
53
+ }
54
+ submitted[f.name] = value;
55
+ }
56
+
57
+ if (!existing) {
58
+ assertNoSentinelInCreate(engine.fields, submitted);
59
+ }
60
+ const merged = existing
61
+ ? mergeWithSentinel(engine.fields, existing, submitted)
62
+ : submitted;
63
+
64
+ await bundle.vault.set(
65
+ { engine: opts.engineId, name: opts.entryName },
66
+ merged,
67
+ { forceWorkspace: opts.forceWorkspace },
68
+ );
69
+ }
@@ -0,0 +1,80 @@
1
+ // src/resources/extensions/coworker-vault/datasource-command.test.ts
2
+ import { describe, it } from 'node:test';
3
+ import assert from 'node:assert/strict';
4
+ import { mkdtempSync, existsSync } from 'node:fs';
5
+ import { tmpdir } from 'node:os';
6
+ import { join } from 'node:path';
7
+ import { createVaultBundle } from './vault-singleton.js';
8
+ import {
9
+ runDatasourceList,
10
+ runDatasourceRemove,
11
+ runDatasourceTest,
12
+ } from './datasource-command.js';
13
+
14
+ async function freshBundle() {
15
+ const root = mkdtempSync(join(tmpdir(), 'datasource-cmd-'));
16
+ const globalDir = join(root, 'global');
17
+ const bundle = await createVaultBundle({ globalDir });
18
+ return { bundle, globalDir };
19
+ }
20
+
21
+ describe('/datasource', () => {
22
+ it('list returns rows with engine, name, scope, fields_set (secret fields marked)', async () => {
23
+ const { bundle } = await freshBundle();
24
+ await bundle.vault.set(
25
+ { engine: 'jira', name: 'prod' },
26
+ { url: 'u', email: 'e', token: 't' },
27
+ );
28
+ const rows = await runDatasourceList(bundle, {});
29
+ assert.equal(rows.length, 1);
30
+ const row = rows[0]!;
31
+ assert.equal(row.engine, 'jira');
32
+ assert.equal(row.name, 'prod');
33
+ assert.equal(row.scope, 'global');
34
+ const sortedFields = [...row.fields].sort((a, b) => a.name.localeCompare(b.name));
35
+ assert.deepEqual(sortedFields, [
36
+ { name: 'email', secret: false, display: 'e' },
37
+ { name: 'token', secret: true, display: '••••••' },
38
+ { name: 'url', secret: false, display: 'u' },
39
+ ]);
40
+ });
41
+
42
+ it('list filters by --engine', async () => {
43
+ const { bundle } = await freshBundle();
44
+ await bundle.vault.set(
45
+ { engine: 'jira', name: 'prod' },
46
+ { url: 'u', email: 'e', token: 't' },
47
+ );
48
+ const jiraRows = await runDatasourceList(bundle, { engine: 'jira' });
49
+ assert.equal(jiraRows.length, 1);
50
+ const datadogRows = await runDatasourceList(bundle, { engine: 'datadog' });
51
+ assert.equal(datadogRows.length, 0);
52
+ });
53
+
54
+ it('remove deletes entry file', async () => {
55
+ const { bundle, globalDir } = await freshBundle();
56
+ await bundle.vault.set(
57
+ { engine: 'jira', name: 'prod' },
58
+ { url: 'u', email: 'e', token: 't' },
59
+ );
60
+ const path = join(globalDir, 'data_vault', 'jira-prod.json');
61
+ assert.equal(existsSync(path), true);
62
+ await runDatasourceRemove(bundle, { ref: 'jira:prod' });
63
+ assert.equal(existsSync(path), false);
64
+ });
65
+
66
+ it('test returns OTTO_DS_* env-var names that would inject (no network)', async () => {
67
+ const { bundle } = await freshBundle();
68
+ await bundle.vault.set(
69
+ { engine: 'jira', name: 'prod' },
70
+ { url: 'u', email: 'e', token: 't' },
71
+ );
72
+ const preview = await runDatasourceTest(bundle, { ref: 'jira:prod' });
73
+ const sorted = [...preview.envVarNames].sort();
74
+ assert.deepEqual(sorted, [
75
+ 'OTTO_DS_JIRA_PROD__EMAIL',
76
+ 'OTTO_DS_JIRA_PROD__TOKEN',
77
+ 'OTTO_DS_JIRA_PROD__URL',
78
+ ]);
79
+ });
80
+ });
@@ -0,0 +1,81 @@
1
+ // src/resources/extensions/coworker-vault/datasource-command.ts
2
+ //
3
+ // Programmatic /datasource list|remove|test commands. `edit` is wired at the
4
+ // routing layer (Task 16) as an alias for runConnect and does not need its own
5
+ // function here. No TUI dependencies.
6
+ import { LocalDataVault, envVarName } from '@otto/coworker-vault';
7
+ import type { VaultBundle } from './vault-singleton.js';
8
+
9
+ export interface ListedField {
10
+ name: string;
11
+ secret: boolean;
12
+ display: string;
13
+ }
14
+
15
+ export interface ListedRow {
16
+ engine: string;
17
+ name: string;
18
+ scope: 'global' | 'workspace';
19
+ fields: ListedField[];
20
+ last_modified_at: string;
21
+ }
22
+
23
+ export async function runDatasourceList(
24
+ bundle: VaultBundle,
25
+ filter: { engine?: string },
26
+ ): Promise<ListedRow[]> {
27
+ const all = await bundle.vault.list();
28
+ const out: ListedRow[] = [];
29
+ for (const row of all) {
30
+ if (filter.engine && row.engine !== filter.engine) continue;
31
+ const engine = bundle.registry.get(row.engine);
32
+ let fields: ListedField[] = [];
33
+ if (engine) {
34
+ const entry = await bundle.vault.get({ engine: row.engine, name: row.name });
35
+ fields = engine.fields
36
+ .map((f) => ({
37
+ name: f.name,
38
+ secret: f.secret,
39
+ display: f.secret ? '••••••' : (entry.fields[f.name] ?? ''),
40
+ }))
41
+ .filter((f) => row.fields_set.includes(f.name));
42
+ } else {
43
+ fields = row.fields_set.map((n) => ({ name: n, secret: false, display: '' }));
44
+ }
45
+ out.push({
46
+ engine: row.engine,
47
+ name: row.name,
48
+ scope: row.scope,
49
+ fields,
50
+ last_modified_at: row.last_modified_at,
51
+ });
52
+ }
53
+ return out;
54
+ }
55
+
56
+ export async function runDatasourceRemove(
57
+ bundle: VaultBundle,
58
+ args: { ref: string },
59
+ ): Promise<void> {
60
+ const ref = LocalDataVault.parseRef(args.ref);
61
+ await bundle.vault.remove(ref);
62
+ }
63
+
64
+ export interface TestPreview {
65
+ ref: string;
66
+ engine: string;
67
+ envVarNames: string[];
68
+ }
69
+
70
+ export async function runDatasourceTest(
71
+ bundle: VaultBundle,
72
+ args: { ref: string },
73
+ ): Promise<TestPreview> {
74
+ const ref = LocalDataVault.parseRef(args.ref);
75
+ const entry = await bundle.vault.get(ref);
76
+ return {
77
+ ref: args.ref,
78
+ engine: ref.engine,
79
+ envVarNames: Object.keys(entry.fields).map((f) => envVarName(ref.engine, ref.name, f)),
80
+ };
81
+ }
@@ -0,0 +1,12 @@
1
+ {
2
+ "id": "coworker-vault",
3
+ "name": "Co-worker Vault",
4
+ "version": "1.0.0",
5
+ "description": "Credential vault: /connect wizard, /datasource manager, /audit reader",
6
+ "tier": "bundled",
7
+ "requires": { "platform": ">=2.29.0" },
8
+ "provides": {
9
+ "commands": ["connect", "datasource", "audit"],
10
+ "hooks": ["session_shutdown"]
11
+ }
12
+ }