@vellumai/assistant 0.4.49 → 0.4.51

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 (353) hide show
  1. package/ARCHITECTURE.md +24 -33
  2. package/README.md +3 -3
  3. package/docs/architecture/integrations.md +2 -2
  4. package/docs/architecture/keychain-broker.md +6 -6
  5. package/docs/architecture/memory.md +180 -119
  6. package/knip.json +32 -0
  7. package/package.json +3 -2
  8. package/src/__tests__/agent-loop.test.ts +3 -1
  9. package/src/__tests__/anthropic-provider.test.ts +114 -23
  10. package/src/__tests__/approval-cascade.test.ts +1 -15
  11. package/src/__tests__/approval-routes-http.test.ts +2 -0
  12. package/src/__tests__/assistant-feature-flag-guard.test.ts +0 -23
  13. package/src/__tests__/btw-routes.test.ts +61 -5
  14. package/src/__tests__/canonical-guardian-store.test.ts +95 -0
  15. package/src/__tests__/checker.test.ts +13 -0
  16. package/src/__tests__/config-schema.test.ts +1 -68
  17. package/src/__tests__/config-watcher.test.ts +8 -0
  18. package/src/__tests__/context-memory-e2e.test.ts +11 -100
  19. package/src/__tests__/conversation-routes-guardian-reply.test.ts +8 -0
  20. package/src/__tests__/conversation-routes-slash-commands.test.ts +1 -0
  21. package/src/__tests__/credential-security-e2e.test.ts +1 -0
  22. package/src/__tests__/credential-security-invariants.test.ts +8 -7
  23. package/src/__tests__/credential-vault-unit.test.ts +23 -18
  24. package/src/__tests__/credential-vault.test.ts +30 -18
  25. package/src/__tests__/credentials-cli.test.ts +257 -82
  26. package/src/__tests__/cu-unified-flow.test.ts +532 -0
  27. package/src/__tests__/date-context.test.ts +93 -77
  28. package/src/__tests__/deterministic-verification-control-plane.test.ts +64 -0
  29. package/src/__tests__/guardian-routing-invariants.test.ts +93 -0
  30. package/src/__tests__/history-repair.test.ts +245 -0
  31. package/src/__tests__/host-cu-proxy.test.ts +165 -3
  32. package/src/__tests__/http-user-message-parity.test.ts +1 -0
  33. package/src/__tests__/inbound-invite-redemption.test.ts +36 -7
  34. package/src/__tests__/integration-status.test.ts +31 -30
  35. package/src/__tests__/invite-redemption-service.test.ts +166 -13
  36. package/src/__tests__/invite-routes-http.test.ts +166 -5
  37. package/src/__tests__/keychain-broker-client.test.ts +4 -4
  38. package/src/__tests__/list-messages-attachments.test.ts +193 -0
  39. package/src/__tests__/memory-context-benchmark.benchmark.test.ts +56 -18
  40. package/src/__tests__/memory-lifecycle-e2e.test.ts +244 -387
  41. package/src/__tests__/memory-recall-quality.test.ts +244 -407
  42. package/src/__tests__/memory-regressions.experimental.test.ts +126 -101
  43. package/src/__tests__/memory-regressions.test.ts +477 -2841
  44. package/src/__tests__/memory-retrieval.benchmark.test.ts +33 -150
  45. package/src/__tests__/memory-upsert-concurrency.test.ts +5 -244
  46. package/src/__tests__/mime-builder.test.ts +28 -0
  47. package/src/__tests__/native-web-search.test.ts +1 -0
  48. package/src/__tests__/oauth-cli.test.ts +824 -31
  49. package/src/__tests__/oauth-provider-profiles.test.ts +1 -1
  50. package/src/__tests__/oauth-store.test.ts +363 -17
  51. package/src/__tests__/qdrant-collection-migration.test.ts +53 -8
  52. package/src/__tests__/registry.test.ts +0 -1
  53. package/src/__tests__/relay-server.test.ts +55 -1
  54. package/src/__tests__/schedule-tools.test.ts +32 -0
  55. package/src/__tests__/script-proxy-certs.test.ts +1 -1
  56. package/src/__tests__/secret-onetime-send.test.ts +1 -0
  57. package/src/__tests__/secret-routes-managed-proxy.test.ts +183 -0
  58. package/src/__tests__/secure-keys.test.ts +78 -18
  59. package/src/__tests__/send-endpoint-busy.test.ts +3 -0
  60. package/src/__tests__/server-history-render.test.ts +2 -2
  61. package/src/__tests__/session-abort-tool-results.test.ts +1 -14
  62. package/src/__tests__/session-agent-loop-overflow.test.ts +1583 -0
  63. package/src/__tests__/session-agent-loop.test.ts +19 -15
  64. package/src/__tests__/session-confirmation-signals.test.ts +1 -15
  65. package/src/__tests__/session-error.test.ts +124 -2
  66. package/src/__tests__/session-history-web-search.test.ts +918 -0
  67. package/src/__tests__/session-pre-run-repair.test.ts +1 -14
  68. package/src/__tests__/session-provider-retry-repair.test.ts +25 -28
  69. package/src/__tests__/session-queue.test.ts +37 -27
  70. package/src/__tests__/session-runtime-assembly.test.ts +54 -0
  71. package/src/__tests__/session-slash-known.test.ts +1 -15
  72. package/src/__tests__/session-slash-queue.test.ts +1 -15
  73. package/src/__tests__/session-slash-unknown.test.ts +1 -15
  74. package/src/__tests__/session-workspace-cache-state.test.ts +3 -33
  75. package/src/__tests__/session-workspace-injection.test.ts +3 -37
  76. package/src/__tests__/session-workspace-tool-tracking.test.ts +3 -37
  77. package/src/__tests__/skills-install-extract.test.ts +93 -0
  78. package/src/__tests__/skills.test.ts +2 -2
  79. package/src/__tests__/skillssh-registry.test.ts +451 -0
  80. package/src/__tests__/slack-channel-config.test.ts +10 -8
  81. package/src/__tests__/trust-store.test.ts +15 -0
  82. package/src/__tests__/twilio-config.test.ts +11 -10
  83. package/src/__tests__/twilio-provider.test.ts +9 -4
  84. package/src/__tests__/voice-invite-redemption.test.ts +85 -5
  85. package/src/agent/ax-tree-compaction.test.ts +51 -0
  86. package/src/agent/loop.ts +39 -12
  87. package/src/approvals/AGENTS.md +1 -1
  88. package/src/approvals/guardian-request-resolvers.ts +14 -2
  89. package/src/bundler/compiler-tools.ts +66 -2
  90. package/src/calls/call-domain.ts +134 -3
  91. package/src/calls/call-store.ts +6 -0
  92. package/src/calls/relay-server.ts +44 -6
  93. package/src/calls/relay-setup-router.ts +17 -1
  94. package/src/calls/twilio-config.ts +5 -4
  95. package/src/calls/twilio-provider.ts +14 -9
  96. package/src/calls/twilio-rest.ts +10 -7
  97. package/src/calls/types.ts +3 -1
  98. package/src/cli/commands/config.ts +14 -9
  99. package/src/cli/commands/contacts.ts +3 -0
  100. package/src/cli/commands/credentials.ts +170 -174
  101. package/src/cli/commands/doctor.ts +11 -8
  102. package/src/cli/commands/keys.ts +9 -9
  103. package/src/cli/commands/mcp.ts +46 -59
  104. package/src/cli/commands/memory.ts +16 -165
  105. package/src/cli/commands/oauth/apps.ts +68 -10
  106. package/src/cli/commands/oauth/connections.ts +475 -105
  107. package/src/cli/commands/oauth/index.ts +3 -3
  108. package/src/cli/commands/oauth/providers.ts +18 -4
  109. package/src/cli/commands/sessions.ts +5 -2
  110. package/src/cli/commands/skills.ts +173 -1
  111. package/src/cli/http-client.ts +0 -20
  112. package/src/cli/main-screen.tsx +2 -2
  113. package/src/cli/program.ts +5 -6
  114. package/src/cli.ts +20 -22
  115. package/src/config/__tests__/feature-flag-registry-bundled.test.ts +39 -0
  116. package/src/config/bundled-skills/computer-use/TOOLS.json +1 -1
  117. package/src/config/bundled-skills/computer-use/tools/computer-use-observe.ts +12 -0
  118. package/src/config/bundled-skills/contacts/SKILL.md +35 -11
  119. package/src/config/bundled-skills/contacts/tools/google-contacts.ts +1 -1
  120. package/src/config/bundled-skills/gmail/SKILL.md +1 -1
  121. package/src/config/bundled-skills/gmail/TOOLS.json +52 -0
  122. package/src/config/bundled-skills/gmail/tools/gmail-archive.ts +13 -3
  123. package/src/config/bundled-skills/gmail/tools/gmail-attachments.ts +9 -2
  124. package/src/config/bundled-skills/gmail/tools/gmail-draft.ts +5 -1
  125. package/src/config/bundled-skills/gmail/tools/gmail-filters.ts +5 -1
  126. package/src/config/bundled-skills/gmail/tools/gmail-follow-up.ts +5 -1
  127. package/src/config/bundled-skills/gmail/tools/gmail-forward.ts +5 -1
  128. package/src/config/bundled-skills/gmail/tools/gmail-label.ts +9 -2
  129. package/src/config/bundled-skills/gmail/tools/gmail-outreach-scan.ts +5 -1
  130. package/src/config/bundled-skills/gmail/tools/gmail-send-draft.ts +5 -1
  131. package/src/config/bundled-skills/gmail/tools/gmail-sender-digest.ts +5 -1
  132. package/src/config/bundled-skills/gmail/tools/gmail-trash.ts +5 -1
  133. package/src/config/bundled-skills/gmail/tools/gmail-unsubscribe.ts +5 -1
  134. package/src/config/bundled-skills/gmail/tools/gmail-vacation.ts +5 -1
  135. package/src/config/bundled-skills/google-calendar/TOOLS.json +20 -0
  136. package/src/config/bundled-skills/google-calendar/tools/calendar-check-availability.ts +2 -1
  137. package/src/config/bundled-skills/google-calendar/tools/calendar-create-event.ts +2 -1
  138. package/src/config/bundled-skills/google-calendar/tools/calendar-get-event.ts +2 -1
  139. package/src/config/bundled-skills/google-calendar/tools/calendar-list-events.ts +2 -1
  140. package/src/config/bundled-skills/google-calendar/tools/calendar-rsvp.ts +2 -1
  141. package/src/config/bundled-skills/google-calendar/tools/shared.ts +8 -2
  142. package/src/config/bundled-skills/messaging/SKILL.md +1 -1
  143. package/src/config/bundled-skills/messaging/tools/messaging-analyze-style.ts +2 -2
  144. package/src/config/bundled-skills/messaging/tools/messaging-archive-by-sender.ts +2 -2
  145. package/src/config/bundled-skills/messaging/tools/messaging-auth-test.ts +2 -2
  146. package/src/config/bundled-skills/messaging/tools/messaging-list-conversations.ts +2 -2
  147. package/src/config/bundled-skills/messaging/tools/messaging-mark-read.ts +2 -2
  148. package/src/config/bundled-skills/messaging/tools/messaging-read.ts +2 -2
  149. package/src/config/bundled-skills/messaging/tools/messaging-search.ts +2 -2
  150. package/src/config/bundled-skills/messaging/tools/messaging-send.ts +2 -2
  151. package/src/config/bundled-skills/messaging/tools/messaging-sender-digest.ts +2 -2
  152. package/src/config/bundled-skills/messaging/tools/shared.ts +7 -5
  153. package/src/config/bundled-skills/slack/tools/shared.ts +1 -1
  154. package/src/config/bundled-skills/slack/tools/slack-add-reaction.ts +1 -1
  155. package/src/config/bundled-skills/slack/tools/slack-channel-details.ts +1 -1
  156. package/src/config/bundled-skills/slack/tools/slack-delete-message.ts +1 -1
  157. package/src/config/bundled-skills/slack/tools/slack-edit-message.ts +1 -1
  158. package/src/config/bundled-skills/slack/tools/slack-leave-channel.ts +1 -1
  159. package/src/config/bundled-skills/slack/tools/slack-scan-digest.ts +1 -1
  160. package/src/config/bundled-tool-registry.ts +2 -5
  161. package/src/config/loader.ts +6 -42
  162. package/src/config/schema.ts +1 -12
  163. package/src/config/schemas/memory-lifecycle.ts +0 -9
  164. package/src/config/schemas/memory-processing.ts +0 -180
  165. package/src/config/schemas/memory-retrieval.ts +32 -104
  166. package/src/config/schemas/memory.ts +0 -10
  167. package/src/config/types.ts +0 -4
  168. package/src/contacts/contact-store.ts +39 -2
  169. package/src/contacts/contacts-write.ts +9 -0
  170. package/src/context/window-manager.ts +4 -1
  171. package/src/daemon/config-watcher.ts +55 -2
  172. package/src/daemon/daemon-control.ts +1 -1
  173. package/src/daemon/date-context.ts +114 -31
  174. package/src/daemon/handlers/config-ingress.ts +2 -2
  175. package/src/daemon/handlers/config-slack-channel.ts +59 -39
  176. package/src/daemon/handlers/config-telegram.ts +23 -14
  177. package/src/daemon/handlers/session-history.ts +1 -358
  178. package/src/daemon/handlers/sessions.ts +18 -13
  179. package/src/daemon/handlers/shared.ts +3 -17
  180. package/src/daemon/handlers/skills.ts +20 -1
  181. package/src/daemon/history-repair.ts +72 -8
  182. package/src/daemon/host-cu-proxy.ts +55 -26
  183. package/src/daemon/lifecycle.ts +39 -4
  184. package/src/daemon/mcp-reload-service.ts +2 -2
  185. package/src/daemon/message-types/computer-use.ts +1 -12
  186. package/src/daemon/message-types/memory.ts +4 -16
  187. package/src/daemon/message-types/messages.ts +1 -0
  188. package/src/daemon/message-types/sessions.ts +4 -42
  189. package/src/daemon/server.ts +6 -1
  190. package/src/daemon/session-agent-loop-handlers.ts +38 -0
  191. package/src/daemon/session-agent-loop.ts +334 -48
  192. package/src/daemon/session-error.ts +89 -6
  193. package/src/daemon/session-history.ts +17 -7
  194. package/src/daemon/session-media-retry.ts +6 -2
  195. package/src/daemon/session-memory.ts +69 -149
  196. package/src/daemon/session-process.ts +10 -1
  197. package/src/daemon/session-runtime-assembly.ts +49 -19
  198. package/src/daemon/session-slash.ts +3 -5
  199. package/src/daemon/session-surfaces.ts +4 -1
  200. package/src/daemon/session-tool-setup.ts +7 -1
  201. package/src/daemon/session.ts +12 -2
  202. package/src/email/providers/index.ts +2 -2
  203. package/src/instrument.ts +61 -1
  204. package/src/media/avatar-router.ts +1 -1
  205. package/src/memory/admin.ts +2 -191
  206. package/src/memory/canonical-guardian-store.ts +38 -2
  207. package/src/memory/conversation-crud.ts +0 -33
  208. package/src/memory/conversation-queries.ts +25 -83
  209. package/src/memory/db-init.ts +32 -0
  210. package/src/memory/embedding-backend.ts +84 -8
  211. package/src/memory/embedding-types.ts +9 -1
  212. package/src/memory/indexer.ts +7 -46
  213. package/src/memory/invite-store.ts +19 -0
  214. package/src/memory/items-extractor.ts +274 -76
  215. package/src/memory/job-handlers/backfill.ts +2 -127
  216. package/src/memory/job-handlers/cleanup.ts +2 -16
  217. package/src/memory/job-handlers/extraction.ts +2 -138
  218. package/src/memory/job-handlers/index-maintenance.ts +1 -6
  219. package/src/memory/job-handlers/summarization.ts +3 -148
  220. package/src/memory/job-utils.ts +21 -59
  221. package/src/memory/jobs-store.ts +1 -159
  222. package/src/memory/jobs-worker.ts +9 -52
  223. package/src/memory/migrations/104-core-indexes.ts +3 -3
  224. package/src/memory/migrations/149-oauth-tables.ts +2 -0
  225. package/src/memory/migrations/150-oauth-apps-client-secret-path.ts +98 -0
  226. package/src/memory/migrations/151-oauth-providers-ping-url.ts +11 -0
  227. package/src/memory/migrations/152-memory-item-supersession.ts +44 -0
  228. package/src/memory/migrations/153-drop-entity-tables.ts +15 -0
  229. package/src/memory/migrations/154-drop-fts.ts +20 -0
  230. package/src/memory/migrations/155-drop-conflicts.ts +7 -0
  231. package/src/memory/migrations/156-call-session-invite-metadata.ts +24 -0
  232. package/src/memory/migrations/157-invite-contact-id.ts +104 -0
  233. package/src/memory/migrations/index.ts +8 -0
  234. package/src/memory/migrations/registry.ts +6 -0
  235. package/src/memory/qdrant-client.ts +148 -51
  236. package/src/memory/raw-query.ts +1 -1
  237. package/src/memory/retriever.test.ts +294 -273
  238. package/src/memory/retriever.ts +421 -645
  239. package/src/memory/schema/calls.ts +2 -0
  240. package/src/memory/schema/contacts.ts +1 -0
  241. package/src/memory/schema/memory-core.ts +3 -48
  242. package/src/memory/schema/oauth.ts +2 -0
  243. package/src/memory/search/formatting.ts +263 -176
  244. package/src/memory/search/lexical.ts +1 -254
  245. package/src/memory/search/ranking.ts +0 -455
  246. package/src/memory/search/semantic.ts +100 -14
  247. package/src/memory/search/staleness.ts +47 -0
  248. package/src/memory/search/tier-classifier.ts +21 -0
  249. package/src/memory/search/types.ts +15 -77
  250. package/src/memory/task-memory-cleanup.ts +4 -6
  251. package/src/messaging/provider.ts +1 -1
  252. package/src/messaging/providers/gmail/adapter.ts +1 -1
  253. package/src/messaging/providers/gmail/mime-builder.ts +17 -7
  254. package/src/messaging/providers/telegram-bot/adapter.ts +17 -8
  255. package/src/messaging/providers/whatsapp/adapter.ts +13 -9
  256. package/src/messaging/registry.ts +9 -5
  257. package/src/oauth/byo-connection.test.ts +40 -25
  258. package/src/oauth/connect-orchestrator.ts +4 -10
  259. package/src/oauth/connection-resolver.ts +20 -6
  260. package/src/oauth/manual-token-connection.ts +5 -5
  261. package/src/oauth/oauth-store.ts +183 -31
  262. package/src/oauth/platform-connection.test.ts +1 -1
  263. package/src/oauth/provider-behaviors.ts +503 -4
  264. package/src/oauth/seed-providers.ts +214 -8
  265. package/src/oauth/token-persistence.ts +31 -16
  266. package/src/permissions/defaults.ts +1 -0
  267. package/src/permissions/trust-store.ts +23 -1
  268. package/src/playbooks/playbook-compiler.ts +1 -1
  269. package/src/prompts/system-prompt.ts +18 -2
  270. package/src/providers/anthropic/client.ts +56 -126
  271. package/src/providers/types.ts +7 -1
  272. package/src/runtime/AGENTS.md +9 -0
  273. package/src/runtime/auth/route-policy.ts +6 -3
  274. package/src/runtime/channel-readiness-service.ts +48 -40
  275. package/src/runtime/guardian-reply-router.ts +24 -22
  276. package/src/runtime/http-server.ts +2 -2
  277. package/src/runtime/http-types.ts +2 -0
  278. package/src/runtime/invite-redemption-service.ts +72 -12
  279. package/src/runtime/invite-service.ts +43 -0
  280. package/src/runtime/middleware/twilio-validation.ts +1 -1
  281. package/src/runtime/pending-interactions.ts +2 -2
  282. package/src/runtime/routes/brain-graph-routes.ts +10 -90
  283. package/src/runtime/routes/btw-routes.ts +10 -5
  284. package/src/runtime/routes/conversation-routes.ts +56 -11
  285. package/src/runtime/routes/inbound-stages/acl-enforcement.ts +21 -12
  286. package/src/runtime/routes/integrations/slack/channel.ts +2 -2
  287. package/src/runtime/routes/integrations/telegram.ts +2 -2
  288. package/src/runtime/routes/integrations/twilio.ts +17 -17
  289. package/src/runtime/routes/invite-routes.ts +29 -4
  290. package/src/runtime/routes/memory-item-routes.test.ts +754 -0
  291. package/src/runtime/routes/memory-item-routes.ts +503 -0
  292. package/src/runtime/routes/secret-routes.ts +17 -0
  293. package/src/runtime/routes/session-management-routes.ts +3 -3
  294. package/src/runtime/routes/settings-routes.ts +3 -3
  295. package/src/runtime/routes/trust-rules-routes.ts +14 -0
  296. package/src/runtime/routes/workspace-routes.ts +9 -4
  297. package/src/runtime/routes/workspace-utils.ts +8 -2
  298. package/src/schedule/integration-status.ts +26 -19
  299. package/src/security/keychain-broker-client.ts +17 -4
  300. package/src/security/oauth2.ts +6 -7
  301. package/src/security/secure-keys.ts +44 -19
  302. package/src/security/token-manager.ts +46 -39
  303. package/src/services/vercel-deploy.ts +0 -24
  304. package/src/signals/confirm.ts +78 -0
  305. package/src/signals/mcp-reload.ts +18 -0
  306. package/src/skills/catalog-install.ts +74 -18
  307. package/src/skills/skillssh-registry.ts +503 -0
  308. package/src/tools/assets/search.ts +5 -1
  309. package/src/tools/computer-use/definitions.ts +0 -10
  310. package/src/tools/computer-use/registry.ts +1 -1
  311. package/src/tools/credentials/vault.ts +22 -7
  312. package/src/tools/memory/definitions.ts +4 -13
  313. package/src/tools/memory/handlers.test.ts +83 -103
  314. package/src/tools/memory/handlers.ts +50 -85
  315. package/src/tools/network/script-proxy/session-manager.ts +8 -8
  316. package/src/tools/schedule/create.ts +10 -3
  317. package/src/tools/schedule/update.ts +8 -1
  318. package/src/tools/skills/load.ts +25 -2
  319. package/src/watcher/provider-types.ts +1 -1
  320. package/src/watcher/providers/github.ts +1 -1
  321. package/src/watcher/providers/gmail.ts +3 -3
  322. package/src/watcher/providers/google-calendar.ts +3 -3
  323. package/src/watcher/providers/linear.ts +1 -1
  324. package/src/__tests__/clarification-resolver.test.ts +0 -193
  325. package/src/__tests__/conflict-intent-tokenization.test.ts +0 -160
  326. package/src/__tests__/conflict-policy.test.ts +0 -269
  327. package/src/__tests__/conflict-store.test.ts +0 -372
  328. package/src/__tests__/contradiction-checker.test.ts +0 -361
  329. package/src/__tests__/entity-extractor.test.ts +0 -211
  330. package/src/__tests__/entity-search.test.ts +0 -1117
  331. package/src/__tests__/profile-compiler.test.ts +0 -392
  332. package/src/__tests__/session-conflict-gate.test.ts +0 -1228
  333. package/src/__tests__/session-profile-injection.test.ts +0 -557
  334. package/src/config/bundled-skills/knowledge-graph/SKILL.md +0 -25
  335. package/src/config/bundled-skills/knowledge-graph/TOOLS.json +0 -66
  336. package/src/config/bundled-skills/knowledge-graph/tools/graph-query.ts +0 -211
  337. package/src/daemon/session-conflict-gate.ts +0 -167
  338. package/src/daemon/session-dynamic-profile.ts +0 -77
  339. package/src/memory/clarification-resolver.ts +0 -417
  340. package/src/memory/conflict-intent.ts +0 -205
  341. package/src/memory/conflict-policy.ts +0 -127
  342. package/src/memory/conflict-store.ts +0 -410
  343. package/src/memory/contradiction-checker.ts +0 -508
  344. package/src/memory/entity-extractor.ts +0 -535
  345. package/src/memory/format-recall.ts +0 -47
  346. package/src/memory/fts-reconciler.ts +0 -165
  347. package/src/memory/job-handlers/conflict.ts +0 -200
  348. package/src/memory/profile-compiler.ts +0 -195
  349. package/src/memory/recall-cache.ts +0 -117
  350. package/src/memory/search/entity.ts +0 -535
  351. package/src/memory/search/query-expansion.test.ts +0 -70
  352. package/src/memory/search/query-expansion.ts +0 -118
  353. package/src/runtime/routes/mcp-routes.ts +0 -20
@@ -1,535 +0,0 @@
1
- import { and, desc, eq, inArray, isNull, or } from "drizzle-orm";
2
-
3
- import type { MemoryEntityConfig } from "../../config/types.js";
4
- import { getLogger } from "../../util/logger.js";
5
- import { getDb, rawAll } from "../db.js";
6
- import {
7
- memoryEntityRelations,
8
- memoryItemEntities,
9
- memoryItems,
10
- memoryItemSources,
11
- } from "../schema.js";
12
- import { computeRecencyScore } from "./ranking.js";
13
- import type {
14
- Candidate,
15
- CandidateSource,
16
- CandidateType,
17
- EntitySearchResult,
18
- MatchedEntityRow,
19
- TraversalOptions,
20
- TraversalResult,
21
- TraversalStep,
22
- } from "./types.js";
23
-
24
- const log = getLogger("memory-retriever");
25
-
26
- /**
27
- * Entity-based retrieval: match seed entities from query text, fetch directly
28
- * linked items, and optionally expand one hop across entity relations.
29
- */
30
- export function entitySearch(
31
- query: string,
32
- entityConfig: MemoryEntityConfig,
33
- scopeIds?: string[],
34
- excludedMessageIds: string[] = [],
35
- ): EntitySearchResult {
36
- const trimmed = query.trim();
37
- if (trimmed.length === 0) return emptyEntitySearchResult();
38
-
39
- const relationConfig = entityConfig.relationRetrieval;
40
- const matchedEntities = findMatchedEntities(
41
- trimmed,
42
- relationConfig.enabled ? relationConfig.maxSeedEntities : 20,
43
- );
44
- if (matchedEntities.length === 0) return emptyEntitySearchResult();
45
-
46
- const seedEntityIds = matchedEntities.map((row) => row.id);
47
- const directCandidates = getEntityLinkedItemCandidates(seedEntityIds, {
48
- scopeIds,
49
- excludedMessageIds,
50
- source: "entity_direct",
51
- });
52
-
53
- if (!relationConfig.enabled) {
54
- return {
55
- candidates: directCandidates,
56
- relationSeedEntityCount: 0,
57
- relationTraversedEdgeCount: 0,
58
- relationNeighborEntityCount: 0,
59
- relationExpandedItemCount: 0,
60
- };
61
- }
62
-
63
- const relationSeedEntityCount = seedEntityIds.length;
64
-
65
- const {
66
- neighborEntityIds,
67
- traversedEdgeCount: relationTraversedEdgeCount,
68
- neighborDepths,
69
- } = findNeighborEntities(seedEntityIds, {
70
- maxEdges: relationConfig.maxEdges,
71
- maxNeighborEntities: relationConfig.maxNeighborEntities,
72
- maxDepth: relationConfig.maxDepth,
73
- });
74
- const relationNeighborEntityCount = neighborEntityIds.length;
75
- const directItemIds = new Set(
76
- directCandidates.map((candidate) => candidate.id),
77
- );
78
- const relationCandidates = getEntityLinkedItemCandidates(neighborEntityIds, {
79
- scopeIds,
80
- excludedMessageIds,
81
- source: "entity_relation",
82
- excludeItemIds: directItemIds,
83
- });
84
- const relationExpandedItemCount = relationCandidates.length;
85
-
86
- // Build candidate key → BFS depth map so ranking can apply distance-based decay
87
- const candidateDepths = new Map<string, number>();
88
- if (relationCandidates.length > 0 && neighborDepths.size > 0) {
89
- const db = getDb();
90
- const itemIds = relationCandidates.map((c) => c.id);
91
- const links = db
92
- .select({
93
- memoryItemId: memoryItemEntities.memoryItemId,
94
- entityId: memoryItemEntities.entityId,
95
- })
96
- .from(memoryItemEntities)
97
- .where(inArray(memoryItemEntities.memoryItemId, itemIds))
98
- .all();
99
-
100
- // For each item, find the minimum depth among its linked neighbor entities
101
- const itemDepthMap = new Map<string, number>();
102
- for (const link of links) {
103
- const depth = neighborDepths.get(link.entityId);
104
- if (depth !== undefined) {
105
- const existing = itemDepthMap.get(link.memoryItemId);
106
- if (existing === undefined || depth < existing) {
107
- itemDepthMap.set(link.memoryItemId, depth);
108
- }
109
- }
110
- }
111
-
112
- for (const candidate of relationCandidates) {
113
- const depth = itemDepthMap.get(candidate.id);
114
- if (depth !== undefined) {
115
- candidateDepths.set(candidate.key, depth);
116
- }
117
- }
118
- }
119
-
120
- return {
121
- candidates: [...directCandidates, ...relationCandidates],
122
- relationSeedEntityCount,
123
- relationTraversedEdgeCount,
124
- relationNeighborEntityCount,
125
- relationExpandedItemCount,
126
- candidateDepths,
127
- };
128
- }
129
-
130
- export function emptyEntitySearchResult(): EntitySearchResult {
131
- return {
132
- candidates: [],
133
- relationSeedEntityCount: 0,
134
- relationTraversedEdgeCount: 0,
135
- relationNeighborEntityCount: 0,
136
- relationExpandedItemCount: 0,
137
- candidateDepths: new Map(),
138
- };
139
- }
140
-
141
- export function findMatchedEntities(
142
- query: string,
143
- maxMatches: number,
144
- ): MatchedEntityRow[] {
145
- const trimmed = query.trim();
146
- if (trimmed.length === 0) return [];
147
-
148
- const safeLimit = Math.max(1, Math.floor(maxMatches));
149
-
150
- // Tokenize query into words for entity matching (min length 3 to reduce false positives)
151
- const tokens = trimmed
152
- .toLowerCase()
153
- .split(/[^a-z0-9_.-]+/g)
154
- .filter((t) => t.length >= 3);
155
- const fullQuery = trimmed.toLowerCase();
156
-
157
- // Use exact matching on entity names and json_each() for individual alias values.
158
- // Also match the full trimmed query to support multi-word entity names (e.g. "Visual Studio Code").
159
- // When tokens is empty (all words < 3 chars), only match on fullQuery.
160
- let entityQuery: string;
161
- let queryParams: string[];
162
- if (tokens.length > 0) {
163
- const namePlaceholders = tokens.map(() => "?").join(",");
164
- entityQuery = `
165
- SELECT DISTINCT me.id, me.name, me.type, me.aliases, me.mention_count
166
- FROM memory_entities me
167
- WHERE LOWER(me.name) IN (${namePlaceholders}) OR LOWER(me.name) = ?
168
- UNION
169
- SELECT DISTINCT me.id, me.name, me.type, me.aliases, me.mention_count
170
- FROM memory_entities me, json_each(me.aliases) je
171
- WHERE me.aliases IS NOT NULL AND (LOWER(je.value) IN (${namePlaceholders}) OR LOWER(je.value) = ?)
172
- LIMIT ${safeLimit}
173
- `;
174
- queryParams = [...tokens, fullQuery, ...tokens, fullQuery];
175
- } else {
176
- entityQuery = `
177
- SELECT DISTINCT me.id, me.name, me.type, me.aliases, me.mention_count
178
- FROM memory_entities me
179
- WHERE LOWER(me.name) = ?
180
- UNION
181
- SELECT DISTINCT me.id, me.name, me.type, me.aliases, me.mention_count
182
- FROM memory_entities me, json_each(me.aliases) je
183
- WHERE me.aliases IS NOT NULL AND LOWER(je.value) = ?
184
- LIMIT ${safeLimit}
185
- `;
186
- queryParams = [fullQuery, fullQuery];
187
- }
188
-
189
- try {
190
- return rawAll<MatchedEntityRow>(entityQuery, ...queryParams);
191
- } catch (err) {
192
- log.warn({ err }, "Entity search query failed");
193
- return [];
194
- }
195
- }
196
-
197
- /**
198
- * BFS traversal across entity relations with visited-set cycle detection
199
- * and configurable max depth to prevent unbounded graph walking.
200
- */
201
- export function findNeighborEntities(
202
- seedEntityIds: string[],
203
- opts: TraversalOptions,
204
- ): TraversalResult {
205
- const {
206
- maxEdges,
207
- maxNeighborEntities,
208
- maxDepth = 3,
209
- relationTypes,
210
- entityTypes,
211
- directed,
212
- } = opts;
213
- if (
214
- seedEntityIds.length === 0 ||
215
- maxEdges <= 0 ||
216
- maxNeighborEntities <= 0 ||
217
- maxDepth <= 0
218
- ) {
219
- return {
220
- neighborEntityIds: [],
221
- traversedEdgeCount: 0,
222
- neighborDepths: new Map(),
223
- };
224
- }
225
-
226
- const db = getDb();
227
- const visited = new Set<string>(seedEntityIds);
228
- const neighbors: string[] = [];
229
- const neighborDepths = new Map<string, number>();
230
- let totalEdgesTraversed = 0;
231
- const filterByEntityType = entityTypes && entityTypes.length > 0;
232
-
233
- // BFS frontier starts with seed entities
234
- let frontier = [...seedEntityIds];
235
-
236
- for (let depth = 0; depth < maxDepth; depth++) {
237
- if (frontier.length === 0 || neighbors.length >= maxNeighborEntities) break;
238
-
239
- const edgeBudget = maxEdges - totalEdgesTraversed;
240
- if (edgeBudget <= 0) break;
241
-
242
- let rows: Array<{ sourceEntityId: string; targetEntityId: string }>;
243
-
244
- if (filterByEntityType) {
245
- // When filtering by entity type, JOIN with memoryEntities on the neighbor
246
- // side so non-matching edges are excluded at the SQL level and don't
247
- // consume the edge budget.
248
- const relationTypeCondition =
249
- relationTypes && relationTypes.length > 0
250
- ? `AND r.relation IN (${relationTypes.map(() => "?").join(",")})`
251
- : "";
252
- const entityTypeFilter = `AND me.type IN (${entityTypes
253
- .map(() => "?")
254
- .join(",")})`;
255
- const frontierPlaceholders = frontier.map(() => "?").join(",");
256
- const limit = Math.max(1, edgeBudget);
257
-
258
- const relationParams =
259
- relationTypes && relationTypes.length > 0 ? relationTypes : [];
260
-
261
- type EdgeRow = { sourceEntityId: string; targetEntityId: string };
262
-
263
- if (directed) {
264
- // GROUP BY deduplicates entity pairs that have multiple relation rows
265
- const q1 = `
266
- SELECT r.source_entity_id AS sourceEntityId, r.target_entity_id AS targetEntityId
267
- FROM memory_entity_relations r
268
- INNER JOIN memory_entities me ON me.id = r.target_entity_id
269
- WHERE r.source_entity_id IN (${frontierPlaceholders})
270
- ${relationTypeCondition} ${entityTypeFilter}
271
- GROUP BY r.source_entity_id, r.target_entity_id
272
- ORDER BY MAX(r.last_seen_at) DESC
273
- LIMIT ?
274
- `;
275
- rows = rawAll<EdgeRow>(
276
- q1,
277
- ...frontier,
278
- ...relationParams,
279
- ...entityTypes,
280
- limit,
281
- );
282
- } else {
283
- // Combine both directions in a single query with global recency
284
- // ordering so the edge budget isn't direction-biased.
285
- const q = `
286
- SELECT sourceEntityId, targetEntityId FROM (
287
- SELECT r.source_entity_id AS sourceEntityId, r.target_entity_id AS targetEntityId, r.last_seen_at
288
- FROM memory_entity_relations r
289
- INNER JOIN memory_entities me ON me.id = r.target_entity_id
290
- WHERE r.source_entity_id IN (${frontierPlaceholders})
291
- ${relationTypeCondition} ${entityTypeFilter}
292
- UNION ALL
293
- SELECT r.source_entity_id AS sourceEntityId, r.target_entity_id AS targetEntityId, r.last_seen_at
294
- FROM memory_entity_relations r
295
- INNER JOIN memory_entities me ON me.id = r.source_entity_id
296
- WHERE r.target_entity_id IN (${frontierPlaceholders})
297
- ${relationTypeCondition} ${entityTypeFilter}
298
- )
299
- GROUP BY sourceEntityId, targetEntityId
300
- ORDER BY MAX(last_seen_at) DESC
301
- LIMIT ?
302
- `;
303
- rows = rawAll<EdgeRow>(
304
- q,
305
- ...frontier,
306
- ...relationParams,
307
- ...entityTypes,
308
- ...frontier,
309
- ...relationParams,
310
- ...entityTypes,
311
- limit,
312
- );
313
- }
314
- } else {
315
- const frontierCondition = directed
316
- ? inArray(memoryEntityRelations.sourceEntityId, frontier)
317
- : or(
318
- inArray(memoryEntityRelations.sourceEntityId, frontier),
319
- inArray(memoryEntityRelations.targetEntityId, frontier),
320
- );
321
- const whereCondition =
322
- relationTypes && relationTypes.length > 0
323
- ? and(
324
- frontierCondition,
325
- inArray(memoryEntityRelations.relation, relationTypes),
326
- )
327
- : frontierCondition;
328
-
329
- rows = db
330
- .select({
331
- sourceEntityId: memoryEntityRelations.sourceEntityId,
332
- targetEntityId: memoryEntityRelations.targetEntityId,
333
- })
334
- .from(memoryEntityRelations)
335
- .where(whereCondition)
336
- .orderBy(desc(memoryEntityRelations.lastSeenAt))
337
- .limit(Math.max(1, edgeBudget))
338
- .all();
339
- }
340
-
341
- totalEdgesTraversed += rows.length;
342
-
343
- const nextFrontier: string[] = [];
344
- const frontierSet = new Set(frontier);
345
- for (const row of rows) {
346
- if (neighbors.length >= maxNeighborEntities) break;
347
- // In directed mode, only follow source→target (frontier is always on source side)
348
- if (
349
- frontierSet.has(row.sourceEntityId) &&
350
- !visited.has(row.targetEntityId)
351
- ) {
352
- visited.add(row.targetEntityId);
353
- neighbors.push(row.targetEntityId);
354
- nextFrontier.push(row.targetEntityId);
355
- neighborDepths.set(row.targetEntityId, depth + 1);
356
- }
357
- if (directed) continue;
358
- if (neighbors.length >= maxNeighborEntities) break;
359
- if (
360
- frontierSet.has(row.targetEntityId) &&
361
- !visited.has(row.sourceEntityId)
362
- ) {
363
- visited.add(row.sourceEntityId);
364
- neighbors.push(row.sourceEntityId);
365
- nextFrontier.push(row.sourceEntityId);
366
- neighborDepths.set(row.sourceEntityId, depth + 1);
367
- }
368
- }
369
-
370
- frontier = nextFrontier;
371
- }
372
-
373
- return {
374
- neighborEntityIds: neighbors.slice(0, maxNeighborEntities),
375
- traversedEdgeCount: totalEdgesTraversed,
376
- neighborDepths,
377
- };
378
- }
379
-
380
- export function getEntityLinkedItemCandidates(
381
- entityIds: string[],
382
- opts: {
383
- scopeIds?: string[];
384
- excludedMessageIds?: string[];
385
- source: CandidateSource;
386
- excludeItemIds?: Set<string>;
387
- },
388
- ): Candidate[] {
389
- if (entityIds.length === 0) return [];
390
- const excludedMessageIds = opts.excludedMessageIds ?? [];
391
-
392
- const db = getDb();
393
- const linkedRows = db
394
- .select({
395
- memoryItemId: memoryItemEntities.memoryItemId,
396
- })
397
- .from(memoryItemEntities)
398
- .where(inArray(memoryItemEntities.entityId, entityIds))
399
- .all();
400
-
401
- if (linkedRows.length === 0) return [];
402
-
403
- const itemIds = [
404
- ...new Set(linkedRows.map((row) => row.memoryItemId)),
405
- ].filter((itemId) => !opts.excludeItemIds?.has(itemId));
406
- if (itemIds.length === 0) return [];
407
-
408
- const itemConditions = [
409
- inArray(memoryItems.id, itemIds),
410
- eq(memoryItems.status, "active"),
411
- isNull(memoryItems.invalidAt),
412
- ];
413
- if (opts.scopeIds && opts.scopeIds.length > 0) {
414
- itemConditions.push(inArray(memoryItems.scopeId, opts.scopeIds));
415
- }
416
- let items = db
417
- .select()
418
- .from(memoryItems)
419
- .where(and(...itemConditions))
420
- .all();
421
- if (items.length === 0) return [];
422
-
423
- if (excludedMessageIds.length > 0) {
424
- const excludedSet = new Set(excludedMessageIds);
425
- const sources = db
426
- .select({
427
- memoryItemId: memoryItemSources.memoryItemId,
428
- messageId: memoryItemSources.messageId,
429
- })
430
- .from(memoryItemSources)
431
- .where(
432
- inArray(
433
- memoryItemSources.memoryItemId,
434
- items.map((item) => item.id),
435
- ),
436
- )
437
- .all();
438
- const hasAnySource = new Set<string>();
439
- const hasNonExcludedSource = new Set<string>();
440
- for (const source of sources) {
441
- hasAnySource.add(source.memoryItemId);
442
- if (!excludedSet.has(source.messageId)) {
443
- hasNonExcludedSource.add(source.memoryItemId);
444
- }
445
- }
446
- items = items.filter(
447
- (item) => !hasAnySource.has(item.id) || hasNonExcludedSource.has(item.id),
448
- );
449
- }
450
- if (items.length === 0) return [];
451
-
452
- return items.map((item) => ({
453
- key: `item:${item.id}`,
454
- type: "item" as CandidateType,
455
- id: item.id,
456
- source: opts.source,
457
- text: `${item.subject}: ${item.statement}`,
458
- kind: item.kind,
459
- confidence: item.confidence,
460
- importance: item.importance ?? 0.5,
461
- createdAt: item.lastSeenAt,
462
- lexical: 0,
463
- semantic: 0,
464
- recency: computeRecencyScore(item.lastSeenAt),
465
- finalScore: 0,
466
- }));
467
- }
468
-
469
- /**
470
- * Multi-step typed traversal: each step expands the frontier through
471
- * edges matching the step's relation/entity type filters.
472
- * Returns entity IDs reachable after all steps are applied in sequence.
473
- */
474
- export function collectTypedNeighbors(
475
- seedEntityIds: string[],
476
- steps: TraversalStep[],
477
- opts?: { maxResultsPerStep?: number; maxEdgesPerStep?: number },
478
- ): string[] {
479
- if (seedEntityIds.length === 0 || steps.length === 0) return [];
480
-
481
- const maxResults = opts?.maxResultsPerStep ?? 20;
482
- const maxEdges = opts?.maxEdgesPerStep ?? 40;
483
-
484
- let currentSeeds = seedEntityIds;
485
-
486
- for (const step of steps) {
487
- if (currentSeeds.length === 0) break;
488
-
489
- const result = findNeighborEntities(currentSeeds, {
490
- maxEdges,
491
- maxNeighborEntities: maxResults,
492
- maxDepth: 1,
493
- relationTypes: step.relationTypes,
494
- entityTypes: step.entityTypes,
495
- directed: true,
496
- });
497
-
498
- currentSeeds = result.neighborEntityIds;
499
- }
500
-
501
- return currentSeeds;
502
- }
503
-
504
- /**
505
- * Find entities reachable from ALL given seeds via their respective
506
- * typed traversal steps, then return the intersection.
507
- */
508
- export function intersectReachable(
509
- queries: Array<{
510
- seedEntityIds: string[];
511
- steps: TraversalStep[];
512
- }>,
513
- opts?: { maxResultsPerStep?: number; maxEdgesPerStep?: number },
514
- ): string[] {
515
- if (queries.length === 0) return [];
516
-
517
- const resultSets: Set<string>[] = [];
518
- for (const query of queries) {
519
- const result = collectTypedNeighbors(
520
- query.seedEntityIds,
521
- query.steps,
522
- opts,
523
- );
524
- resultSets.push(new Set(result));
525
- }
526
-
527
- if (resultSets.length === 0) return [];
528
-
529
- // Intersect all sets: keep only entities present in ALL sets
530
- const intersection = [...resultSets[0]].filter((id) =>
531
- resultSets.every((set) => set.has(id)),
532
- );
533
-
534
- return intersection;
535
- }
@@ -1,70 +0,0 @@
1
- import { describe, expect, test } from "bun:test";
2
-
3
- import { buildFTSQuery, expandQueryForFTS } from "./query-expansion.js";
4
-
5
- describe("expandQueryForFTS", () => {
6
- test("extracts meaningful keywords from conversational input", () => {
7
- const result = expandQueryForFTS(
8
- "what did we discuss about the API design?",
9
- );
10
- expect(result).toEqual(["discuss", "API", "design"]);
11
- });
12
-
13
- test("extracts all tokens from technical input (no stop words)", () => {
14
- const result = expandQueryForFTS("React component lifecycle hooks");
15
- expect(result).toEqual(["React", "component", "lifecycle", "hooks"]);
16
- });
17
-
18
- test("returns single keyword as-is", () => {
19
- const result = expandQueryForFTS("authentication");
20
- expect(result).toEqual(["authentication"]);
21
- });
22
-
23
- test("returns empty array for empty input", () => {
24
- expect(expandQueryForFTS("")).toEqual([]);
25
- });
26
-
27
- test("returns empty array for whitespace-only input", () => {
28
- expect(expandQueryForFTS(" ")).toEqual([]);
29
- });
30
-
31
- test("returns empty array for punctuation-only input", () => {
32
- expect(expandQueryForFTS("???")).toEqual([]);
33
- });
34
-
35
- test("returns original tokens when all are stop words", () => {
36
- const result = expandQueryForFTS("what is the");
37
- expect(result).toEqual(["what", "is", "the"]);
38
- });
39
-
40
- test("splits punctuation-delimited words into separate tokens", () => {
41
- const result = expandQueryForFTS("error-handling config.yaml");
42
- expect(result).toEqual(["error", "handling", "config", "yaml"]);
43
- });
44
-
45
- test("normalizes contractions instead of splitting on apostrophes", () => {
46
- const result = expandQueryForFTS("can't we discuss what's happening?");
47
- expect(result).toEqual(["cant", "discuss", "whats", "happening"]);
48
- });
49
- });
50
-
51
- describe("buildFTSQuery", () => {
52
- test("joins multiple keywords with OR", () => {
53
- const result = buildFTSQuery(["API", "design"]);
54
- expect(result).toBe('"API" OR "design"');
55
- });
56
-
57
- test("wraps single keyword in quotes", () => {
58
- const result = buildFTSQuery(["auth"]);
59
- expect(result).toBe('"auth"');
60
- });
61
-
62
- test("strips double-quote characters from keywords", () => {
63
- const result = buildFTSQuery(['say "hello"', "world"]);
64
- expect(result).toBe('"say hello" OR "world"');
65
- });
66
-
67
- test("returns undefined for empty keywords", () => {
68
- expect(buildFTSQuery([])).toBeUndefined();
69
- });
70
- });