@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,1117 +0,0 @@
1
- import { mkdtempSync, rmSync } from "node:fs";
2
- import { tmpdir } from "node:os";
3
- import { join } from "node:path";
4
- import {
5
- afterAll,
6
- beforeAll,
7
- beforeEach,
8
- describe,
9
- expect,
10
- mock,
11
- test,
12
- } from "bun:test";
13
-
14
- const testDir = mkdtempSync(join(tmpdir(), "entity-search-test-"));
15
-
16
- mock.module("../util/platform.js", () => ({
17
- getDataDir: () => testDir,
18
- isMacOS: () => process.platform === "darwin",
19
- isLinux: () => process.platform === "linux",
20
- isWindows: () => process.platform === "win32",
21
- getPidPath: () => join(testDir, "test.pid"),
22
- getDbPath: () => join(testDir, "test.db"),
23
- getLogPath: () => join(testDir, "test.log"),
24
- ensureDataDir: () => {},
25
- }));
26
-
27
- mock.module("../util/logger.js", () => ({
28
- getLogger: () =>
29
- new Proxy({} as Record<string, unknown>, {
30
- get: () => () => {},
31
- }),
32
- }));
33
-
34
- import { Database } from "bun:sqlite";
35
-
36
- import { getDb, initializeDb, resetDb } from "../memory/db.js";
37
- import {
38
- upsertEntity,
39
- upsertEntityRelation,
40
- } from "../memory/entity-extractor.js";
41
- import { memoryItemEntities, memoryItems } from "../memory/schema.js";
42
- import {
43
- collectTypedNeighbors,
44
- findMatchedEntities,
45
- findNeighborEntities,
46
- getEntityLinkedItemCandidates,
47
- intersectReachable,
48
- } from "../memory/search/entity.js";
49
-
50
- function getRawDb(): Database {
51
- return (getDb() as unknown as { $client: Database }).$client;
52
- }
53
-
54
- function insertMemoryItem(
55
- id: string,
56
- opts?: { scopeId?: string; status?: string; invalidAt?: number | null },
57
- ) {
58
- const db = getDb();
59
- const now = Date.now();
60
- db.insert(memoryItems)
61
- .values({
62
- id,
63
- kind: "fact",
64
- subject: `Subject ${id}`,
65
- statement: `Statement for ${id}`,
66
- confidence: 0.9,
67
- importance: 0.5,
68
- status: opts?.status ?? "active",
69
- invalidAt: opts?.invalidAt ?? null,
70
- scopeId: opts?.scopeId ?? "default",
71
- fingerprint: `fp-${id}`,
72
- firstSeenAt: now,
73
- lastSeenAt: now,
74
- accessCount: 0,
75
- lastUsedAt: null,
76
- verificationState: "assistant_inferred",
77
- })
78
- .run();
79
- }
80
-
81
- function linkItemToEntity(memoryItemId: string, entityId: string) {
82
- const db = getDb();
83
- db.insert(memoryItemEntities).values({ memoryItemId, entityId }).run();
84
- }
85
-
86
- function insertMemoryItemSource(memoryItemId: string, messageId: string) {
87
- // Bypass foreign key checks since we don't need actual message rows for these tests
88
- const raw = getRawDb();
89
- raw.run("PRAGMA foreign_keys = OFF");
90
- raw.run(
91
- `INSERT INTO memory_item_sources (memory_item_id, message_id, evidence, created_at)
92
- VALUES (?, ?, NULL, ?)`,
93
- [memoryItemId, messageId, Date.now()],
94
- );
95
- raw.run("PRAGMA foreign_keys = ON");
96
- }
97
-
98
- describe("entity search", () => {
99
- beforeAll(() => {
100
- initializeDb();
101
- });
102
-
103
- beforeEach(() => {
104
- const db = getDb();
105
- db.run("DELETE FROM memory_item_sources");
106
- db.run("DELETE FROM memory_item_entities");
107
- db.run("DELETE FROM memory_entity_relations");
108
- db.run("DELETE FROM memory_entities");
109
- db.run("DELETE FROM memory_items");
110
- db.run("DELETE FROM memory_checkpoints");
111
- });
112
-
113
- afterAll(() => {
114
- resetDb();
115
- try {
116
- rmSync(testDir, { recursive: true, force: true });
117
- } catch {
118
- // best effort cleanup
119
- }
120
- });
121
-
122
- // ── findNeighborEntities ───────────────────────────────────────────
123
-
124
- describe("findNeighborEntities", () => {
125
- test("returns empty for empty seed list", () => {
126
- const result = findNeighborEntities([], {
127
- maxEdges: 10,
128
- maxNeighborEntities: 10,
129
- maxDepth: 3,
130
- });
131
- expect(result.neighborEntityIds).toEqual([]);
132
- expect(result.traversedEdgeCount).toBe(0);
133
- });
134
-
135
- test("returns empty when no edges exist", () => {
136
- const entityId = upsertEntity({
137
- name: "Lonely",
138
- type: "concept",
139
- aliases: [],
140
- });
141
- const result = findNeighborEntities([entityId], {
142
- maxEdges: 10,
143
- maxNeighborEntities: 10,
144
- maxDepth: 3,
145
- });
146
- expect(result.neighborEntityIds).toEqual([]);
147
- expect(result.traversedEdgeCount).toBe(0);
148
- });
149
-
150
- test("single-hop: seed A has edge to B returns [B]", () => {
151
- const a = upsertEntity({ name: "Alpha", type: "project", aliases: [] });
152
- const b = upsertEntity({ name: "Beta", type: "tool", aliases: [] });
153
-
154
- upsertEntityRelation({
155
- sourceEntityId: a,
156
- targetEntityId: b,
157
- relation: "uses",
158
- evidence: "Alpha uses Beta",
159
- });
160
-
161
- const result = findNeighborEntities([a], {
162
- maxEdges: 10,
163
- maxNeighborEntities: 10,
164
- maxDepth: 3,
165
- });
166
- expect(result.neighborEntityIds).toContain(b);
167
- expect(result.neighborEntityIds).toHaveLength(1);
168
- expect(result.traversedEdgeCount).toBeGreaterThan(0);
169
- });
170
-
171
- test("multi-hop: A->B->C with maxDepth=2 returns [B, C]", () => {
172
- const a = upsertEntity({ name: "NodeA", type: "concept", aliases: [] });
173
- const b = upsertEntity({ name: "NodeB", type: "concept", aliases: [] });
174
- const c = upsertEntity({ name: "NodeC", type: "concept", aliases: [] });
175
-
176
- upsertEntityRelation({
177
- sourceEntityId: a,
178
- targetEntityId: b,
179
- relation: "related_to",
180
- evidence: null,
181
- });
182
- upsertEntityRelation({
183
- sourceEntityId: b,
184
- targetEntityId: c,
185
- relation: "related_to",
186
- evidence: null,
187
- });
188
-
189
- const result = findNeighborEntities([a], {
190
- maxEdges: 20,
191
- maxNeighborEntities: 10,
192
- maxDepth: 2,
193
- });
194
- expect(result.neighborEntityIds).toContain(b);
195
- expect(result.neighborEntityIds).toContain(c);
196
- expect(result.neighborEntityIds).toHaveLength(2);
197
- });
198
-
199
- test("cycle detection: A->B->A returns [B], does not loop", () => {
200
- const a = upsertEntity({ name: "CycleA", type: "person", aliases: [] });
201
- const b = upsertEntity({ name: "CycleB", type: "person", aliases: [] });
202
-
203
- upsertEntityRelation({
204
- sourceEntityId: a,
205
- targetEntityId: b,
206
- relation: "collaborates_with",
207
- evidence: null,
208
- });
209
- upsertEntityRelation({
210
- sourceEntityId: b,
211
- targetEntityId: a,
212
- relation: "collaborates_with",
213
- evidence: null,
214
- });
215
-
216
- const result = findNeighborEntities([a], {
217
- maxEdges: 20,
218
- maxNeighborEntities: 10,
219
- maxDepth: 5,
220
- });
221
- expect(result.neighborEntityIds).toEqual([b]);
222
- });
223
-
224
- test("maxDepth=1 stops after first hop", () => {
225
- const a = upsertEntity({ name: "DepthA", type: "concept", aliases: [] });
226
- const b = upsertEntity({ name: "DepthB", type: "concept", aliases: [] });
227
- const c = upsertEntity({ name: "DepthC", type: "concept", aliases: [] });
228
-
229
- upsertEntityRelation({
230
- sourceEntityId: a,
231
- targetEntityId: b,
232
- relation: "depends_on",
233
- evidence: null,
234
- });
235
- upsertEntityRelation({
236
- sourceEntityId: b,
237
- targetEntityId: c,
238
- relation: "depends_on",
239
- evidence: null,
240
- });
241
-
242
- const result = findNeighborEntities([a], {
243
- maxEdges: 20,
244
- maxNeighborEntities: 10,
245
- maxDepth: 1,
246
- });
247
- expect(result.neighborEntityIds).toContain(b);
248
- expect(result.neighborEntityIds).not.toContain(c);
249
- expect(result.neighborEntityIds).toHaveLength(1);
250
- });
251
-
252
- test("maxEdges budget exhaustion stops traversal", () => {
253
- const a = upsertEntity({ name: "BudgetA", type: "concept", aliases: [] });
254
- const b = upsertEntity({ name: "BudgetB", type: "concept", aliases: [] });
255
- const c = upsertEntity({ name: "BudgetC", type: "concept", aliases: [] });
256
- const d = upsertEntity({ name: "BudgetD", type: "concept", aliases: [] });
257
-
258
- upsertEntityRelation({
259
- sourceEntityId: a,
260
- targetEntityId: b,
261
- relation: "related_to",
262
- evidence: null,
263
- });
264
- upsertEntityRelation({
265
- sourceEntityId: a,
266
- targetEntityId: c,
267
- relation: "related_to",
268
- evidence: null,
269
- });
270
- upsertEntityRelation({
271
- sourceEntityId: b,
272
- targetEntityId: d,
273
- relation: "related_to",
274
- evidence: null,
275
- });
276
-
277
- // Allow only 1 edge total, so BFS can't explore much
278
- const result = findNeighborEntities([a], {
279
- maxEdges: 1,
280
- maxNeighborEntities: 10,
281
- maxDepth: 3,
282
- });
283
- expect(result.traversedEdgeCount).toBeLessThanOrEqual(1);
284
- });
285
-
286
- test("maxNeighborEntities cap limits result size", () => {
287
- const seed = upsertEntity({
288
- name: "HubNode",
289
- type: "concept",
290
- aliases: [],
291
- });
292
- const neighbors: string[] = [];
293
- for (let i = 0; i < 5; i++) {
294
- const n = upsertEntity({
295
- name: `Spoke${i}`,
296
- type: "concept",
297
- aliases: [],
298
- });
299
- neighbors.push(n);
300
- upsertEntityRelation({
301
- sourceEntityId: seed,
302
- targetEntityId: n,
303
- relation: "related_to",
304
- evidence: null,
305
- });
306
- }
307
-
308
- const result = findNeighborEntities([seed], {
309
- maxEdges: 20,
310
- maxNeighborEntities: 2,
311
- maxDepth: 3,
312
- });
313
- expect(result.neighborEntityIds).toHaveLength(2);
314
- });
315
-
316
- test("bidirectional: edge from X->A discovers X from seed [A]", () => {
317
- const a = upsertEntity({ name: "TargetNode", type: "tool", aliases: [] });
318
- const x = upsertEntity({ name: "SourceNode", type: "tool", aliases: [] });
319
-
320
- upsertEntityRelation({
321
- sourceEntityId: x,
322
- targetEntityId: a,
323
- relation: "uses",
324
- evidence: null,
325
- });
326
-
327
- const result = findNeighborEntities([a], {
328
- maxEdges: 10,
329
- maxNeighborEntities: 10,
330
- maxDepth: 3,
331
- });
332
- expect(result.neighborEntityIds).toContain(x);
333
- });
334
-
335
- test("multiple seeds: [A, B] discovers neighbors of both", () => {
336
- const a = upsertEntity({ name: "SeedA", type: "project", aliases: [] });
337
- const b = upsertEntity({ name: "SeedB", type: "project", aliases: [] });
338
- const na = upsertEntity({
339
- name: "NeighborOfA",
340
- type: "tool",
341
- aliases: [],
342
- });
343
- const nb = upsertEntity({
344
- name: "NeighborOfB",
345
- type: "tool",
346
- aliases: [],
347
- });
348
-
349
- upsertEntityRelation({
350
- sourceEntityId: a,
351
- targetEntityId: na,
352
- relation: "uses",
353
- evidence: null,
354
- });
355
- upsertEntityRelation({
356
- sourceEntityId: b,
357
- targetEntityId: nb,
358
- relation: "uses",
359
- evidence: null,
360
- });
361
-
362
- const result = findNeighborEntities([a, b], {
363
- maxEdges: 20,
364
- maxNeighborEntities: 10,
365
- maxDepth: 3,
366
- });
367
- expect(result.neighborEntityIds).toContain(na);
368
- expect(result.neighborEntityIds).toContain(nb);
369
- });
370
-
371
- test("relationTypes filter: only follows specified edge types", () => {
372
- const idA = upsertEntity({
373
- name: "PersonAlpha",
374
- type: "person",
375
- aliases: [],
376
- });
377
- const idB = upsertEntity({ name: "ToolBeta", type: "tool", aliases: [] });
378
- const idC = upsertEntity({
379
- name: "ProjectGamma",
380
- type: "project",
381
- aliases: [],
382
- });
383
-
384
- upsertEntityRelation({
385
- sourceEntityId: idA,
386
- targetEntityId: idB,
387
- relation: "uses",
388
- });
389
- upsertEntityRelation({
390
- sourceEntityId: idA,
391
- targetEntityId: idC,
392
- relation: "works_on",
393
- });
394
-
395
- const result = findNeighborEntities([idA], {
396
- maxEdges: 10,
397
- maxNeighborEntities: 10,
398
- maxDepth: 1,
399
- relationTypes: ["uses"],
400
- });
401
-
402
- expect(result.neighborEntityIds).toContain(idB);
403
- expect(result.neighborEntityIds).not.toContain(idC);
404
- });
405
-
406
- test("relationTypes filter: omitting filter follows all edge types", () => {
407
- const idA = upsertEntity({
408
- name: "PersonDelta",
409
- type: "person",
410
- aliases: [],
411
- });
412
- const idB = upsertEntity({
413
- name: "ToolEpsilon",
414
- type: "tool",
415
- aliases: [],
416
- });
417
- const idC = upsertEntity({
418
- name: "ProjectZeta",
419
- type: "project",
420
- aliases: [],
421
- });
422
-
423
- upsertEntityRelation({
424
- sourceEntityId: idA,
425
- targetEntityId: idB,
426
- relation: "uses",
427
- });
428
- upsertEntityRelation({
429
- sourceEntityId: idA,
430
- targetEntityId: idC,
431
- relation: "works_on",
432
- });
433
-
434
- const result = findNeighborEntities([idA], {
435
- maxEdges: 10,
436
- maxNeighborEntities: 10,
437
- maxDepth: 1,
438
- });
439
-
440
- expect(result.neighborEntityIds).toContain(idB);
441
- expect(result.neighborEntityIds).toContain(idC);
442
- });
443
-
444
- test("entityTypes filter: only returns entities of specified types", () => {
445
- const idPerson = upsertEntity({
446
- name: "PersonEta",
447
- type: "person",
448
- aliases: [],
449
- });
450
- const idProject = upsertEntity({
451
- name: "ProjectTheta",
452
- type: "project",
453
- aliases: [],
454
- });
455
- const idTool = upsertEntity({
456
- name: "ToolIota",
457
- type: "tool",
458
- aliases: [],
459
- });
460
-
461
- upsertEntityRelation({
462
- sourceEntityId: idPerson,
463
- targetEntityId: idProject,
464
- relation: "works_on",
465
- });
466
- upsertEntityRelation({
467
- sourceEntityId: idPerson,
468
- targetEntityId: idTool,
469
- relation: "uses",
470
- });
471
-
472
- const result = findNeighborEntities([idPerson], {
473
- maxEdges: 10,
474
- maxNeighborEntities: 10,
475
- maxDepth: 1,
476
- entityTypes: ["project"],
477
- });
478
-
479
- expect(result.neighborEntityIds).toContain(idProject);
480
- expect(result.neighborEntityIds).not.toContain(idTool);
481
- });
482
-
483
- test("entityTypes filter: omitting filter returns all entity types", () => {
484
- const idPerson = upsertEntity({
485
- name: "PersonKappa",
486
- type: "person",
487
- aliases: [],
488
- });
489
- const idProject = upsertEntity({
490
- name: "ProjectLambda",
491
- type: "project",
492
- aliases: [],
493
- });
494
- const idTool = upsertEntity({
495
- name: "ToolMu",
496
- type: "tool",
497
- aliases: [],
498
- });
499
-
500
- upsertEntityRelation({
501
- sourceEntityId: idPerson,
502
- targetEntityId: idProject,
503
- relation: "works_on",
504
- });
505
- upsertEntityRelation({
506
- sourceEntityId: idPerson,
507
- targetEntityId: idTool,
508
- relation: "uses",
509
- });
510
-
511
- const result = findNeighborEntities([idPerson], {
512
- maxEdges: 10,
513
- maxNeighborEntities: 10,
514
- maxDepth: 1,
515
- });
516
-
517
- expect(result.neighborEntityIds).toContain(idProject);
518
- expect(result.neighborEntityIds).toContain(idTool);
519
- });
520
-
521
- test("neighborDepths tracks BFS depth for each neighbor", () => {
522
- // A -> B -> C (chain)
523
- const idA = upsertEntity({
524
- name: "DepthAlpha",
525
- type: "person",
526
- aliases: [],
527
- });
528
- const idB = upsertEntity({
529
- name: "DepthBeta",
530
- type: "tool",
531
- aliases: [],
532
- });
533
- const idC = upsertEntity({
534
- name: "DepthGamma",
535
- type: "project",
536
- aliases: [],
537
- });
538
-
539
- upsertEntityRelation({
540
- sourceEntityId: idA,
541
- targetEntityId: idB,
542
- relation: "uses",
543
- });
544
- upsertEntityRelation({
545
- sourceEntityId: idB,
546
- targetEntityId: idC,
547
- relation: "depends_on",
548
- });
549
-
550
- const result = findNeighborEntities([idA], {
551
- maxEdges: 10,
552
- maxNeighborEntities: 10,
553
- maxDepth: 2,
554
- });
555
-
556
- expect(result.neighborEntityIds).toContain(idB);
557
- expect(result.neighborEntityIds).toContain(idC);
558
- expect(result.neighborDepths.get(idB)).toBe(1);
559
- expect(result.neighborDepths.get(idC)).toBe(2);
560
- });
561
-
562
- test("neighborDepths is empty when no neighbors found", () => {
563
- const idA = upsertEntity({
564
- name: "DepthDelta",
565
- type: "person",
566
- aliases: [],
567
- });
568
- const result = findNeighborEntities([idA], {
569
- maxEdges: 10,
570
- maxNeighborEntities: 10,
571
- maxDepth: 1,
572
- });
573
- expect(result.neighborDepths.size).toBe(0);
574
- });
575
-
576
- test("deep chain: maxDepth caps traversal on a long linear chain", () => {
577
- // Build a linear chain of 20 entities: N0 -> N1 -> ... -> N19
578
- const chain: string[] = [];
579
- for (let i = 0; i < 20; i++) {
580
- chain.push(
581
- upsertEntity({ name: `DeepChain${i}`, type: "concept", aliases: [] }),
582
- );
583
- }
584
- for (let i = 0; i < chain.length - 1; i++) {
585
- upsertEntityRelation({
586
- sourceEntityId: chain[i],
587
- targetEntityId: chain[i + 1],
588
- relation: "related_to",
589
- evidence: null,
590
- });
591
- }
592
-
593
- const maxDepth = 3;
594
- const result = findNeighborEntities([chain[0]], {
595
- maxEdges: 200,
596
- maxNeighborEntities: 200,
597
- maxDepth,
598
- });
599
-
600
- // Should find exactly nodes at depth 1..3 (chain[1], chain[2], chain[3])
601
- expect(result.neighborEntityIds).toHaveLength(maxDepth);
602
- for (let d = 1; d <= maxDepth; d++) {
603
- expect(result.neighborEntityIds).toContain(chain[d]);
604
- expect(result.neighborDepths.get(chain[d])).toBe(d);
605
- }
606
- // Nodes beyond maxDepth should not be reached
607
- for (let i = maxDepth + 1; i < chain.length; i++) {
608
- expect(result.neighborEntityIds).not.toContain(chain[i]);
609
- }
610
- });
611
-
612
- test("large cycle: traversal terminates on a fully-connected ring", () => {
613
- // Build a ring: N0 -> N1 -> ... -> N9 -> N0
614
- const ringSize = 10;
615
- const ring: string[] = [];
616
- for (let i = 0; i < ringSize; i++) {
617
- ring.push(
618
- upsertEntity({ name: `Ring${i}`, type: "concept", aliases: [] }),
619
- );
620
- }
621
- for (let i = 0; i < ringSize; i++) {
622
- upsertEntityRelation({
623
- sourceEntityId: ring[i],
624
- targetEntityId: ring[(i + 1) % ringSize],
625
- relation: "related_to",
626
- evidence: null,
627
- });
628
- }
629
-
630
- // With maxDepth high enough to go around the ring multiple times if
631
- // cycle detection were broken, the visited set must prevent revisiting.
632
- const result = findNeighborEntities([ring[0]], {
633
- maxEdges: 500,
634
- maxNeighborEntities: 500,
635
- maxDepth: 20,
636
- });
637
-
638
- // Should discover exactly ringSize - 1 neighbors (all except the seed)
639
- expect(result.neighborEntityIds).toHaveLength(ringSize - 1);
640
- for (let i = 1; i < ringSize; i++) {
641
- expect(result.neighborEntityIds).toContain(ring[i]);
642
- }
643
- });
644
-
645
- test("dense cyclic graph: traversal terminates with multiple cycles and back-edges", () => {
646
- // Build a graph where every node connects to multiple others with back-edges
647
- const nodes: string[] = [];
648
- for (let i = 0; i < 8; i++) {
649
- nodes.push(
650
- upsertEntity({ name: `Dense${i}`, type: "concept", aliases: [] }),
651
- );
652
- }
653
- // Create a mesh: each node connects to the next 2 nodes (wrapping)
654
- for (let i = 0; i < nodes.length; i++) {
655
- for (let offset = 1; offset <= 2; offset++) {
656
- upsertEntityRelation({
657
- sourceEntityId: nodes[i],
658
- targetEntityId: nodes[(i + offset) % nodes.length],
659
- relation: "related_to",
660
- evidence: null,
661
- });
662
- }
663
- }
664
-
665
- const result = findNeighborEntities([nodes[0]], {
666
- maxEdges: 500,
667
- maxNeighborEntities: 500,
668
- maxDepth: 10,
669
- });
670
-
671
- // All non-seed nodes should be reachable, and traversal must terminate
672
- expect(result.neighborEntityIds).toHaveLength(nodes.length - 1);
673
- // No duplicate IDs in the result
674
- expect(new Set(result.neighborEntityIds).size).toBe(
675
- result.neighborEntityIds.length,
676
- );
677
- });
678
- });
679
-
680
- // ── findMatchedEntities ────────────────────────────────────────────
681
-
682
- describe("findMatchedEntities", () => {
683
- test("exact canonical name match", () => {
684
- const entityId = upsertEntity({
685
- name: "Qdrant",
686
- type: "tool",
687
- aliases: [],
688
- });
689
- const results = findMatchedEntities("Qdrant", 10);
690
- expect(results.length).toBeGreaterThanOrEqual(1);
691
- expect(results.some((r) => r.id === entityId)).toBe(true);
692
- });
693
-
694
- test("alias match", () => {
695
- const entityId = upsertEntity({
696
- name: "Visual Studio Code",
697
- type: "tool",
698
- aliases: ["vscode", "VS Code"],
699
- });
700
- const results = findMatchedEntities("vscode", 10);
701
- expect(results.length).toBeGreaterThanOrEqual(1);
702
- expect(results.some((r) => r.id === entityId)).toBe(true);
703
- });
704
-
705
- test("multi-word entity name match (full query)", () => {
706
- const entityId = upsertEntity({
707
- name: "Visual Studio Code",
708
- type: "tool",
709
- aliases: [],
710
- });
711
- const results = findMatchedEntities("Visual Studio Code", 10);
712
- expect(results.length).toBeGreaterThanOrEqual(1);
713
- expect(results.some((r) => r.id === entityId)).toBe(true);
714
- });
715
-
716
- test("tokens < 3 chars are ignored but full query still matches", () => {
717
- // "VS" has only 2 chars, so it is filtered as a token.
718
- // But the full query "VS" is still matched against entity names and aliases.
719
- const entityId = upsertEntity({ name: "VS", type: "tool", aliases: [] });
720
- const results = findMatchedEntities("VS", 10);
721
- expect(results.length).toBeGreaterThanOrEqual(1);
722
- expect(results.some((r) => r.id === entityId)).toBe(true);
723
- });
724
-
725
- test("returns empty for no matches", () => {
726
- upsertEntity({ name: "Existing", type: "concept", aliases: [] });
727
- const results = findMatchedEntities("NonExistentEntity", 10);
728
- expect(results).toEqual([]);
729
- });
730
-
731
- test("respects maxMatches limit", () => {
732
- // Insert entities directly via raw DB to avoid upsertEntity dedup logic.
733
- // All share the alias "gadget" so they all match the same query.
734
- const raw = getRawDb();
735
- const now = Date.now();
736
- for (let i = 0; i < 5; i++) {
737
- const id = crypto.randomUUID();
738
- raw.run(
739
- `INSERT INTO memory_entities (id, name, type, aliases, description, first_seen_at, last_seen_at, mention_count)
740
- VALUES (?, ?, 'concept', '["gadget"]', NULL, ?, ?, 1)`,
741
- [id, `Gadget${i}`, now, now],
742
- );
743
- }
744
-
745
- const results = findMatchedEntities("gadget", 2);
746
- expect(results.length).toBeLessThanOrEqual(2);
747
- });
748
- });
749
-
750
- // ── getEntityLinkedItemCandidates ──────────────────────────────────
751
-
752
- describe("getEntityLinkedItemCandidates", () => {
753
- test("returns items linked to given entity IDs", () => {
754
- const entityId = upsertEntity({
755
- name: "LinkedEntity",
756
- type: "project",
757
- aliases: [],
758
- });
759
- insertMemoryItem("item-linked-1");
760
- linkItemToEntity("item-linked-1", entityId);
761
-
762
- const candidates = getEntityLinkedItemCandidates([entityId], {
763
- source: "entity_direct",
764
- });
765
-
766
- expect(candidates.length).toBe(1);
767
- expect(candidates[0].id).toBe("item-linked-1");
768
- expect(candidates[0].source).toBe("entity_direct");
769
- expect(candidates[0].type).toBe("item");
770
- });
771
-
772
- test("excludes items from excluded message IDs", () => {
773
- const entityId = upsertEntity({
774
- name: "ExcludeEntity",
775
- type: "tool",
776
- aliases: [],
777
- });
778
-
779
- insertMemoryItem("item-excl-1");
780
- linkItemToEntity("item-excl-1", entityId);
781
- // Source the item from a message we will exclude
782
- insertMemoryItemSource("item-excl-1", "msg-to-exclude");
783
-
784
- insertMemoryItem("item-excl-2");
785
- linkItemToEntity("item-excl-2", entityId);
786
- // Source from a non-excluded message
787
- insertMemoryItemSource("item-excl-2", "msg-ok");
788
-
789
- const candidates = getEntityLinkedItemCandidates([entityId], {
790
- source: "entity_direct",
791
- excludedMessageIds: ["msg-to-exclude"],
792
- });
793
-
794
- expect(candidates.some((c) => c.id === "item-excl-1")).toBe(false);
795
- expect(candidates.some((c) => c.id === "item-excl-2")).toBe(true);
796
- });
797
-
798
- test("returns empty for entity IDs with no linked items", () => {
799
- const entityId = upsertEntity({
800
- name: "NoItems",
801
- type: "concept",
802
- aliases: [],
803
- });
804
-
805
- const candidates = getEntityLinkedItemCandidates([entityId], {
806
- source: "entity_direct",
807
- });
808
-
809
- expect(candidates).toEqual([]);
810
- });
811
- });
812
-
813
- // ── collectTypedNeighbors ────────────────────────────────────────────
814
-
815
- describe("collectTypedNeighbors", () => {
816
- test("multi-step: person -> projects -> tools", () => {
817
- const person = upsertEntity({
818
- name: "StepPerson1",
819
- type: "person",
820
- aliases: [],
821
- });
822
- const project1 = upsertEntity({
823
- name: "StepProject1",
824
- type: "project",
825
- aliases: [],
826
- });
827
- const project2 = upsertEntity({
828
- name: "StepProject2",
829
- type: "project",
830
- aliases: [],
831
- });
832
- const tool1 = upsertEntity({
833
- name: "StepTool1",
834
- type: "tool",
835
- aliases: [],
836
- });
837
- const tool2 = upsertEntity({
838
- name: "StepTool2",
839
- type: "tool",
840
- aliases: [],
841
- });
842
- const tool3 = upsertEntity({
843
- name: "StepTool3",
844
- type: "tool",
845
- aliases: [],
846
- });
847
-
848
- // person works_on project1 and project2
849
- upsertEntityRelation({
850
- sourceEntityId: person,
851
- targetEntityId: project1,
852
- relation: "works_on",
853
- });
854
- upsertEntityRelation({
855
- sourceEntityId: person,
856
- targetEntityId: project2,
857
- relation: "works_on",
858
- });
859
- // project1 uses tool1 and tool2
860
- upsertEntityRelation({
861
- sourceEntityId: project1,
862
- targetEntityId: tool1,
863
- relation: "uses",
864
- });
865
- upsertEntityRelation({
866
- sourceEntityId: project1,
867
- targetEntityId: tool2,
868
- relation: "uses",
869
- });
870
- // project2 uses tool2 and tool3
871
- upsertEntityRelation({
872
- sourceEntityId: project2,
873
- targetEntityId: tool2,
874
- relation: "uses",
875
- });
876
- upsertEntityRelation({
877
- sourceEntityId: project2,
878
- targetEntityId: tool3,
879
- relation: "uses",
880
- });
881
-
882
- const result = collectTypedNeighbors(
883
- [person],
884
- [
885
- { relationTypes: ["works_on"], entityTypes: ["project"] },
886
- { relationTypes: ["uses"], entityTypes: ["tool"] },
887
- ],
888
- );
889
-
890
- expect(result).toContain(tool1);
891
- expect(result).toContain(tool2);
892
- expect(result).toContain(tool3);
893
- // Should NOT include person or projects in final result
894
- expect(result).not.toContain(person);
895
- expect(result).not.toContain(project1);
896
- expect(result).not.toContain(project2);
897
- });
898
-
899
- test("returns empty for empty seeds", () => {
900
- const result = collectTypedNeighbors([], [{ relationTypes: ["uses"] }]);
901
- expect(result).toEqual([]);
902
- });
903
-
904
- test("returns empty for empty steps", () => {
905
- const person = upsertEntity({
906
- name: "StepPerson2",
907
- type: "person",
908
- aliases: [],
909
- });
910
- const result = collectTypedNeighbors([person], []);
911
- expect(result).toEqual([]);
912
- });
913
-
914
- test("single step equivalent to filtered BFS", () => {
915
- const person = upsertEntity({
916
- name: "StepPerson3",
917
- type: "person",
918
- aliases: [],
919
- });
920
- const tool = upsertEntity({
921
- name: "StepTool4",
922
- type: "tool",
923
- aliases: [],
924
- });
925
- const project = upsertEntity({
926
- name: "StepProject3",
927
- type: "project",
928
- aliases: [],
929
- });
930
-
931
- upsertEntityRelation({
932
- sourceEntityId: person,
933
- targetEntityId: tool,
934
- relation: "uses",
935
- });
936
- upsertEntityRelation({
937
- sourceEntityId: person,
938
- targetEntityId: project,
939
- relation: "works_on",
940
- });
941
-
942
- const result = collectTypedNeighbors(
943
- [person],
944
- [{ relationTypes: ["uses"], entityTypes: ["tool"] }],
945
- );
946
-
947
- expect(result).toContain(tool);
948
- expect(result).not.toContain(project);
949
- });
950
-
951
- test("chain breaks when intermediate step finds no matches", () => {
952
- const person = upsertEntity({
953
- name: "StepPerson4",
954
- type: "person",
955
- aliases: [],
956
- });
957
- // person has no edges
958
- const result = collectTypedNeighbors(
959
- [person],
960
- [
961
- { relationTypes: ["works_on"], entityTypes: ["project"] },
962
- { relationTypes: ["uses"], entityTypes: ["tool"] },
963
- ],
964
- );
965
-
966
- expect(result).toEqual([]);
967
- });
968
- });
969
-
970
- // ── intersectReachable ───────────────────────────────────────────────
971
-
972
- describe("intersectReachable", () => {
973
- test("finds shared projects between two people", () => {
974
- const alice = upsertEntity({
975
- name: "IntersectAlice",
976
- type: "person",
977
- aliases: [],
978
- });
979
- const bob = upsertEntity({
980
- name: "IntersectBob",
981
- type: "person",
982
- aliases: [],
983
- });
984
- const sharedProject = upsertEntity({
985
- name: "IntersectSharedProj",
986
- type: "project",
987
- aliases: [],
988
- });
989
- const aliceOnly = upsertEntity({
990
- name: "IntersectAliceProj",
991
- type: "project",
992
- aliases: [],
993
- });
994
- const bobOnly = upsertEntity({
995
- name: "IntersectBobProj",
996
- type: "project",
997
- aliases: [],
998
- });
999
-
1000
- upsertEntityRelation({
1001
- sourceEntityId: alice,
1002
- targetEntityId: sharedProject,
1003
- relation: "works_on",
1004
- });
1005
- upsertEntityRelation({
1006
- sourceEntityId: alice,
1007
- targetEntityId: aliceOnly,
1008
- relation: "works_on",
1009
- });
1010
- upsertEntityRelation({
1011
- sourceEntityId: bob,
1012
- targetEntityId: sharedProject,
1013
- relation: "works_on",
1014
- });
1015
- upsertEntityRelation({
1016
- sourceEntityId: bob,
1017
- targetEntityId: bobOnly,
1018
- relation: "works_on",
1019
- });
1020
-
1021
- const result = intersectReachable([
1022
- {
1023
- seedEntityIds: [alice],
1024
- steps: [{ relationTypes: ["works_on"], entityTypes: ["project"] }],
1025
- },
1026
- {
1027
- seedEntityIds: [bob],
1028
- steps: [{ relationTypes: ["works_on"], entityTypes: ["project"] }],
1029
- },
1030
- ]);
1031
-
1032
- expect(result).toContain(sharedProject);
1033
- expect(result).not.toContain(aliceOnly);
1034
- expect(result).not.toContain(bobOnly);
1035
- });
1036
-
1037
- test("returns empty when no overlap", () => {
1038
- const alice = upsertEntity({
1039
- name: "IntersectAlice2",
1040
- type: "person",
1041
- aliases: [],
1042
- });
1043
- const bob = upsertEntity({
1044
- name: "IntersectBob2",
1045
- type: "person",
1046
- aliases: [],
1047
- });
1048
- const projA = upsertEntity({
1049
- name: "IntersectProjA",
1050
- type: "project",
1051
- aliases: [],
1052
- });
1053
- const projB = upsertEntity({
1054
- name: "IntersectProjB",
1055
- type: "project",
1056
- aliases: [],
1057
- });
1058
-
1059
- upsertEntityRelation({
1060
- sourceEntityId: alice,
1061
- targetEntityId: projA,
1062
- relation: "works_on",
1063
- });
1064
- upsertEntityRelation({
1065
- sourceEntityId: bob,
1066
- targetEntityId: projB,
1067
- relation: "works_on",
1068
- });
1069
-
1070
- const result = intersectReachable([
1071
- {
1072
- seedEntityIds: [alice],
1073
- steps: [{ relationTypes: ["works_on"], entityTypes: ["project"] }],
1074
- },
1075
- {
1076
- seedEntityIds: [bob],
1077
- steps: [{ relationTypes: ["works_on"], entityTypes: ["project"] }],
1078
- },
1079
- ]);
1080
-
1081
- expect(result).toEqual([]);
1082
- });
1083
-
1084
- test("single query is equivalent to collectTypedNeighbors", () => {
1085
- const person = upsertEntity({
1086
- name: "IntersectSingle",
1087
- type: "person",
1088
- aliases: [],
1089
- });
1090
- const tool = upsertEntity({
1091
- name: "IntersectTool",
1092
- type: "tool",
1093
- aliases: [],
1094
- });
1095
-
1096
- upsertEntityRelation({
1097
- sourceEntityId: person,
1098
- targetEntityId: tool,
1099
- relation: "uses",
1100
- });
1101
-
1102
- const result = intersectReachable([
1103
- {
1104
- seedEntityIds: [person],
1105
- steps: [{ relationTypes: ["uses"], entityTypes: ["tool"] }],
1106
- },
1107
- ]);
1108
-
1109
- expect(result).toContain(tool);
1110
- });
1111
-
1112
- test("returns empty for empty queries array", () => {
1113
- const result = intersectReachable([]);
1114
- expect(result).toEqual([]);
1115
- });
1116
- });
1117
- });