@vellumai/assistant 0.7.2 → 0.7.3

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 (347) hide show
  1. package/ARCHITECTURE.md +16 -1
  2. package/docs/architecture/memory.md +5 -2
  3. package/node_modules/@vellumai/gateway-client/src/ipc-client.ts +13 -4
  4. package/node_modules/@vellumai/skill-host-contracts/src/assistant-event.ts +0 -9
  5. package/node_modules/@vellumai/slack-text/src/index.test.ts +18 -35
  6. package/node_modules/@vellumai/slack-text/src/index.ts +2 -48
  7. package/openapi.yaml +449 -22
  8. package/package.json +1 -1
  9. package/src/__tests__/app-control-flow.test.ts +21 -11
  10. package/src/__tests__/assistant-event-hub.test.ts +48 -0
  11. package/src/__tests__/assistant-event.test.ts +0 -10
  12. package/src/__tests__/assistant-events-sse-hardening.test.ts +2 -7
  13. package/src/__tests__/assistant-feature-flags-integration.test.ts +18 -0
  14. package/src/__tests__/auto-analysis-end-to-end.test.ts +62 -1
  15. package/src/__tests__/background-workers-disk-pressure.test.ts +268 -0
  16. package/src/__tests__/call-conversation-messages.test.ts +8 -2
  17. package/src/__tests__/channel-inbound-disk-pressure.test.ts +537 -0
  18. package/src/__tests__/channel-readiness-service.test.ts +4 -2
  19. package/src/__tests__/config-loader-backfill.test.ts +379 -0
  20. package/src/__tests__/config-schema.test.ts +1 -0
  21. package/src/__tests__/config-watcher-cleanup-throttle.test.ts +18 -9
  22. package/src/__tests__/config-watcher.test.ts +140 -69
  23. package/src/__tests__/context-search-agent-runner.test.ts +61 -3
  24. package/src/__tests__/context-search-conversations-source.test.ts +0 -24
  25. package/src/__tests__/context-search-fanout.test.ts +0 -1
  26. package/src/__tests__/context-search-memory-source.test.ts +3 -7
  27. package/src/__tests__/context-search-memory-v2-source.test.ts +0 -2
  28. package/src/__tests__/context-search-pkb-source.test.ts +0 -1
  29. package/src/__tests__/context-search-workspace-source.test.ts +0 -1
  30. package/src/__tests__/conversation-abort-tool-results.test.ts +6 -0
  31. package/src/__tests__/conversation-agent-loop-disk-pressure.test.ts +223 -0
  32. package/src/__tests__/conversation-agent-loop.test.ts +454 -5
  33. package/src/__tests__/conversation-error.test.ts +150 -3
  34. package/src/__tests__/conversation-process-callsite.test.ts +43 -0
  35. package/src/__tests__/conversation-provider-retry-repair.test.ts +6 -0
  36. package/src/__tests__/conversation-runtime-assembly.test.ts +65 -0
  37. package/src/__tests__/conversation-slash-unknown.test.ts +6 -0
  38. package/src/__tests__/conversation-speed-override.test.ts +0 -3
  39. package/src/__tests__/conversation-store.test.ts +0 -18
  40. package/src/__tests__/conversation-surfaces-app-control.test.ts +15 -4
  41. package/src/__tests__/conversation-surfaces-data-persist.test.ts +404 -0
  42. package/src/__tests__/conversation-tool-setup-app-refresh.test.ts +2 -5
  43. package/src/__tests__/conversation-workspace-injection.test.ts +6 -0
  44. package/src/__tests__/conversation-workspace-tool-tracking.test.ts +6 -0
  45. package/src/__tests__/credentials-cli.test.ts +7 -0
  46. package/src/__tests__/cu-unified-flow.test.ts +176 -10
  47. package/src/__tests__/date-context.test.ts +164 -2
  48. package/src/__tests__/disk-pressure-guard.test.ts +262 -0
  49. package/src/__tests__/disk-pressure-lifecycle.test.ts +168 -0
  50. package/src/__tests__/disk-pressure-policy.test.ts +241 -0
  51. package/src/__tests__/disk-pressure-routes.test.ts +379 -0
  52. package/src/__tests__/disk-pressure-tools.test.ts +277 -0
  53. package/src/__tests__/disk-usage.test.ts +150 -0
  54. package/src/__tests__/events-client-registration.test.ts +52 -0
  55. package/src/__tests__/events-dev-bypass-actor.test.ts +162 -0
  56. package/src/__tests__/file-write-tool.test.ts +4 -10
  57. package/src/__tests__/filing-service.test.ts +3 -4
  58. package/src/__tests__/heartbeat-disk-pressure.test.ts +183 -0
  59. package/src/__tests__/heartbeat-service.test.ts +260 -11
  60. package/src/__tests__/host-app-control-proxy.test.ts +195 -25
  61. package/src/__tests__/host-bash-proxy.test.ts +227 -34
  62. package/src/__tests__/host-bash-routes.test.ts +178 -13
  63. package/src/__tests__/host-cu-proxy.test.ts +210 -3
  64. package/src/__tests__/host-cu-routes-targeted.test.ts +141 -12
  65. package/src/__tests__/host-file-proxy-targeted.test.ts +48 -9
  66. package/src/__tests__/host-file-proxy.test.ts +268 -6
  67. package/src/__tests__/host-file-routes-targeted.test.ts +175 -17
  68. package/src/__tests__/host-transfer-proxy-targeted.test.ts +408 -59
  69. package/src/__tests__/host-transfer-routes-targeted.test.ts +232 -17
  70. package/src/__tests__/http-user-message-parity.test.ts +107 -1
  71. package/src/__tests__/injector-chain.test.ts +18 -6
  72. package/src/__tests__/injector-disk-pressure.test.ts +224 -0
  73. package/src/__tests__/managed-profile-guard.test.ts +18 -0
  74. package/src/__tests__/mcp-abort-signal.test.ts +130 -0
  75. package/src/__tests__/memory-admin-recall.test.ts +3 -11
  76. package/src/__tests__/memory-retrieval-pipeline.test.ts +22 -1
  77. package/src/__tests__/normalize-onboarding.test.ts +180 -0
  78. package/src/__tests__/oauth-connect-routes.test.ts +316 -0
  79. package/src/__tests__/oauth-provider-seed-logos.test.ts +24 -2
  80. package/src/__tests__/onboarding-persona-write.test.ts +308 -0
  81. package/src/__tests__/openai-provider.test.ts +45 -8
  82. package/src/__tests__/persist-onboarding-artifacts.test.ts +44 -64
  83. package/src/__tests__/platform-callback-registration.test.ts +21 -4
  84. package/src/__tests__/platform.test.ts +2 -1
  85. package/src/__tests__/playbook-execution.test.ts +0 -43
  86. package/src/__tests__/plugin-tool-contribution.test.ts +47 -0
  87. package/src/__tests__/prechat-onboarding-contract.test.ts +214 -27
  88. package/src/__tests__/provider-tool-name.test.ts +23 -0
  89. package/src/__tests__/relay-server.test.ts +15 -4
  90. package/src/__tests__/runtime-events-sse.test.ts +4 -8
  91. package/src/__tests__/scheduler-disk-pressure.test.ts +148 -0
  92. package/src/__tests__/secret-ingress-http.test.ts +0 -1
  93. package/src/__tests__/suggestion-routes.test.ts +46 -0
  94. package/src/__tests__/twilio-validation.test.ts +2 -2
  95. package/src/__tests__/workspace-migration-065-bump-stale-heartbeat-interval.test.ts +122 -0
  96. package/src/__tests__/workspace-migration-066-seed-heartbeat-callsite-cost-default.test.ts +285 -0
  97. package/src/__tests__/workspace-migration-068-release-notes-local-timezone.test.ts +90 -0
  98. package/src/__tests__/workspace-migration-safe-storage-limits-release.test.ts +90 -0
  99. package/src/approvals/guardian-decision-primitive.ts +13 -0
  100. package/src/approvals/guardian-request-resolvers.ts +16 -17
  101. package/src/backup/snapshot-lock.ts +2 -27
  102. package/src/bundler/compiler-tools.ts +3 -2
  103. package/src/calls/call-conversation-messages.ts +46 -10
  104. package/src/cli/commands/__tests__/webhooks.test.ts +0 -4
  105. package/src/cli/commands/bash.ts +35 -108
  106. package/src/cli/commands/contacts.ts +64 -25
  107. package/src/cli/commands/credentials.ts +56 -0
  108. package/src/cli/commands/memory-v2.ts +7 -6
  109. package/src/cli/commands/oauth/__tests__/connect.test.ts +437 -1
  110. package/src/cli/commands/oauth/connect.ts +127 -1
  111. package/src/cli/commands/platform/__tests__/callback-routes-list.test.ts +0 -3
  112. package/src/cli/commands/platform/__tests__/connect.test.ts +7 -1
  113. package/src/cli/commands/platform/__tests__/disconnect.test.ts +7 -1
  114. package/src/cli/commands/platform/__tests__/status.test.ts +103 -6
  115. package/src/cli/commands/platform/index.ts +16 -7
  116. package/src/cli/commands/status.ts +57 -0
  117. package/src/cli/program.ts +4 -2
  118. package/src/config/assistant-feature-flags.ts +13 -3
  119. package/src/config/bundled-skills/messaging/tools/messaging-analyze-style.ts +4 -3
  120. package/src/config/bundled-skills/phone-calls/references/TROUBLESHOOTING.md +13 -7
  121. package/src/config/bundled-skills/playbooks/tools/playbook-create.ts +2 -2
  122. package/src/config/bundled-skills/playbooks/tools/playbook-delete.ts +2 -2
  123. package/src/config/bundled-skills/playbooks/tools/playbook-list.ts +2 -2
  124. package/src/config/bundled-skills/playbooks/tools/playbook-update.ts +2 -2
  125. package/src/config/env.ts +0 -8
  126. package/src/config/feature-flag-registry.json +27 -3
  127. package/src/config/loader.ts +127 -8
  128. package/src/config/schemas/__tests__/memory-v2.test.ts +10 -5
  129. package/src/config/schemas/call-site-catalog.ts +14 -0
  130. package/src/config/schemas/channels.ts +0 -5
  131. package/src/config/schemas/heartbeat.ts +1 -1
  132. package/src/config/schemas/llm.ts +2 -0
  133. package/src/config/schemas/memory-lifecycle.ts +13 -0
  134. package/src/config/schemas/memory-v2.ts +75 -11
  135. package/src/config/schemas/platform.ts +43 -3
  136. package/src/config/schemas/services.ts +28 -0
  137. package/src/config/seed-inference-profiles.ts +230 -33
  138. package/src/contacts/contact-store.ts +0 -25
  139. package/src/daemon/__tests__/conversation-tool-setup.test.ts +86 -25
  140. package/src/daemon/assistant-attachments.ts +4 -4
  141. package/src/daemon/config-watcher.ts +85 -57
  142. package/src/daemon/conversation-agent-loop-handlers.ts +6 -0
  143. package/src/daemon/conversation-agent-loop.ts +170 -33
  144. package/src/daemon/conversation-error.ts +87 -15
  145. package/src/daemon/conversation-lifecycle.ts +1 -3
  146. package/src/daemon/conversation-process.ts +8 -0
  147. package/src/daemon/conversation-runtime-assembly.ts +26 -0
  148. package/src/daemon/conversation-store.ts +2 -2
  149. package/src/daemon/conversation-surfaces.ts +195 -15
  150. package/src/daemon/conversation-tool-setup.ts +57 -14
  151. package/src/daemon/conversation.ts +17 -22
  152. package/src/daemon/date-context.ts +71 -22
  153. package/src/daemon/disk-pressure-background-gate.ts +73 -0
  154. package/src/daemon/disk-pressure-guard.ts +343 -0
  155. package/src/daemon/disk-pressure-policy.ts +163 -0
  156. package/src/daemon/handlers/shared.ts +0 -1
  157. package/src/daemon/handlers/skills.ts +3 -4
  158. package/src/daemon/host-app-control-proxy.ts +137 -41
  159. package/src/daemon/host-bash-proxy.ts +46 -21
  160. package/src/daemon/host-cu-proxy.ts +49 -3
  161. package/src/daemon/host-file-proxy.ts +43 -7
  162. package/src/daemon/host-transfer-proxy.ts +95 -4
  163. package/src/daemon/lifecycle.ts +79 -28
  164. package/src/daemon/meet-host-supervisor.ts +4 -4
  165. package/src/daemon/meet-manifest-loader.ts +0 -1
  166. package/src/daemon/memory-v2-startup.ts +14 -4
  167. package/src/daemon/message-protocol.ts +3 -0
  168. package/src/daemon/message-types/conversations.ts +4 -0
  169. package/src/daemon/message-types/disk-pressure.ts +9 -0
  170. package/src/daemon/message-types/messages.ts +3 -0
  171. package/src/daemon/profiler-run-store.ts +5 -5
  172. package/src/daemon/tool-setup-types.ts +2 -2
  173. package/src/documents/document-store.ts +85 -0
  174. package/src/filing/filing-service.ts +30 -5
  175. package/src/heartbeat/__tests__/heartbeat-feed-event.test.ts +9 -16
  176. package/src/heartbeat/__tests__/heartbeat-run-store.test.ts +36 -0
  177. package/src/heartbeat/heartbeat-run-store.ts +13 -0
  178. package/src/heartbeat/heartbeat-service.ts +205 -31
  179. package/src/home/feed-scheduler.ts +18 -0
  180. package/src/inbound/platform-callback-registration.ts +8 -15
  181. package/src/ipc/__tests__/clients-list-ipc.test.ts +169 -0
  182. package/src/ipc/assistant-server.ts +56 -2
  183. package/src/ipc/gateway-client.ts +37 -3
  184. package/src/live-voice/live-voice-archive.ts +4 -4
  185. package/src/live-voice/protocol.ts +5 -7
  186. package/src/media/image-service.ts +1 -7
  187. package/src/memory/__tests__/fixtures/memory-v2-activation-fixtures.ts +21 -13
  188. package/src/memory/__tests__/jobs-worker-v2-schedule.test.ts +52 -22
  189. package/src/memory/__tests__/memory-v2-activation-log-store.test.ts +0 -6
  190. package/src/memory/__tests__/memory-v2-concept-frequency.test.ts +272 -0
  191. package/src/memory/admin.ts +5 -9
  192. package/src/memory/context-search/agent-runner.ts +19 -2
  193. package/src/memory/context-search/sources/conversations.ts +2 -11
  194. package/src/memory/context-search/sources/memory-v2.ts +5 -4
  195. package/src/memory/context-search/sources/memory.ts +0 -1
  196. package/src/memory/context-search/types.ts +0 -1
  197. package/src/memory/conversation-crud.ts +4 -12
  198. package/src/memory/db-init.ts +2 -0
  199. package/src/memory/embedding-runtime-manager.ts +119 -5
  200. package/src/memory/graph/__tests__/conversation-graph-memory-v2-routing.test.ts +32 -21
  201. package/src/memory/graph/conversation-graph-memory.ts +42 -54
  202. package/src/memory/graph/extraction.ts +1 -3
  203. package/src/memory/graph/graph-search.test.ts +10 -67
  204. package/src/memory/graph/graph-search.ts +1 -20
  205. package/src/memory/graph/retriever.test.ts +6 -0
  206. package/src/memory/graph/retriever.ts +6 -10
  207. package/src/memory/indexer.ts +54 -45
  208. package/src/memory/job-handlers/backfill.ts +2 -11
  209. package/src/memory/job-handlers/cleanup.ts +43 -0
  210. package/src/memory/job-handlers/embedding.ts +6 -8
  211. package/src/memory/job-handlers/summarization.ts +2 -7
  212. package/src/memory/jobs-store.ts +48 -0
  213. package/src/memory/jobs-worker.ts +81 -43
  214. package/src/memory/memory-v2-activation-log-store.ts +32 -14
  215. package/src/memory/memory-v2-concept-frequency.ts +169 -0
  216. package/src/memory/migrations/239-trace-events-created-at-index.ts +18 -0
  217. package/src/memory/migrations/index.ts +1 -0
  218. package/src/memory/pkb/pkb-search.test.ts +6 -0
  219. package/src/memory/qdrant-client.ts +0 -13
  220. package/src/memory/rerank-local.ts +374 -0
  221. package/src/memory/search/semantic.ts +6 -67
  222. package/src/memory/trace-event-store.ts +1 -17
  223. package/src/memory/v2/__tests__/activation.test.ts +311 -250
  224. package/src/memory/v2/__tests__/consolidation-job.test.ts +40 -8
  225. package/src/memory/v2/__tests__/injection.test.ts +157 -167
  226. package/src/memory/v2/__tests__/prompts-consolidation.test.ts +61 -2
  227. package/src/memory/v2/__tests__/qdrant.test.ts +16 -0
  228. package/src/memory/v2/__tests__/reranker.test.ts +338 -0
  229. package/src/memory/v2/__tests__/sim.test.ts +5 -199
  230. package/src/memory/v2/__tests__/skill-store.test.ts +71 -65
  231. package/src/memory/v2/__tests__/static-context.test.ts +76 -1
  232. package/src/memory/v2/activation.ts +149 -156
  233. package/src/memory/v2/consolidation-job.ts +62 -12
  234. package/src/memory/v2/injection.ts +47 -60
  235. package/src/memory/v2/prompts/consolidation.ts +36 -1
  236. package/src/memory/v2/qdrant.ts +99 -0
  237. package/src/memory/v2/reranker.ts +177 -0
  238. package/src/memory/v2/sim.ts +10 -84
  239. package/src/memory/v2/skill-content.ts +4 -3
  240. package/src/memory/v2/skill-store.ts +82 -59
  241. package/src/memory/v2/static-context.ts +22 -0
  242. package/src/memory/v2/types.ts +10 -10
  243. package/src/notifications/copy-composer.ts +13 -0
  244. package/src/notifications/signal.ts +4 -0
  245. package/src/oauth/AGENTS.md +3 -1
  246. package/src/oauth/__tests__/oauth-connect-state.test.ts +137 -0
  247. package/src/oauth/connect-orchestrator.ts +2 -0
  248. package/src/oauth/connection-resolver.test.ts +66 -1
  249. package/src/oauth/connection-resolver.ts +55 -1
  250. package/src/oauth/oauth-connect-state.ts +77 -0
  251. package/src/oauth/seed-providers.ts +58 -1
  252. package/src/plugins/defaults/injectors.ts +35 -2
  253. package/src/plugins/defaults/memory-retrieval.ts +5 -6
  254. package/src/plugins/types.ts +7 -0
  255. package/src/proactive-artifact/aux-message-injector.ts +74 -0
  256. package/src/proactive-artifact/decision.test.ts +226 -0
  257. package/src/proactive-artifact/decision.ts +165 -0
  258. package/src/proactive-artifact/index.ts +7 -0
  259. package/src/proactive-artifact/job.test.ts +867 -0
  260. package/src/proactive-artifact/job.ts +352 -0
  261. package/src/proactive-artifact/message-copy.ts +41 -0
  262. package/src/proactive-artifact/trigger-state.test.ts +277 -0
  263. package/src/proactive-artifact/trigger-state.ts +119 -0
  264. package/src/prompts/normalize-onboarding.ts +80 -0
  265. package/src/prompts/persona-resolver.ts +101 -9
  266. package/src/prompts/system-prompt.ts +21 -7
  267. package/src/prompts/templates/BOOTSTRAP.md +13 -5
  268. package/src/providers/__tests__/retry-callsite.test.ts +222 -1
  269. package/src/providers/model-intents.ts +7 -0
  270. package/src/providers/openrouter/client.ts +8 -0
  271. package/src/providers/retry.ts +50 -0
  272. package/src/providers/types.ts +1 -0
  273. package/src/runtime/__tests__/agent-wake.test.ts +456 -3
  274. package/src/runtime/agent-wake.ts +238 -100
  275. package/src/runtime/assistant-event-hub.ts +36 -6
  276. package/src/runtime/assistant-event.ts +0 -1
  277. package/src/runtime/auth/__tests__/route-policy.test.ts +64 -0
  278. package/src/runtime/auth/route-policy.ts +14 -1
  279. package/src/runtime/auth/same-actor.ts +216 -0
  280. package/src/runtime/channel-retry-sweep.ts +65 -1
  281. package/src/runtime/guardian-reply-router.ts +10 -0
  282. package/src/runtime/local-actor-identity.ts +52 -11
  283. package/src/runtime/pending-interactions.ts +8 -0
  284. package/src/runtime/routes/__tests__/client-routes.test.ts +155 -0
  285. package/src/runtime/routes/__tests__/conversation-query-routes.test.ts +0 -5
  286. package/src/runtime/routes/__tests__/heartbeat-routes.test.ts +1 -1
  287. package/src/runtime/routes/client-routes.ts +20 -2
  288. package/src/runtime/routes/contact-routes.ts +0 -25
  289. package/src/runtime/routes/conversation-routes.ts +35 -26
  290. package/src/runtime/routes/debug-bash-routes.ts +163 -0
  291. package/src/runtime/routes/disk-pressure-routes.ts +121 -0
  292. package/src/runtime/routes/document-pdf-renderer.ts +6 -2
  293. package/src/runtime/routes/documents-routes.ts +2 -75
  294. package/src/runtime/routes/events-routes.ts +41 -9
  295. package/src/runtime/routes/host-bash-routes.ts +23 -3
  296. package/src/runtime/routes/host-cu-routes.ts +33 -6
  297. package/src/runtime/routes/host-file-routes.ts +32 -6
  298. package/src/runtime/routes/host-transfer-routes.ts +79 -16
  299. package/src/runtime/routes/identity-routes.ts +7 -138
  300. package/src/runtime/routes/inbound-message-handler.ts +77 -12
  301. package/src/runtime/routes/inbound-stages/guardian-reply-intercept.ts +3 -0
  302. package/src/runtime/routes/index.ts +6 -0
  303. package/src/runtime/routes/memory-item-routes.test.ts +41 -15
  304. package/src/runtime/routes/memory-v2-routes.ts +33 -0
  305. package/src/runtime/routes/oauth-connect-routes.ts +153 -0
  306. package/src/runtime/verification-outbound-actions.ts +4 -4
  307. package/src/schedule/run-script.ts +37 -5
  308. package/src/schedule/scheduler.ts +20 -1
  309. package/src/security/encrypted-store.ts +2 -0
  310. package/src/security/secure-keys.ts +55 -0
  311. package/src/skills/remote-skill-policy.ts +4 -10
  312. package/src/subagent/index.ts +1 -7
  313. package/src/subagent/manager.ts +1 -15
  314. package/src/tasks/task-runner.ts +0 -1
  315. package/src/tasks/task-store.ts +0 -3
  316. package/src/tools/background-tool-registry.ts +17 -3
  317. package/src/tools/host-filesystem/edit.test.ts +151 -0
  318. package/src/tools/host-filesystem/edit.ts +43 -1
  319. package/src/tools/host-filesystem/read.test.ts +129 -0
  320. package/src/tools/host-filesystem/read.ts +43 -1
  321. package/src/tools/host-filesystem/transfer.test.ts +127 -2
  322. package/src/tools/host-filesystem/transfer.ts +56 -11
  323. package/src/tools/host-filesystem/write.test.ts +134 -0
  324. package/src/tools/host-filesystem/write.ts +43 -1
  325. package/src/tools/host-terminal/host-shell.ts +13 -6
  326. package/src/tools/mcp/mcp-tool-factory.ts +2 -1
  327. package/src/tools/memory/register.test.ts +12 -9
  328. package/src/tools/memory/register.ts +1 -2
  329. package/src/tools/provider-tool-name.ts +28 -0
  330. package/src/tools/registry.ts +30 -9
  331. package/src/tools/terminal/shell.ts +9 -1
  332. package/src/tools/tool-approval-handler.ts +31 -6
  333. package/src/tools/types.ts +24 -2
  334. package/src/tts/provider-catalog.ts +3 -5
  335. package/src/util/disk-usage.ts +138 -0
  336. package/src/util/platform.ts +21 -11
  337. package/src/util/process-liveness.ts +26 -0
  338. package/src/workspace/heartbeat-service.ts +19 -0
  339. package/src/workspace/migrations/065-bump-stale-heartbeat-interval.ts +60 -0
  340. package/src/workspace/migrations/066-seed-heartbeat-callsite-cost-default.ts +146 -0
  341. package/src/workspace/migrations/067-release-notes-safe-storage-limits.ts +72 -0
  342. package/src/workspace/migrations/068-release-notes-local-timezone.ts +65 -0
  343. package/src/workspace/migrations/registry.ts +8 -0
  344. package/src/__tests__/conversation-tool-setup-memory-scope.test.ts +0 -167
  345. package/src/memory/v2/__tests__/skill-qdrant.test.ts +0 -657
  346. package/src/memory/v2/skill-qdrant.ts +0 -404
  347. package/src/signals/bash.ts +0 -198
@@ -0,0 +1,226 @@
1
+ import { describe, expect, test } from "bun:test";
2
+
3
+ import {
4
+ type DecisionOutput,
5
+ formatTranscript,
6
+ parseDecisionOutput,
7
+ } from "./decision.js";
8
+
9
+ describe("parseDecisionOutput", () => {
10
+ test("parses SHOULD_BUILD: yes with all fields correctly", () => {
11
+ const text = `SHOULD_BUILD: yes
12
+ ARTIFACT_TYPE: app
13
+ ARTIFACT_TITLE: Sarah's Marathon Training Pace Calculator
14
+ ARTIFACT_DESCRIPTION: An interactive pace calculator that accounts for Sarah's goal of a sub-4-hour marathon, her current 9:30/mile easy pace, and the hilly terrain of the Boston course.`;
15
+
16
+ const result = parseDecisionOutput(text);
17
+ expect(result).toEqual({
18
+ shouldBuild: true,
19
+ artifactType: "app",
20
+ artifactTitle: "Sarah's Marathon Training Pace Calculator",
21
+ artifactDescription:
22
+ "An interactive pace calculator that accounts for Sarah's goal of a sub-4-hour marathon, her current 9:30/mile easy pace, and the hilly terrain of the Boston course.",
23
+ });
24
+ });
25
+
26
+ test("parses SHOULD_BUILD: no with skip reason", () => {
27
+ const text = `SHOULD_BUILD: no
28
+ SKIP_REASON: The conversation is too early and generic — the user has only asked a factual question with no personal context.`;
29
+
30
+ const result = parseDecisionOutput(text);
31
+ expect(result).toEqual({
32
+ shouldBuild: false,
33
+ skipReason:
34
+ "The conversation is too early and generic — the user has only asked a factual question with no personal context.",
35
+ });
36
+ });
37
+
38
+ test("parses SHOULD_BUILD: no without skip reason defaults to 'no reason given'", () => {
39
+ const text = `SHOULD_BUILD: no`;
40
+
41
+ const result = parseDecisionOutput(text);
42
+ expect(result).toEqual({
43
+ shouldBuild: false,
44
+ skipReason: "no reason given",
45
+ });
46
+ });
47
+
48
+ test("returns null for missing SHOULD_BUILD line", () => {
49
+ const text = `ARTIFACT_TYPE: app
50
+ ARTIFACT_TITLE: Some Title
51
+ ARTIFACT_DESCRIPTION: Some description.`;
52
+
53
+ const result = parseDecisionOutput(text);
54
+ expect(result).toBeNull();
55
+ });
56
+
57
+ test("returns null when yes but ARTIFACT_TYPE is missing", () => {
58
+ const text = `SHOULD_BUILD: yes
59
+ ARTIFACT_TITLE: Some Title
60
+ ARTIFACT_DESCRIPTION: Some description.`;
61
+
62
+ const result = parseDecisionOutput(text);
63
+ expect(result).toBeNull();
64
+ });
65
+
66
+ test("returns null when yes but ARTIFACT_TYPE is invalid", () => {
67
+ const text = `SHOULD_BUILD: yes
68
+ ARTIFACT_TYPE: widget
69
+ ARTIFACT_TITLE: Some Title
70
+ ARTIFACT_DESCRIPTION: Some description.`;
71
+
72
+ const result = parseDecisionOutput(text);
73
+ expect(result).toBeNull();
74
+ });
75
+
76
+ test("returns null when yes but ARTIFACT_TITLE is missing", () => {
77
+ const text = `SHOULD_BUILD: yes
78
+ ARTIFACT_TYPE: document
79
+ ARTIFACT_DESCRIPTION: Some description.`;
80
+
81
+ const result = parseDecisionOutput(text);
82
+ expect(result).toBeNull();
83
+ });
84
+
85
+ test("returns null when yes but ARTIFACT_DESCRIPTION is missing", () => {
86
+ const text = `SHOULD_BUILD: yes
87
+ ARTIFACT_TYPE: document
88
+ ARTIFACT_TITLE: Some Title`;
89
+
90
+ const result = parseDecisionOutput(text);
91
+ expect(result).toBeNull();
92
+ });
93
+
94
+ test("returns null when yes but ARTIFACT_TITLE is empty", () => {
95
+ const text = `SHOULD_BUILD: yes
96
+ ARTIFACT_TYPE: app
97
+ ARTIFACT_TITLE:
98
+ ARTIFACT_DESCRIPTION: Some description.`;
99
+
100
+ const result = parseDecisionOutput(text);
101
+ expect(result).toBeNull();
102
+ });
103
+
104
+ test("handles multi-line ARTIFACT_DESCRIPTION", () => {
105
+ const text = `SHOULD_BUILD: yes
106
+ ARTIFACT_TYPE: document
107
+ ARTIFACT_TITLE: Jake's Q3 OKR Tracker
108
+ ARTIFACT_DESCRIPTION: A structured comparison table for Jake's three competing priorities:
109
+ scaling the data pipeline from 10M to 50M events/day,
110
+ hiring two senior engineers by September,
111
+ and reducing p99 latency below 200ms.`;
112
+
113
+ const result = parseDecisionOutput(text) as DecisionOutput & {
114
+ shouldBuild: true;
115
+ };
116
+ expect(result).not.toBeNull();
117
+ expect(result.shouldBuild).toBe(true);
118
+ expect(result.artifactType).toBe("document");
119
+ expect(result.artifactTitle).toBe("Jake's Q3 OKR Tracker");
120
+ expect(result.artifactDescription).toContain(
121
+ "A structured comparison table",
122
+ );
123
+ expect(result.artifactDescription).toContain(
124
+ "reducing p99 latency below 200ms",
125
+ );
126
+ });
127
+
128
+ test("handles case-insensitive SHOULD_BUILD value", () => {
129
+ const text = `SHOULD_BUILD: Yes
130
+ ARTIFACT_TYPE: app
131
+ ARTIFACT_TITLE: Test Title
132
+ ARTIFACT_DESCRIPTION: Test description.`;
133
+
134
+ const result = parseDecisionOutput(text);
135
+ expect(result).not.toBeNull();
136
+ expect(result!.shouldBuild).toBe(true);
137
+ });
138
+
139
+ test("handles case-insensitive ARTIFACT_TYPE value", () => {
140
+ const text = `SHOULD_BUILD: yes
141
+ ARTIFACT_TYPE: Document
142
+ ARTIFACT_TITLE: Test Title
143
+ ARTIFACT_DESCRIPTION: Test description.`;
144
+
145
+ const result = parseDecisionOutput(text);
146
+ expect(result).not.toBeNull();
147
+ expect((result as { artifactType: string }).artifactType).toBe("document");
148
+ });
149
+ });
150
+
151
+ describe("formatTranscript", () => {
152
+ test("formats plain text messages correctly", () => {
153
+ const messages = [
154
+ { role: "user", content: "Hello, I need help with my project" },
155
+ {
156
+ role: "assistant",
157
+ content: "I'd be happy to help! What kind of project?",
158
+ },
159
+ {
160
+ role: "user",
161
+ content: "I'm building a fitness tracker for marathon training",
162
+ },
163
+ ];
164
+
165
+ const result = formatTranscript(messages);
166
+ expect(result).toBe(
167
+ `[User]: Hello, I need help with my project
168
+
169
+ [Assistant]: I'd be happy to help! What kind of project?
170
+
171
+ [User]: I'm building a fitness tracker for marathon training`,
172
+ );
173
+ });
174
+
175
+ test("handles JSON content blocks", () => {
176
+ const messages = [
177
+ {
178
+ role: "user",
179
+ content: JSON.stringify([{ type: "text", text: "What is 2+2?" }]),
180
+ },
181
+ {
182
+ role: "assistant",
183
+ content: JSON.stringify([
184
+ { type: "text", text: "The answer is 4." },
185
+ { type: "text", text: "Would you like to know more?" },
186
+ ]),
187
+ },
188
+ ];
189
+
190
+ const result = formatTranscript(messages);
191
+ expect(result).toBe(
192
+ `[User]: What is 2+2?
193
+
194
+ [Assistant]: The answer is 4.
195
+ Would you like to know more?`,
196
+ );
197
+ });
198
+
199
+ test("handles mixed JSON and plain text messages", () => {
200
+ const messages = [
201
+ {
202
+ role: "user",
203
+ content: JSON.stringify([
204
+ { type: "text", text: "Help me plan my week" },
205
+ ]),
206
+ },
207
+ { role: "assistant", content: "Sure, what do you have coming up?" },
208
+ ];
209
+
210
+ const result = formatTranscript(messages);
211
+ expect(result).toBe(
212
+ `[User]: Help me plan my week
213
+
214
+ [Assistant]: Sure, what do you have coming up?`,
215
+ );
216
+ });
217
+
218
+ test("handles unknown roles", () => {
219
+ const messages = [
220
+ { role: "system", content: "You are a helpful assistant" },
221
+ ];
222
+
223
+ const result = formatTranscript(messages);
224
+ expect(result).toBe("[system]: You are a helpful assistant");
225
+ });
226
+ });
@@ -0,0 +1,165 @@
1
+ export type DecisionOutput =
2
+ | { shouldBuild: false; skipReason: string }
3
+ | {
4
+ shouldBuild: true;
5
+ artifactType: "app" | "document";
6
+ artifactTitle: string;
7
+ artifactDescription: string;
8
+ };
9
+
10
+ export function buildDecisionPrompt(transcript: string): string {
11
+ return `You are deciding whether to proactively build a personalized artifact for the user based on their conversation so far.
12
+
13
+ Read the conversation below carefully. Your job:
14
+ 1. Identify what the user cares about — their goals, context, specific details they've shared.
15
+ 2. Decide: should we build a small interactive app or document that would delight this specific user?
16
+ 3. Quality test: Could you have built the same thing for any random person? If yes, too generic — output SHOULD_BUILD: no.
17
+
18
+ Rules:
19
+ - Only say yes if you can build something SPECIFIC to this user's situation, using details from their conversation.
20
+ - An "app" is a small interactive web application (calculator, tracker, visualizer, planner, etc.)
21
+ - A "document" is a structured reference (checklist, guide, comparison table, template, etc.)
22
+ - The title and description must reference specifics from the conversation — names, numbers, goals, constraints the user mentioned.
23
+ - Do NOT include a MESSAGE field.
24
+
25
+ Conversation:
26
+ ${transcript}
27
+
28
+ Respond in EXACTLY this format (no extra text before or after):
29
+
30
+ SHOULD_BUILD: [yes|no]
31
+ SKIP_REASON: [required if no — why this conversation isn't a good fit]
32
+ ARTIFACT_TYPE: [app|document]
33
+ ARTIFACT_TITLE: [specific title seeded with user context]
34
+ ARTIFACT_DESCRIPTION: [1-3 sentence build spec with user-specific details]
35
+
36
+ If SHOULD_BUILD is no, omit ARTIFACT_TYPE, ARTIFACT_TITLE, and ARTIFACT_DESCRIPTION.
37
+ If SHOULD_BUILD is yes, omit SKIP_REASON.`;
38
+ }
39
+
40
+ export function parseDecisionOutput(text: string): DecisionOutput | null {
41
+ const lines = text.trim().split("\n");
42
+
43
+ const shouldBuildLine = lines.find((line) =>
44
+ line.trim().startsWith("SHOULD_BUILD:"),
45
+ );
46
+ if (!shouldBuildLine) return null;
47
+
48
+ const shouldBuildValue = shouldBuildLine
49
+ .split(":")
50
+ .slice(1)
51
+ .join(":")
52
+ .trim()
53
+ .toLowerCase();
54
+
55
+ if (shouldBuildValue === "no") {
56
+ const skipReasonLine = lines.find((line) =>
57
+ line.trim().startsWith("SKIP_REASON:"),
58
+ );
59
+ const skipReason = skipReasonLine
60
+ ? skipReasonLine.split(":").slice(1).join(":").trim()
61
+ : "no reason given";
62
+ return { shouldBuild: false, skipReason };
63
+ }
64
+
65
+ if (shouldBuildValue === "yes") {
66
+ const artifactTypeLine = lines.find((line) =>
67
+ line.trim().startsWith("ARTIFACT_TYPE:"),
68
+ );
69
+ if (!artifactTypeLine) return null;
70
+ const artifactType = artifactTypeLine
71
+ .split(":")
72
+ .slice(1)
73
+ .join(":")
74
+ .trim()
75
+ .toLowerCase();
76
+ if (artifactType !== "app" && artifactType !== "document") return null;
77
+
78
+ const artifactTitleLine = lines.find((line) =>
79
+ line.trim().startsWith("ARTIFACT_TITLE:"),
80
+ );
81
+ if (!artifactTitleLine) return null;
82
+ const artifactTitle = artifactTitleLine
83
+ .split(":")
84
+ .slice(1)
85
+ .join(":")
86
+ .trim();
87
+ if (!artifactTitle) return null;
88
+
89
+ const artifactDescriptionStartIndex = lines.findIndex((line) =>
90
+ line.trim().startsWith("ARTIFACT_DESCRIPTION:"),
91
+ );
92
+ if (artifactDescriptionStartIndex === -1) return null;
93
+
94
+ const firstDescLine = lines[artifactDescriptionStartIndex]
95
+ .split(":")
96
+ .slice(1)
97
+ .join(":")
98
+ .trim();
99
+
100
+ // Collect continuation lines (lines after ARTIFACT_DESCRIPTION that aren't other fields)
101
+ const descriptionParts = [firstDescLine];
102
+ for (let i = artifactDescriptionStartIndex + 1; i < lines.length; i++) {
103
+ const line = lines[i].trim();
104
+ if (
105
+ line.startsWith("SHOULD_BUILD:") ||
106
+ line.startsWith("SKIP_REASON:") ||
107
+ line.startsWith("ARTIFACT_TYPE:") ||
108
+ line.startsWith("ARTIFACT_TITLE:")
109
+ ) {
110
+ break;
111
+ }
112
+ descriptionParts.push(line);
113
+ }
114
+
115
+ const artifactDescription = descriptionParts.join("\n").trim();
116
+ if (!artifactDescription) return null;
117
+
118
+ return {
119
+ shouldBuild: true,
120
+ artifactType: artifactType as "app" | "document",
121
+ artifactTitle,
122
+ artifactDescription,
123
+ };
124
+ }
125
+
126
+ return null;
127
+ }
128
+
129
+ export function formatTranscript(
130
+ messages: Array<{ role: string; content: string }>,
131
+ ): string {
132
+ return messages
133
+ .map((msg) => {
134
+ const label =
135
+ msg.role === "user"
136
+ ? "[User]"
137
+ : msg.role === "assistant"
138
+ ? "[Assistant]"
139
+ : `[${msg.role}]`;
140
+ const content = parseContent(msg.content);
141
+ return `${label}: ${content}`;
142
+ })
143
+ .join("\n\n");
144
+ }
145
+
146
+ function parseContent(content: string): string {
147
+ // Try to parse as JSON content block array
148
+ try {
149
+ const parsed = JSON.parse(content);
150
+ if (Array.isArray(parsed)) {
151
+ return parsed
152
+ .map((block) => {
153
+ if (typeof block === "string") return block;
154
+ if (block.type === "text" && typeof block.text === "string")
155
+ return block.text;
156
+ return "";
157
+ })
158
+ .filter(Boolean)
159
+ .join("\n");
160
+ }
161
+ } catch {
162
+ // Not JSON, treat as plain text
163
+ }
164
+ return content;
165
+ }
@@ -0,0 +1,7 @@
1
+ export { runProactiveArtifactJob } from "./job.js";
2
+ export {
3
+ backfillGuardIfNeeded,
4
+ hasProactiveArtifactCompleted,
5
+ releaseProactiveArtifactClaim,
6
+ tryClaimProactiveArtifactTrigger,
7
+ } from "./trigger-state.js";