@vellumai/assistant 0.5.12 → 0.5.14

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 (442) hide show
  1. package/.env.example +1 -6
  2. package/AGENTS.md +4 -0
  3. package/ARCHITECTURE.md +0 -1
  4. package/Dockerfile +41 -9
  5. package/bunfig.toml +1 -0
  6. package/docs/architecture/memory.md +3 -3
  7. package/node_modules/@vellumai/ces-contracts/src/index.ts +7 -0
  8. package/node_modules/@vellumai/ces-contracts/src/rpc.ts +5 -0
  9. package/openapi.yaml +127 -22
  10. package/package.json +1 -1
  11. package/src/__tests__/access-request-decision.test.ts +2 -32
  12. package/src/__tests__/actor-token-service.test.ts +1 -31
  13. package/src/__tests__/anthropic-provider.test.ts +53 -40
  14. package/src/__tests__/app-git-history.test.ts +9 -17
  15. package/src/__tests__/app-git-service.test.ts +14 -20
  16. package/src/__tests__/app-store-dir-names.test.ts +10 -20
  17. package/src/__tests__/approval-cascade.test.ts +2 -19
  18. package/src/__tests__/approval-primitive.test.ts +2 -27
  19. package/src/__tests__/approval-routes-http.test.ts +2 -30
  20. package/src/__tests__/assistant-events-sse-hardening.test.ts +2 -28
  21. package/src/__tests__/assistant-feature-flags-integration.test.ts +2 -45
  22. package/src/__tests__/attachments-store.test.ts +5 -32
  23. package/src/__tests__/audit-log-rotation.test.ts +5 -36
  24. package/src/__tests__/avatar-e2e.test.ts +1 -9
  25. package/src/__tests__/avatar-generator.test.ts +1 -7
  26. package/src/__tests__/browser-fill-credential.test.ts +0 -4
  27. package/src/__tests__/browser-manager.test.ts +0 -6
  28. package/src/__tests__/call-controller.test.ts +1 -22
  29. package/src/__tests__/call-conversation-messages.test.ts +0 -21
  30. package/src/__tests__/call-domain.test.ts +0 -25
  31. package/src/__tests__/call-pointer-messages.test.ts +0 -21
  32. package/src/__tests__/call-recovery.test.ts +0 -22
  33. package/src/__tests__/call-routes-http.test.ts +0 -24
  34. package/src/__tests__/call-store.test.ts +0 -21
  35. package/src/__tests__/cancel-resolves-conversation-key.test.ts +0 -24
  36. package/src/__tests__/canonical-guardian-store.test.ts +48 -21
  37. package/src/__tests__/channel-approval-routes.test.ts +6 -26
  38. package/src/__tests__/channel-approvals.test.ts +1 -38
  39. package/src/__tests__/channel-delivery-store.test.ts +0 -21
  40. package/src/__tests__/channel-guardian.test.ts +0 -26
  41. package/src/__tests__/channel-reply-delivery.test.ts +5 -0
  42. package/src/__tests__/channel-retry-sweep.test.ts +0 -21
  43. package/src/__tests__/checker.test.ts +26 -61
  44. package/src/__tests__/clawhub.test.ts +9 -25
  45. package/src/__tests__/cli-command-risk-guard.test.ts +0 -18
  46. package/src/__tests__/config-loader-backfill.test.ts +9 -28
  47. package/src/__tests__/config-schema-cmd.test.ts +5 -25
  48. package/src/__tests__/config-schema.test.ts +21 -40
  49. package/src/__tests__/config-watcher.test.ts +4 -91
  50. package/src/__tests__/confirmation-request-guardian-bridge.test.ts +0 -21
  51. package/src/__tests__/contacts-tools.test.ts +0 -21
  52. package/src/__tests__/context-memory-e2e.test.ts +0 -21
  53. package/src/__tests__/context-window-manager.test.ts +130 -3
  54. package/src/__tests__/conversation-abort-tool-results.test.ts +0 -4
  55. package/src/__tests__/conversation-agent-loop-overflow.test.ts +0 -4
  56. package/src/__tests__/conversation-agent-loop.test.ts +0 -4
  57. package/src/__tests__/conversation-attachments.test.ts +1 -24
  58. package/src/__tests__/conversation-attention-store.test.ts +0 -21
  59. package/src/__tests__/conversation-attention-telegram.test.ts +0 -22
  60. package/src/__tests__/conversation-clear-safety.test.ts +0 -22
  61. package/src/__tests__/conversation-confirmation-signals.test.ts +2 -21
  62. package/src/__tests__/conversation-delete-schedule-cleanup.test.ts +0 -24
  63. package/src/__tests__/conversation-disk-view-integration.test.ts +1 -23
  64. package/src/__tests__/conversation-disk-view.test.ts +5 -27
  65. package/src/__tests__/conversation-error.test.ts +1 -1
  66. package/src/__tests__/conversation-fork-crud.test.ts +1 -33
  67. package/src/__tests__/conversation-fork-route.test.ts +0 -27
  68. package/src/__tests__/conversation-history-web-search.test.ts +23 -16
  69. package/src/__tests__/conversation-init.benchmark.test.ts +22 -43
  70. package/src/__tests__/conversation-key-store-disk-view.test.ts +8 -34
  71. package/src/__tests__/conversation-load-history-repair.test.ts +0 -4
  72. package/src/__tests__/conversation-pre-run-repair.test.ts +0 -4
  73. package/src/__tests__/conversation-provider-retry-repair.test.ts +0 -4
  74. package/src/__tests__/conversation-queue.test.ts +8 -8
  75. package/src/__tests__/conversation-routes-disk-view.test.ts +13 -51
  76. package/src/__tests__/conversation-runtime-assembly.test.ts +64 -38
  77. package/src/__tests__/conversation-slash-commands.test.ts +5 -0
  78. package/src/__tests__/conversation-slash-queue.test.ts +0 -4
  79. package/src/__tests__/conversation-slash-unknown.test.ts +0 -4
  80. package/src/__tests__/conversation-speed-override.test.ts +326 -0
  81. package/src/__tests__/conversation-starter-routes.test.ts +0 -23
  82. package/src/__tests__/conversation-store.test.ts +0 -21
  83. package/src/__tests__/conversation-unread-route.test.ts +0 -24
  84. package/src/__tests__/conversation-usage.test.ts +56 -21
  85. package/src/__tests__/conversation-wipe.test.ts +0 -21
  86. package/src/__tests__/conversation-workspace-cache-state.test.ts +0 -4
  87. package/src/__tests__/conversation-workspace-injection.test.ts +0 -4
  88. package/src/__tests__/conversation-workspace-tool-tracking.test.ts +0 -4
  89. package/src/__tests__/credential-execution-shell-lockdown.test.ts +8 -5
  90. package/src/__tests__/credential-vault-unit.test.ts +9 -428
  91. package/src/__tests__/credentials-cli.test.ts +10 -10
  92. package/src/__tests__/daemon-assistant-events.test.ts +0 -19
  93. package/src/__tests__/date-context.test.ts +77 -97
  94. package/src/__tests__/db-conversation-fork-lineage-migration.test.ts +7 -24
  95. package/src/__tests__/db-llm-request-log-provider-migration.test.ts +29 -42
  96. package/src/__tests__/delete-managed-skill-tool.test.ts +2 -10
  97. package/src/__tests__/deterministic-verification-control-plane.test.ts +1 -26
  98. package/src/__tests__/docker-signing-key-bootstrap.test.ts +61 -15
  99. package/src/__tests__/dynamic-skill-workflow-prompt.test.ts +2 -36
  100. package/src/__tests__/email-cli.test.ts +6 -6
  101. package/src/__tests__/ephemeral-permissions.test.ts +5 -17
  102. package/src/__tests__/first-greeting.test.ts +11 -32
  103. package/src/__tests__/followup-tools.test.ts +0 -21
  104. package/src/__tests__/gateway-only-enforcement.test.ts +0 -20
  105. package/src/__tests__/guardian-action-conversation-turn.test.ts +0 -23
  106. package/src/__tests__/guardian-action-followup-executor.test.ts +0 -23
  107. package/src/__tests__/guardian-action-followup-store.test.ts +0 -21
  108. package/src/__tests__/guardian-action-grant-mint-consume.test.ts +0 -21
  109. package/src/__tests__/guardian-action-late-reply.test.ts +0 -21
  110. package/src/__tests__/guardian-action-store.test.ts +0 -21
  111. package/src/__tests__/guardian-action-sweep.test.ts +0 -21
  112. package/src/__tests__/guardian-binding-drift-heal.test.ts +0 -23
  113. package/src/__tests__/guardian-decision-primitive-canonical.test.ts +172 -22
  114. package/src/__tests__/guardian-dispatch.test.ts +0 -21
  115. package/src/__tests__/guardian-grant-minting.test.ts +0 -22
  116. package/src/__tests__/guardian-outbound-http.test.ts +0 -22
  117. package/src/__tests__/guardian-principal-id-roundtrip.test.ts +0 -23
  118. package/src/__tests__/guardian-routing-invariants.test.ts +0 -22
  119. package/src/__tests__/guardian-routing-state.test.ts +0 -22
  120. package/src/__tests__/guardian-verification-voice-binding.test.ts +0 -24
  121. package/src/__tests__/headless-browser-interactions.test.ts +0 -4
  122. package/src/__tests__/headless-browser-navigate.test.ts +0 -4
  123. package/src/__tests__/headless-browser-read-tools.test.ts +0 -4
  124. package/src/__tests__/headless-browser-snapshot.test.ts +0 -4
  125. package/src/__tests__/heartbeat-service.test.ts +99 -26
  126. package/src/__tests__/hooks-blocking.test.ts +3 -3
  127. package/src/__tests__/hooks-config.test.ts +7 -7
  128. package/src/__tests__/hooks-discovery.test.ts +3 -3
  129. package/src/__tests__/hooks-integration.test.ts +5 -5
  130. package/src/__tests__/hooks-manager.test.ts +3 -3
  131. package/src/__tests__/hooks-runner.test.ts +5 -23
  132. package/src/__tests__/hooks-settings.test.ts +3 -3
  133. package/src/__tests__/hooks-templates.test.ts +3 -3
  134. package/src/__tests__/http-conversation-lineage.test.ts +0 -27
  135. package/src/__tests__/identity-intro-cache.test.ts +0 -4
  136. package/src/__tests__/inbound-invite-redemption.test.ts +0 -22
  137. package/src/__tests__/inline-skill-load-permissions.test.ts +5 -16
  138. package/src/__tests__/intent-routing.test.ts +2 -55
  139. package/src/__tests__/invite-redemption-service.test.ts +0 -21
  140. package/src/__tests__/invite-routes-http.test.ts +0 -21
  141. package/src/__tests__/jobs-store-qdrant-breaker.test.ts +0 -17
  142. package/src/__tests__/journal-context.test.ts +8 -75
  143. package/src/__tests__/list-messages-attachments.test.ts +0 -22
  144. package/src/__tests__/llm-context-route-provider.test.ts +0 -21
  145. package/src/__tests__/llm-request-log-turn-query.test.ts +46 -28
  146. package/src/__tests__/llm-usage-store.test.ts +0 -21
  147. package/src/__tests__/log-export-workspace.test.ts +1 -1
  148. package/src/__tests__/managed-skill-lifecycle.test.ts +1 -1
  149. package/src/__tests__/managed-store.test.ts +1 -1
  150. package/src/__tests__/mcp-cli.test.ts +7 -10
  151. package/src/__tests__/memory-context-benchmark.benchmark.test.ts +0 -21
  152. package/src/__tests__/memory-jobs-worker-backoff.test.ts +0 -11
  153. package/src/__tests__/memory-lifecycle-e2e.test.ts +0 -21
  154. package/src/__tests__/memory-recall-log-store.test.ts +0 -27
  155. package/src/__tests__/memory-recall-quality.test.ts +0 -21
  156. package/src/__tests__/memory-regressions.experimental.test.ts +31 -30
  157. package/src/__tests__/memory-regressions.test.ts +282 -70
  158. package/src/__tests__/memory-retrieval.benchmark.test.ts +0 -21
  159. package/src/__tests__/memory-upsert-concurrency.test.ts +0 -21
  160. package/src/__tests__/messaging-send-tool.test.ts +201 -0
  161. package/src/__tests__/migration-cross-version-compatibility.test.ts +18 -13
  162. package/src/__tests__/migration-export-http.test.ts +7 -1
  163. package/src/__tests__/migration-import-commit-http.test.ts +16 -14
  164. package/src/__tests__/migration-import-preflight-http.test.ts +27 -44
  165. package/src/__tests__/migration-validate-http.test.ts +1 -28
  166. package/src/__tests__/native-web-search.test.ts +25 -22
  167. package/src/__tests__/navigate-settings-tab.test.ts +6 -2
  168. package/src/__tests__/non-member-access-request.test.ts +0 -22
  169. package/src/__tests__/notification-guardian-path.test.ts +0 -21
  170. package/src/__tests__/notification-schedule-dedup.test.ts +1 -25
  171. package/src/__tests__/oauth-apps-routes.test.ts +103 -2
  172. package/src/__tests__/oauth-cli.test.ts +52 -0
  173. package/src/__tests__/oauth-provider-profiles.test.ts +0 -16
  174. package/src/__tests__/oauth-provider-serializer.test.ts +232 -0
  175. package/src/__tests__/oauth-providers-routes.test.ts +257 -0
  176. package/src/__tests__/oauth-store.test.ts +0 -21
  177. package/src/__tests__/onboarding-template-contract.test.ts +2 -2
  178. package/src/__tests__/openai-provider.test.ts +261 -0
  179. package/src/__tests__/pairing-concurrent.test.ts +6 -6
  180. package/src/__tests__/pairing-routes.test.ts +7 -1
  181. package/src/__tests__/path-policy.test.ts +1 -1
  182. package/src/__tests__/platform.test.ts +62 -251
  183. package/src/__tests__/playbook-execution.test.ts +0 -21
  184. package/src/__tests__/playbook-tools.test.ts +0 -21
  185. package/src/__tests__/pricing.test.ts +100 -0
  186. package/src/__tests__/relay-server.test.ts +1 -25
  187. package/src/__tests__/runtime-attachment-metadata.test.ts +0 -24
  188. package/src/__tests__/runtime-events-sse-parity.test.ts +2 -24
  189. package/src/__tests__/runtime-events-sse.test.ts +0 -24
  190. package/src/__tests__/sandbox-diagnostics.test.ts +2 -1
  191. package/src/__tests__/scaffold-managed-skill-tool.test.ts +1 -1
  192. package/src/__tests__/schedule-store.test.ts +0 -21
  193. package/src/__tests__/schedule-tools.test.ts +0 -21
  194. package/src/__tests__/scheduler-recurrence.test.ts +0 -21
  195. package/src/__tests__/scoped-approval-grants.test.ts +0 -21
  196. package/src/__tests__/scoped-grant-security-matrix.test.ts +0 -21
  197. package/src/__tests__/secret-allowlist.test.ts +1 -1
  198. package/src/__tests__/secret-ingress-channel.test.ts +0 -5
  199. package/src/__tests__/secret-ingress-cli.test.ts +0 -6
  200. package/src/__tests__/secret-ingress-http.test.ts +0 -5
  201. package/src/__tests__/secret-ingress.test.ts +0 -5
  202. package/src/__tests__/send-endpoint-busy.test.ts +0 -24
  203. package/src/__tests__/sequence-store.test.ts +0 -21
  204. package/src/__tests__/server-history-render.test.ts +0 -24
  205. package/src/__tests__/shell-tool-proxy-mode.test.ts +0 -4
  206. package/src/__tests__/skill-feature-flags.test.ts +8 -0
  207. package/src/__tests__/skill-load-inline-command.test.ts +9 -0
  208. package/src/__tests__/skill-load-inline-includes.test.ts +9 -0
  209. package/src/__tests__/skill-load-tool.test.ts +11 -0
  210. package/src/__tests__/skill-secret-handling-guard.test.ts +212 -0
  211. package/src/__tests__/skills-uninstall.test.ts +10 -8
  212. package/src/__tests__/skills.test.ts +1 -1
  213. package/src/__tests__/slack-channel-config.test.ts +1 -1
  214. package/src/__tests__/slack-inbound-verification.test.ts +0 -22
  215. package/src/__tests__/starter-bundle.test.ts +4 -1
  216. package/src/__tests__/suggestion-routes.test.ts +2 -0
  217. package/src/__tests__/system-prompt.test.ts +1 -1
  218. package/src/__tests__/terminal-tools.test.ts +1 -1
  219. package/src/__tests__/test-preload.ts +31 -0
  220. package/src/__tests__/token-estimator-accuracy.benchmark.test.ts +1 -1
  221. package/src/__tests__/tool-execution-abort-cleanup.test.ts +1 -1
  222. package/src/__tests__/tool-execution-pipeline.benchmark.test.ts +1 -1
  223. package/src/__tests__/tool-executor.test.ts +0 -20
  224. package/src/__tests__/tool-input-summary.test.ts +124 -0
  225. package/src/__tests__/tool-preview-lifecycle.test.ts +2 -1
  226. package/src/__tests__/trust-store.test.ts +7 -1
  227. package/src/__tests__/trusted-contact-inline-approval-integration.test.ts +1 -1
  228. package/src/__tests__/trusted-contact-lifecycle-notifications.test.ts +1 -1
  229. package/src/__tests__/trusted-contact-multichannel.test.ts +1 -1
  230. package/src/__tests__/trusted-contact-verification.test.ts +1 -1
  231. package/src/__tests__/turn-boundary-resolution.test.ts +1 -1
  232. package/src/__tests__/twilio-routes.test.ts +1 -1
  233. package/src/__tests__/update-bulletin.test.ts +1 -1
  234. package/src/__tests__/vbundle-pax-and-symlink.test.ts +1 -1
  235. package/src/__tests__/vellum-self-knowledge-inline-command.test.ts +1 -0
  236. package/src/__tests__/voice-scoped-grant-consumer.test.ts +1 -1
  237. package/src/__tests__/voice-session-bridge.test.ts +1 -1
  238. package/src/__tests__/workspace-migration-009-backfill-conversation-disk-view.test.ts +4 -4
  239. package/src/__tests__/workspace-migration-013-repair-conversation-disk-view.test.ts +1 -1
  240. package/src/__tests__/workspace-migration-down-functions.test.ts +15 -3
  241. package/src/__tests__/workspace-migration-seed-device-id.test.ts +40 -4
  242. package/src/agent/loop.ts +6 -9
  243. package/src/approvals/guardian-decision-primitive.ts +46 -18
  244. package/src/approvals/guardian-request-resolvers.ts +19 -2
  245. package/src/calls/active-call-lease.ts +2 -2
  246. package/src/cli/AGENTS.md +1 -1
  247. package/src/cli/commands/doctor.ts +9 -9
  248. package/src/cli/commands/memory.ts +142 -0
  249. package/src/cli/commands/oauth/__tests__/connect.test.ts +13 -11
  250. package/src/cli/commands/oauth/__tests__/ping.test.ts +1 -1
  251. package/src/cli/commands/oauth/connect.ts +13 -12
  252. package/src/cli/commands/oauth/index.ts +1 -1
  253. package/src/cli/commands/oauth/providers.ts +47 -62
  254. package/src/cli/commands/platform/__tests__/connect.test.ts +250 -0
  255. package/src/cli/commands/platform/__tests__/disconnect.test.ts +290 -0
  256. package/src/cli/commands/platform/__tests__/status.test.ts +282 -0
  257. package/src/cli/commands/platform/connect.ts +17 -7
  258. package/src/cli/commands/platform/disconnect.ts +28 -3
  259. package/src/cli/commands/platform/index.ts +3 -3
  260. package/src/cli.ts +1 -299
  261. package/src/config/assistant-feature-flags.ts +23 -15
  262. package/src/config/bundled-skills/app-builder/TOOLS.json +16 -0
  263. package/src/config/bundled-skills/app-builder/tools/app-create.ts +4 -0
  264. package/src/config/bundled-skills/app-builder/tools/app-delete.ts +5 -1
  265. package/src/config/bundled-skills/app-builder/tools/app-generate-icon.ts +9 -1
  266. package/src/config/bundled-skills/app-builder/tools/app-refresh.ts +5 -1
  267. package/src/config/bundled-skills/contacts/TOOLS.json +8 -0
  268. package/src/config/bundled-skills/contacts/tools/contact-search.ts +10 -1
  269. package/src/config/bundled-skills/contacts/tools/contact-upsert.ts +16 -2
  270. package/src/config/bundled-skills/media-processing/tools/ingest-media.ts +1 -0
  271. package/src/config/bundled-skills/messaging/SKILL.md +7 -7
  272. package/src/config/bundled-skills/messaging/tools/messaging-send.ts +37 -0
  273. package/src/config/bundled-skills/settings/TOOLS.json +5 -3
  274. package/src/config/bundled-skills/settings/tools/navigate-settings-tab.ts +4 -2
  275. package/src/config/bundled-skills/slack/SKILL.md +18 -0
  276. package/src/config/env-registry.ts +15 -11
  277. package/src/config/env.ts +1 -11
  278. package/src/config/feature-flag-registry.json +17 -1
  279. package/src/config/schema.ts +4 -0
  280. package/src/config/schemas/heartbeat.ts +6 -1
  281. package/src/config/schemas/inference.ts +14 -3
  282. package/src/config/schemas/memory-processing.ts +16 -8
  283. package/src/config/schemas/memory-retrieval.ts +3 -3
  284. package/src/config/skills.ts +1 -1
  285. package/src/context/window-manager.ts +174 -51
  286. package/src/credential-execution/client.ts +14 -2
  287. package/src/credential-execution/executable-discovery.ts +2 -2
  288. package/src/daemon/approved-devices-store.ts +2 -2
  289. package/src/daemon/assistant-attachments.ts +2 -0
  290. package/src/daemon/config-watcher.ts +4 -50
  291. package/src/daemon/conversation-agent-loop-handlers.ts +9 -1
  292. package/src/daemon/conversation-agent-loop.ts +12 -0
  293. package/src/daemon/conversation-error.ts +3 -5
  294. package/src/daemon/conversation-history.ts +7 -3
  295. package/src/daemon/conversation-lifecycle.ts +16 -0
  296. package/src/daemon/conversation-messaging.ts +1 -0
  297. package/src/daemon/conversation-notifiers.ts +67 -30
  298. package/src/daemon/conversation-process.ts +161 -2
  299. package/src/daemon/conversation-queue-manager.ts +2 -0
  300. package/src/daemon/conversation-runtime-assembly.ts +33 -11
  301. package/src/daemon/conversation-slash.ts +14 -3
  302. package/src/daemon/conversation-tool-setup.ts +2 -0
  303. package/src/daemon/conversation-usage.ts +32 -4
  304. package/src/daemon/conversation.ts +33 -1
  305. package/src/daemon/daemon-control.ts +32 -16
  306. package/src/daemon/date-context.ts +47 -45
  307. package/src/daemon/dictation-profile-store.ts +2 -2
  308. package/src/daemon/first-greeting.ts +6 -1
  309. package/src/daemon/handlers/conversations.ts +19 -0
  310. package/src/daemon/handlers/shared.ts +14 -21
  311. package/src/daemon/lifecycle.ts +18 -15
  312. package/src/daemon/message-types/conversations.ts +2 -0
  313. package/src/daemon/message-types/guardian-actions.ts +3 -17
  314. package/src/daemon/message-types/integrations.ts +11 -1
  315. package/src/daemon/message-types/messages.ts +1 -0
  316. package/src/daemon/pairing-store.ts +2 -79
  317. package/src/daemon/server.ts +154 -8
  318. package/src/daemon/watch-handler.ts +65 -21
  319. package/src/email/guardrails.ts +3 -3
  320. package/src/heartbeat/heartbeat-service.ts +14 -7
  321. package/src/hooks/cli.ts +2 -2
  322. package/src/hooks/config.ts +2 -2
  323. package/src/hooks/discovery.ts +2 -2
  324. package/src/hooks/manager.ts +2 -2
  325. package/src/hooks/runner.ts +5 -2
  326. package/src/hooks/templates.ts +2 -2
  327. package/src/index.ts +0 -12
  328. package/src/memory/admin.ts +181 -2
  329. package/src/memory/app-git-service.ts +61 -4
  330. package/src/memory/attachments-store.ts +2 -0
  331. package/src/memory/canonical-guardian-store.ts +16 -0
  332. package/src/memory/conversation-queries.ts +6 -6
  333. package/src/memory/db-init.ts +8 -0
  334. package/src/memory/embedding-local.ts +5 -2
  335. package/src/memory/indexer.ts +44 -26
  336. package/src/memory/items-extractor.ts +34 -82
  337. package/src/memory/job-handlers/batch-extraction.ts +741 -0
  338. package/src/memory/job-handlers/journal-carry-forward.test.ts +383 -0
  339. package/src/memory/job-handlers/journal-carry-forward.ts +255 -0
  340. package/src/memory/jobs-store.ts +28 -0
  341. package/src/memory/jobs-worker.ts +56 -9
  342. package/src/memory/journal-memory.ts +8 -2
  343. package/src/memory/lifecycle-events-store.ts +4 -2
  344. package/src/memory/llm-request-log-store.ts +40 -2
  345. package/src/memory/llm-usage-store.ts +4 -3
  346. package/src/memory/migrations/199-guardian-request-enrichment-columns.ts +71 -0
  347. package/src/memory/migrations/200-usage-llm-call-count.ts +20 -0
  348. package/src/memory/migrations/index.ts +2 -0
  349. package/src/memory/query-expansion.ts +83 -0
  350. package/src/memory/retriever.test.ts +119 -0
  351. package/src/memory/retriever.ts +513 -105
  352. package/src/memory/schema/guardian.ts +4 -0
  353. package/src/memory/schema/infrastructure.ts +1 -0
  354. package/src/memory/search/formatting.test.ts +140 -0
  355. package/src/memory/search/formatting.ts +143 -198
  356. package/src/memory/search/mmr.ts +136 -0
  357. package/src/memory/search/staleness.ts +0 -15
  358. package/src/memory/search/tier-classifier.ts +10 -21
  359. package/src/memory/search/types.ts +17 -0
  360. package/src/messaging/providers/slack/adapter.ts +51 -5
  361. package/src/notifications/broadcaster.ts +13 -0
  362. package/src/notifications/copy-composer.ts +8 -0
  363. package/src/oauth/connect-orchestrator.ts +1 -1
  364. package/src/oauth/connection-resolver.ts +2 -2
  365. package/src/oauth/provider-serializer.ts +116 -0
  366. package/src/permissions/trust-store.ts +24 -7
  367. package/src/prompts/__tests__/build-cli-reference-section.test.ts +5 -0
  368. package/src/prompts/journal-context.ts +54 -36
  369. package/src/prompts/persona-resolver.ts +1 -1
  370. package/src/prompts/system-prompt.ts +38 -28
  371. package/src/prompts/templates/BOOTSTRAP.md +14 -1
  372. package/src/prompts/templates/HEARTBEAT.md +10 -0
  373. package/src/prompts/templates/NOW.md +19 -25
  374. package/src/prompts/templates/SOUL.md +13 -1
  375. package/src/prompts/templates/UPDATES.md +12 -0
  376. package/src/prompts/update-bulletin.ts +1 -1
  377. package/src/providers/anthropic/client.ts +89 -18
  378. package/src/providers/model-catalog.ts +22 -2
  379. package/src/providers/model-intents.ts +2 -2
  380. package/src/providers/openai/client.ts +40 -1
  381. package/src/providers/retry.ts +23 -4
  382. package/src/providers/types.ts +2 -0
  383. package/src/runtime/assistant-scope.ts +1 -1
  384. package/src/runtime/auth/__tests__/credential-service.test.ts +1 -0
  385. package/src/runtime/auth/route-policy.ts +1 -0
  386. package/src/runtime/auth/token-service.ts +51 -29
  387. package/src/runtime/confirmation-request-guardian-bridge.ts +3 -1
  388. package/src/runtime/guardian-decision-types.ts +16 -10
  389. package/src/runtime/http-server.ts +10 -29
  390. package/src/runtime/http-types.ts +1 -0
  391. package/src/runtime/migrations/rebind-secrets-screen.ts +2 -2
  392. package/src/runtime/migrations/vbundle-builder.ts +7 -4
  393. package/src/runtime/migrations/vbundle-import-analyzer.ts +0 -4
  394. package/src/runtime/migrations/vbundle-importer.ts +1 -1
  395. package/src/runtime/routes/conversation-query-routes.ts +40 -8
  396. package/src/runtime/routes/conversation-routes.ts +125 -3
  397. package/src/runtime/routes/guardian-action-routes.ts +9 -3
  398. package/src/runtime/routes/identity-routes.ts +25 -4
  399. package/src/runtime/routes/llm-context-normalization.ts +1 -0
  400. package/src/runtime/routes/log-export-routes.ts +34 -12
  401. package/src/runtime/routes/migration-routes.ts +6 -10
  402. package/src/runtime/routes/oauth-apps.ts +2 -9
  403. package/src/runtime/routes/oauth-providers.ts +60 -0
  404. package/src/runtime/routes/pairing-routes.ts +0 -8
  405. package/src/runtime/routes/secret-routes.ts +9 -2
  406. package/src/runtime/routes/settings-routes.ts +0 -1
  407. package/src/runtime/routes/telemetry-routes.ts +16 -4
  408. package/src/security/encrypted-store.ts +2 -2
  409. package/src/security/secret-allowlist.ts +3 -3
  410. package/src/signals/emit-event.ts +42 -0
  411. package/src/signals/user-message.ts +37 -0
  412. package/src/telemetry/usage-telemetry-reporter.test.ts +83 -19
  413. package/src/telemetry/usage-telemetry-reporter.ts +23 -17
  414. package/src/tools/browser/browser-manager.ts +2 -2
  415. package/src/tools/browser/runtime-check.ts +2 -2
  416. package/src/tools/credentials/vault.ts +2 -249
  417. package/src/tools/memory/definitions.ts +1 -1
  418. package/src/tools/memory/handlers.test.ts +50 -8
  419. package/src/tools/memory/handlers.ts +3 -1
  420. package/src/tools/side-effects.ts +1 -6
  421. package/src/tools/terminal/safe-env.ts +3 -2
  422. package/src/tools/terminal/shell.ts +11 -14
  423. package/src/tools/tool-approval-handler.ts +20 -1
  424. package/src/tools/tool-input-summary.ts +66 -0
  425. package/src/tools/types.ts +4 -0
  426. package/src/usage/types.ts +4 -0
  427. package/src/util/device-id.ts +10 -10
  428. package/src/util/platform.ts +72 -124
  429. package/src/util/pricing.ts +19 -6
  430. package/src/util/strip-comment-lines.ts +28 -0
  431. package/src/workspace/git-service.ts +8 -18
  432. package/src/workspace/migrations/003-seed-device-id.ts +6 -4
  433. package/src/workspace/migrations/016-extract-feature-flags-to-protected.ts +7 -1
  434. package/src/workspace/migrations/017-seed-persona-dirs.ts +2 -4
  435. package/src/workspace/migrations/021-move-signals-to-workspace.ts +84 -0
  436. package/src/workspace/migrations/022-move-hooks-to-workspace.ts +94 -0
  437. package/src/workspace/migrations/023-move-config-files-to-workspace.ts +86 -0
  438. package/src/workspace/migrations/024-move-runtime-files-to-workspace.ts +126 -0
  439. package/src/workspace/migrations/migrate-to-workspace-volume.ts +3 -6
  440. package/src/workspace/migrations/registry.ts +8 -0
  441. package/src/signals/confirm.ts +0 -82
  442. package/src/signals/trust-rule.ts +0 -174
@@ -0,0 +1,326 @@
1
+ /**
2
+ * Tests for per-conversation speed override.
3
+ *
4
+ * Verifies that the Conversation constructor resolves speed from the
5
+ * per-conversation speedOverride parameter first, falling back to the
6
+ * global config speed setting.
7
+ */
8
+ import { describe, expect, mock, test } from "bun:test";
9
+
10
+ import type {
11
+ AgentEvent,
12
+ AgentLoopConfig,
13
+ CheckpointDecision,
14
+ CheckpointInfo,
15
+ } from "../agent/loop.js";
16
+ import type { ServerMessage } from "../daemon/message-protocol.js";
17
+ import type { Message, ProviderResponse } from "../providers/types.js";
18
+
19
+ // ---------------------------------------------------------------------------
20
+ // Mocks — must precede Conversation import
21
+ // ---------------------------------------------------------------------------
22
+
23
+ function makeLoggerStub(): Record<string, unknown> {
24
+ const stub: Record<string, unknown> = {};
25
+ for (const m of [
26
+ "info",
27
+ "warn",
28
+ "error",
29
+ "debug",
30
+ "trace",
31
+ "fatal",
32
+ "silent",
33
+ "child",
34
+ ]) {
35
+ stub[m] = m === "child" ? () => makeLoggerStub() : () => {};
36
+ }
37
+ return stub;
38
+ }
39
+
40
+ mock.module("../util/logger.js", () => ({
41
+ getLogger: () => makeLoggerStub(),
42
+ }));
43
+
44
+ mock.module("../memory/guardian-action-store.js", () => ({
45
+ getGuardianActionRequest: () => null,
46
+ resolveGuardianActionRequest: () => {},
47
+ }));
48
+
49
+ mock.module("../providers/registry.js", () => ({
50
+ getProvider: () => ({ name: "mock-provider" }),
51
+ initializeProviders: () => {},
52
+ }));
53
+
54
+ // Controllable config mock — speed and feature flag behavior are test-specific.
55
+ let mockConfigSpeed: "standard" | "fast" = "fast";
56
+
57
+ mock.module("../config/loader.js", () => ({
58
+ getConfig: () => ({
59
+ ui: {},
60
+ provider: "mock-provider",
61
+ maxTokens: 4096,
62
+ thinking: false,
63
+ speed: mockConfigSpeed,
64
+ effort: "high",
65
+ contextWindow: {
66
+ maxInputTokens: 100000,
67
+ thresholdTokens: 80000,
68
+ preserveRecentMessages: 6,
69
+ summaryModel: "mock-model",
70
+ maxSummaryTokens: 512,
71
+ },
72
+ rateLimit: { maxRequestsPerMinute: 0 },
73
+ timeouts: { permissionTimeoutSec: 1 },
74
+ skills: { entries: {}, allowBundled: true },
75
+ permissions: { mode: "workspace" },
76
+ }),
77
+ loadRawConfig: () => ({}),
78
+ saveRawConfig: () => {},
79
+ invalidateConfigCache: () => {},
80
+ }));
81
+
82
+ // Feature flag mock — fast-mode enabled for all tests in this file.
83
+ mock.module("../config/assistant-feature-flags.js", () => ({
84
+ isAssistantFeatureFlagEnabled: (key: string) => {
85
+ if (key === "fast-mode") return true;
86
+ return true;
87
+ },
88
+ }));
89
+
90
+ mock.module("../prompts/system-prompt.js", () => ({
91
+ buildSystemPrompt: () => "system prompt",
92
+ }));
93
+
94
+ mock.module("../config/skills.js", () => ({
95
+ loadSkillCatalog: () => [],
96
+ loadSkillBySelector: () => ({ skill: null }),
97
+ ensureSkillIcon: async () => null,
98
+ }));
99
+
100
+ mock.module("../config/skill-state.js", () => ({
101
+ resolveSkillStates: () => [],
102
+ }));
103
+
104
+ mock.module("../permissions/trust-store.js", () => ({
105
+ addRule: () => {},
106
+ findHighestPriorityRule: () => null,
107
+ clearCache: () => {},
108
+ }));
109
+
110
+ mock.module("../security/secret-allowlist.js", () => ({
111
+ resetAllowlist: () => {},
112
+ }));
113
+
114
+ mock.module("../memory/conversation-crud.js", () => ({
115
+ getConversationType: () => "default",
116
+ setConversationOriginChannelIfUnset: () => {},
117
+ updateConversationContextWindow: () => {},
118
+ deleteMessageById: () => {},
119
+ provenanceFromTrustContext: () => ({
120
+ source: "user",
121
+ trustContext: undefined,
122
+ }),
123
+ getConversationOriginInterface: () => null,
124
+ getConversationOriginChannel: () => null,
125
+ getMessages: () => [],
126
+ getConversation: () => ({
127
+ id: "conv-1",
128
+ contextSummary: null,
129
+ contextCompactedMessageCount: 0,
130
+ totalInputTokens: 0,
131
+ totalOutputTokens: 0,
132
+ totalEstimatedCost: 0,
133
+ }),
134
+ createConversation: () => ({ id: "conv-1" }),
135
+ addMessage: () => ({ id: `msg-${Date.now()}` }),
136
+ updateConversationUsage: () => {},
137
+ updateConversationTitle: () => {},
138
+ }));
139
+
140
+ mock.module("../memory/conversation-queries.js", () => ({
141
+ listConversations: () => [],
142
+ }));
143
+
144
+ mock.module("../memory/attachments-store.js", () => ({
145
+ uploadAttachment: () => ({ id: `att-${Date.now()}` }),
146
+ linkAttachmentToMessage: () => {},
147
+ }));
148
+
149
+ mock.module("../memory/retriever.js", () => ({
150
+ buildMemoryRecall: async () => ({
151
+ enabled: false,
152
+ degraded: false,
153
+ injectedText: "",
154
+ semanticHits: 0,
155
+ injectedTokens: 0,
156
+ latencyMs: 0,
157
+ }),
158
+ injectMemoryRecallAsUserBlock: (msgs: Message[]) => msgs,
159
+ }));
160
+
161
+ mock.module("../context/window-manager.js", () => ({
162
+ ContextWindowManager: class {
163
+ constructor() {}
164
+ shouldCompact() {
165
+ return { needed: false, estimatedTokens: 0 };
166
+ }
167
+ async maybeCompact() {
168
+ return { compacted: false };
169
+ }
170
+ },
171
+ createContextSummaryMessage: () => ({
172
+ role: "user",
173
+ content: [{ type: "text", text: "summary" }],
174
+ }),
175
+ getSummaryFromContextMessage: () => null,
176
+ }));
177
+
178
+ mock.module("../memory/llm-usage-store.js", () => ({
179
+ recordUsageEvent: () => ({ id: "mock-id", createdAt: Date.now() }),
180
+ listUsageEvents: () => [],
181
+ }));
182
+
183
+ // Capture AgentLoop constructor config for assertions.
184
+ let lastAgentLoopConfig: Partial<AgentLoopConfig> | undefined;
185
+
186
+ mock.module("../agent/loop.js", () => ({
187
+ AgentLoop: class {
188
+ constructor(
189
+ _provider: unknown,
190
+ _systemPrompt: string,
191
+ config?: Partial<AgentLoopConfig>,
192
+ ) {
193
+ lastAgentLoopConfig = config;
194
+ }
195
+ getToolTokenBudget() {
196
+ return 0;
197
+ }
198
+ async run(
199
+ _messages: Message[],
200
+ _onEvent: (event: AgentEvent) => void,
201
+ _signal?: AbortSignal,
202
+ _requestId?: string,
203
+ _onCheckpoint?: (checkpoint: CheckpointInfo) => CheckpointDecision,
204
+ ): Promise<Message[]> {
205
+ return [];
206
+ }
207
+ },
208
+ }));
209
+
210
+ mock.module("../memory/canonical-guardian-store.js", () => ({
211
+ listPendingCanonicalGuardianRequestsByDestinationConversation: () => [],
212
+ listCanonicalGuardianRequests: () => [],
213
+ listPendingRequestsByConversationScope: () => [],
214
+ createCanonicalGuardianRequest: () => ({
215
+ id: "mock-cg-id",
216
+ code: "MOCK",
217
+ status: "pending",
218
+ }),
219
+ getCanonicalGuardianRequest: () => null,
220
+ getCanonicalGuardianRequestByCode: () => null,
221
+ updateCanonicalGuardianRequest: () => {},
222
+ resolveCanonicalGuardianRequest: () => {},
223
+ createCanonicalGuardianDelivery: () => ({ id: "mock-cgd-id" }),
224
+ listCanonicalGuardianDeliveries: () => [],
225
+ listPendingCanonicalGuardianRequestsByDestinationChat: () => [],
226
+ updateCanonicalGuardianDelivery: () => {},
227
+ generateCanonicalRequestCode: () => "MOCK-CODE",
228
+ }));
229
+
230
+ // ---------------------------------------------------------------------------
231
+ // Import Conversation AFTER mocks
232
+ // ---------------------------------------------------------------------------
233
+
234
+ import { Conversation } from "../daemon/conversation.js";
235
+
236
+ // ---------------------------------------------------------------------------
237
+ // Helpers
238
+ // ---------------------------------------------------------------------------
239
+
240
+ function makeProvider() {
241
+ return {
242
+ name: "mock",
243
+ async sendMessage(): Promise<ProviderResponse> {
244
+ return {
245
+ content: [],
246
+ model: "mock",
247
+ usage: { inputTokens: 0, outputTokens: 0 },
248
+ stopReason: "end_turn",
249
+ };
250
+ },
251
+ };
252
+ }
253
+
254
+ function makeSendToClient(): (msg: ServerMessage) => void {
255
+ return () => {};
256
+ }
257
+
258
+ // ---------------------------------------------------------------------------
259
+ // Tests
260
+ // ---------------------------------------------------------------------------
261
+
262
+ describe("per-conversation speed override", () => {
263
+ test("speedOverride 'standard' prevents fast mode even when global config is 'fast'", () => {
264
+ mockConfigSpeed = "fast";
265
+ lastAgentLoopConfig = undefined;
266
+
267
+ new Conversation(
268
+ "conv-speed-override-1",
269
+ makeProvider(),
270
+ "system prompt",
271
+ 4096,
272
+ makeSendToClient(),
273
+ "/tmp",
274
+ undefined, // broadcastToAllClients
275
+ undefined, // memoryPolicy
276
+ undefined, // sharedCesClient
277
+ "standard", // speedOverride
278
+ );
279
+
280
+ expect(lastAgentLoopConfig).toBeDefined();
281
+ // When speedOverride is "standard", the AgentLoop should NOT receive speed: "fast"
282
+ expect(lastAgentLoopConfig!.speed).toBeUndefined();
283
+ });
284
+
285
+ test("no speedOverride uses global config speed", () => {
286
+ mockConfigSpeed = "fast";
287
+ lastAgentLoopConfig = undefined;
288
+
289
+ new Conversation(
290
+ "conv-speed-global-1",
291
+ makeProvider(),
292
+ "system prompt",
293
+ 4096,
294
+ makeSendToClient(),
295
+ "/tmp",
296
+ undefined, // broadcastToAllClients
297
+ undefined, // memoryPolicy
298
+ undefined, // sharedCesClient
299
+ // no speedOverride — should fall back to global config "fast"
300
+ );
301
+
302
+ expect(lastAgentLoopConfig).toBeDefined();
303
+ expect(lastAgentLoopConfig!.speed).toBe("fast");
304
+ });
305
+
306
+ test("speedOverride 'fast' enables fast mode even when global config is 'standard'", () => {
307
+ mockConfigSpeed = "standard";
308
+ lastAgentLoopConfig = undefined;
309
+
310
+ new Conversation(
311
+ "conv-speed-override-fast-1",
312
+ makeProvider(),
313
+ "system prompt",
314
+ 4096,
315
+ makeSendToClient(),
316
+ "/tmp",
317
+ undefined, // broadcastToAllClients
318
+ undefined, // memoryPolicy
319
+ undefined, // sharedCesClient
320
+ "fast", // speedOverride
321
+ );
322
+
323
+ expect(lastAgentLoopConfig).toBeDefined();
324
+ expect(lastAgentLoopConfig!.speed).toBe("fast");
325
+ });
326
+ });
@@ -1,25 +1,7 @@
1
- import { mkdtempSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
1
  import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
5
2
 
6
3
  import { v4 as uuid } from "uuid";
7
4
 
8
- const testDir = mkdtempSync(
9
- join(tmpdir(), "conversation-starter-routes-test-"),
10
- );
11
-
12
- mock.module("../util/platform.js", () => ({
13
- getDataDir: () => testDir,
14
- isMacOS: () => process.platform === "darwin",
15
- isLinux: () => process.platform === "linux",
16
- isWindows: () => process.platform === "win32",
17
- getPidPath: () => join(testDir, "test.pid"),
18
- getDbPath: () => join(testDir, "test.db"),
19
- getLogPath: () => join(testDir, "test.log"),
20
- ensureDataDir: () => {},
21
- }));
22
-
23
5
  mock.module("../util/logger.js", () => ({
24
6
  getLogger: () =>
25
7
  new Proxy({} as Record<string, unknown>, {
@@ -37,11 +19,6 @@ initializeDb();
37
19
 
38
20
  afterAll(() => {
39
21
  resetDb();
40
- try {
41
- rmSync(testDir, { recursive: true });
42
- } catch {
43
- /* best effort */
44
- }
45
22
  });
46
23
 
47
24
  const routes = conversationStarterRouteDefinitions();
@@ -1,21 +1,5 @@
1
- import { mkdtempSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
1
  import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
5
2
 
6
- const testDir = mkdtempSync(join(tmpdir(), "conv-store-test-"));
7
-
8
- mock.module("../util/platform.js", () => ({
9
- getDataDir: () => testDir,
10
- isMacOS: () => process.platform === "darwin",
11
- isLinux: () => process.platform === "linux",
12
- isWindows: () => process.platform === "win32",
13
- getPidPath: () => join(testDir, "test.pid"),
14
- getDbPath: () => join(testDir, "test.db"),
15
- getLogPath: () => join(testDir, "test.log"),
16
- ensureDataDir: () => {},
17
- }));
18
-
19
3
  mock.module("../util/logger.js", () => ({
20
4
  getLogger: () =>
21
5
  new Proxy({} as Record<string, unknown>, {
@@ -47,11 +31,6 @@ initializeDb();
47
31
 
48
32
  afterAll(() => {
49
33
  resetDb();
50
- try {
51
- rmSync(testDir, { recursive: true });
52
- } catch {
53
- /* best effort */
54
- }
55
34
  });
56
35
 
57
36
  describe("deleteLastExchange", () => {
@@ -1,24 +1,5 @@
1
- import { mkdtempSync, realpathSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
1
  import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
5
2
 
6
- const testDir = realpathSync(
7
- mkdtempSync(join(tmpdir(), "conversation-unread-route-test-")),
8
- );
9
-
10
- mock.module("../util/platform.js", () => ({
11
- getRootDir: () => testDir,
12
- getDataDir: () => testDir,
13
- isMacOS: () => process.platform === "darwin",
14
- isLinux: () => process.platform === "linux",
15
- isWindows: () => process.platform === "win32",
16
- getPidPath: () => join(testDir, "test.pid"),
17
- getDbPath: () => join(testDir, "test.db"),
18
- getLogPath: () => join(testDir, "test.log"),
19
- ensureDataDir: () => {},
20
- }));
21
-
22
3
  mock.module("../util/logger.js", () => ({
23
4
  getLogger: () =>
24
5
  new Proxy({} as Record<string, unknown>, {
@@ -64,11 +45,6 @@ describe("POST /v1/conversations/unread", () => {
64
45
 
65
46
  afterAll(async () => {
66
47
  await server?.stop();
67
- try {
68
- rmSync(testDir, { recursive: true, force: true });
69
- } catch {
70
- /* best effort */
71
- }
72
48
  });
73
49
 
74
50
  async function startServer(): Promise<void> {
@@ -1,10 +1,5 @@
1
- import { mkdtempSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
1
  import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
5
2
 
6
- const testDir = mkdtempSync(join(tmpdir(), "session-usage-test-"));
7
-
8
3
  const updateConversationUsageCalls: Array<{
9
4
  conversationId: string;
10
5
  inputTokens: number;
@@ -12,17 +7,6 @@ const updateConversationUsageCalls: Array<{
12
7
  estimatedCost: number;
13
8
  }> = [];
14
9
 
15
- mock.module("../util/platform.js", () => ({
16
- getDataDir: () => testDir,
17
- isMacOS: () => process.platform === "darwin",
18
- isLinux: () => process.platform === "linux",
19
- isWindows: () => process.platform === "win32",
20
- getPidPath: () => join(testDir, "test.pid"),
21
- getDbPath: () => join(testDir, "test.db"),
22
- getLogPath: () => join(testDir, "test.log"),
23
- ensureDataDir: () => {},
24
- }));
25
-
26
10
  mock.module("../util/logger.js", () => ({
27
11
  getLogger: () =>
28
12
  new Proxy({} as Record<string, unknown>, {
@@ -62,11 +46,6 @@ initializeDb();
62
46
 
63
47
  afterAll(() => {
64
48
  resetDb();
65
- try {
66
- rmSync(testDir, { recursive: true });
67
- } catch {
68
- /* best effort */
69
- }
70
49
  });
71
50
 
72
51
  describe("recordUsage", () => {
@@ -76,6 +55,62 @@ describe("recordUsage", () => {
76
55
  updateConversationUsageCalls.length = 0;
77
56
  });
78
57
 
58
+ test("applies fast mode pricing when any response has speed: fast", () => {
59
+ const usageStats = {
60
+ inputTokens: 0,
61
+ outputTokens: 0,
62
+ estimatedCost: 0,
63
+ };
64
+
65
+ // First response is standard, second is fast — should detect fast
66
+ const rawResponses = [
67
+ { usage: { speed: "standard" } },
68
+ { usage: { speed: "fast" } },
69
+ ];
70
+
71
+ recordUsage(
72
+ {
73
+ conversationId: "conv-speed-1",
74
+ providerName: "anthropic",
75
+ usageStats,
76
+ },
77
+ 1_000_000,
78
+ 1_000_000,
79
+ "claude-opus-4-6",
80
+ () => {},
81
+ "main_agent",
82
+ "req-speed-1",
83
+ 0,
84
+ 0,
85
+ rawResponses,
86
+ );
87
+
88
+ const events = listUsageEvents();
89
+ expect(events).toHaveLength(1);
90
+
91
+ // With fast mode, pricing should use the 6x multiplier
92
+ const fastUsage: PricingUsage = {
93
+ directInputTokens: 1_000_000,
94
+ outputTokens: 1_000_000,
95
+ cacheCreationInputTokens: 0,
96
+ cacheReadInputTokens: 0,
97
+ anthropicCacheCreation: null,
98
+ speed: "fast",
99
+ };
100
+ const expectedPricing = resolvePricingForUsageWithOverrides(
101
+ "anthropic",
102
+ "claude-opus-4-6",
103
+ fastUsage,
104
+ [],
105
+ );
106
+
107
+ expect(events[0].estimatedCostUsd).toBe(
108
+ expectedPricing.estimatedCostUsd ?? null,
109
+ );
110
+ // Sanity: fast should be 6x standard ($30 * 6 = $180)
111
+ expect(expectedPricing.estimatedCostUsd).toBe(180);
112
+ });
113
+
79
114
  test("stores direct input separately from Anthropic cache usage while keeping live totals combined", () => {
80
115
  const usageStats = {
81
116
  inputTokens: 0,
@@ -1,21 +1,5 @@
1
- import { mkdtempSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
1
  import { afterAll, beforeEach, describe, expect, mock, test } from "bun:test";
5
2
 
6
- const testDir = mkdtempSync(join(tmpdir(), "conv-wipe-test-"));
7
-
8
- mock.module("../util/platform.js", () => ({
9
- getDataDir: () => testDir,
10
- isMacOS: () => process.platform === "darwin",
11
- isLinux: () => process.platform === "linux",
12
- isWindows: () => process.platform === "win32",
13
- getPidPath: () => join(testDir, "test.pid"),
14
- getDbPath: () => join(testDir, "test.db"),
15
- getLogPath: () => join(testDir, "test.log"),
16
- ensureDataDir: () => {},
17
- }));
18
-
19
3
  mock.module("../util/logger.js", () => ({
20
4
  getLogger: () =>
21
5
  new Proxy({} as Record<string, unknown>, {
@@ -39,11 +23,6 @@ initializeDb();
39
23
 
40
24
  afterAll(() => {
41
25
  resetDb();
42
- try {
43
- rmSync(testDir, { recursive: true });
44
- } catch {
45
- /* best effort */
46
- }
47
26
  });
48
27
 
49
28
  describe("wipeConversation", () => {
@@ -16,10 +16,6 @@ mock.module("../util/logger.js", () => ({
16
16
  new Proxy({} as Record<string, unknown>, { get: () => () => {} }),
17
17
  }));
18
18
 
19
- mock.module("../util/platform.js", () => ({
20
- getDataDir: () => "/tmp",
21
- }));
22
-
23
19
  mock.module("../providers/registry.js", () => ({
24
20
  getProvider: () => ({ name: "mock-provider" }),
25
21
  initializeProviders: () => {},
@@ -21,10 +21,6 @@ mock.module("../util/logger.js", () => ({
21
21
  new Proxy({} as Record<string, unknown>, { get: () => () => {} }),
22
22
  }));
23
23
 
24
- mock.module("../util/platform.js", () => ({
25
- getDataDir: () => "/tmp",
26
- }));
27
-
28
24
  mock.module("../memory/guardian-action-store.js", () => ({
29
25
  getGuardianActionRequest: () => null,
30
26
  resolveGuardianActionRequest: () => {},
@@ -18,10 +18,6 @@ mock.module("../util/logger.js", () => ({
18
18
  new Proxy({} as Record<string, unknown>, { get: () => () => {} }),
19
19
  }));
20
20
 
21
- mock.module("../util/platform.js", () => ({
22
- getDataDir: () => "/tmp",
23
- }));
24
-
25
21
  mock.module("../memory/guardian-action-store.js", () => ({
26
22
  getGuardianActionRequest: () => null,
27
23
  resolveGuardianActionRequest: () => {},
@@ -10,10 +10,11 @@
10
10
  * - CLI commands deny raw secret/token reveal when VELLUM_UNTRUSTED_SHELL=1.
11
11
  */
12
12
 
13
+ import { join } from "node:path";
13
14
  import { describe, expect, test } from "bun:test";
14
15
 
15
16
  import { isUntrustedTrustClass } from "../runtime/actor-trust-resolver.js";
16
- import { getRootDir } from "../util/platform.js";
17
+ import { getProtectedDir, getWorkspaceDir } from "../util/platform.js";
17
18
 
18
19
  // ---------------------------------------------------------------------------
19
20
  // Trust class categorization (foundational for lockdown decisions)
@@ -120,10 +121,12 @@ describe("CES shell lockdown activation", () => {
120
121
 
121
122
  describe("CES protected paths for sandbox deny-read", () => {
122
123
  test("protected paths include the protected dir and db dir", () => {
123
- // The buildCesProtectedPaths function constructs paths from getRootDir().
124
- // We verify the pattern: paths should end with /protected and /workspace/data/db.
125
- const root = getRootDir();
126
- const expectedPaths = [`${root}/protected`, `${root}/workspace/data/db`];
124
+ // The buildCesProtectedPaths function constructs paths from getProtectedDir()
125
+ // and getWorkspaceDir(). We verify the pattern: paths should be absolute and clean.
126
+ const expectedPaths = [
127
+ getProtectedDir(),
128
+ join(getWorkspaceDir(), "data", "db"),
129
+ ];
127
130
 
128
131
  // Each expected path should be a valid absolute path pattern
129
132
  for (const p of expectedPaths) {