claude-code-swarm 0.3.3 → 0.3.5

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 (273) hide show
  1. package/.claude-plugin/marketplace.json +1 -1
  2. package/.claude-plugin/plugin.json +22 -1
  3. package/.claude-plugin/run-agent-inbox-mcp.sh +76 -0
  4. package/.claude-plugin/run-minimem-mcp.sh +98 -0
  5. package/.claude-plugin/run-opentasks-mcp.sh +65 -0
  6. package/CLAUDE.md +200 -36
  7. package/README.md +65 -0
  8. package/e2e/helpers/cleanup.mjs +17 -3
  9. package/e2e/helpers/map-mock-server.mjs +201 -25
  10. package/e2e/helpers/sidecar.mjs +222 -0
  11. package/e2e/helpers/workspace.mjs +2 -1
  12. package/e2e/tier5-sidecar-inbox.test.mjs +900 -0
  13. package/e2e/tier6-inbox-mcp.test.mjs +173 -0
  14. package/e2e/tier6-live-agent.test.mjs +759 -0
  15. package/e2e/vitest.config.e2e.mjs +1 -1
  16. package/hooks/hooks.json +15 -8
  17. package/package.json +13 -1
  18. package/references/agent-inbox/CLAUDE.md +151 -0
  19. package/references/agent-inbox/README.md +238 -0
  20. package/references/agent-inbox/docs/CLAUDE-CODE-SWARM-PROPOSAL.md +137 -0
  21. package/references/agent-inbox/docs/DESIGN.md +1156 -0
  22. package/references/agent-inbox/hooks/inbox-hook.mjs +119 -0
  23. package/references/agent-inbox/hooks/register-hook.mjs +69 -0
  24. package/references/agent-inbox/package-lock.json +3347 -0
  25. package/references/agent-inbox/package.json +58 -0
  26. package/references/agent-inbox/rules/agent-inbox.md +78 -0
  27. package/references/agent-inbox/src/federation/address.ts +61 -0
  28. package/references/agent-inbox/src/federation/connection-manager.ts +573 -0
  29. package/references/agent-inbox/src/federation/delivery-queue.ts +222 -0
  30. package/references/agent-inbox/src/federation/index.ts +6 -0
  31. package/references/agent-inbox/src/federation/routing-engine.ts +188 -0
  32. package/references/agent-inbox/src/federation/trust.ts +71 -0
  33. package/references/agent-inbox/src/index.ts +390 -0
  34. package/references/agent-inbox/src/ipc/ipc-server.ts +207 -0
  35. package/references/agent-inbox/src/jsonrpc/mail-server.ts +382 -0
  36. package/references/agent-inbox/src/map/map-client.ts +414 -0
  37. package/references/agent-inbox/src/mcp/mcp-server.ts +272 -0
  38. package/references/agent-inbox/src/mesh/delivery-bridge.ts +110 -0
  39. package/references/agent-inbox/src/mesh/mesh-connector.ts +41 -0
  40. package/references/agent-inbox/src/mesh/mesh-transport.ts +157 -0
  41. package/references/agent-inbox/src/mesh/type-mapper.ts +239 -0
  42. package/references/agent-inbox/src/push/notifier.ts +233 -0
  43. package/references/agent-inbox/src/registry/warm-registry.ts +255 -0
  44. package/references/agent-inbox/src/router/message-router.ts +175 -0
  45. package/references/agent-inbox/src/storage/interface.ts +48 -0
  46. package/references/agent-inbox/src/storage/memory.ts +145 -0
  47. package/references/agent-inbox/src/storage/sqlite.ts +671 -0
  48. package/references/agent-inbox/src/traceability/traceability.ts +183 -0
  49. package/references/agent-inbox/src/types.ts +303 -0
  50. package/references/agent-inbox/test/federation/address.test.ts +101 -0
  51. package/references/agent-inbox/test/federation/connection-manager.test.ts +546 -0
  52. package/references/agent-inbox/test/federation/delivery-queue.test.ts +159 -0
  53. package/references/agent-inbox/test/federation/integration.test.ts +857 -0
  54. package/references/agent-inbox/test/federation/routing-engine.test.ts +117 -0
  55. package/references/agent-inbox/test/federation/sdk-integration.test.ts +744 -0
  56. package/references/agent-inbox/test/federation/trust.test.ts +89 -0
  57. package/references/agent-inbox/test/ipc-jsonrpc.test.ts +113 -0
  58. package/references/agent-inbox/test/ipc-server.test.ts +197 -0
  59. package/references/agent-inbox/test/mail-server.test.ts +285 -0
  60. package/references/agent-inbox/test/map-client.test.ts +408 -0
  61. package/references/agent-inbox/test/mesh/delivery-bridge.test.ts +178 -0
  62. package/references/agent-inbox/test/mesh/e2e-mesh.test.ts +527 -0
  63. package/references/agent-inbox/test/mesh/e2e-real-meshpeer.test.ts +629 -0
  64. package/references/agent-inbox/test/mesh/federation-mesh.test.ts +269 -0
  65. package/references/agent-inbox/test/mesh/mesh-connector.test.ts +66 -0
  66. package/references/agent-inbox/test/mesh/mesh-transport.test.ts +191 -0
  67. package/references/agent-inbox/test/mesh/meshpeer-integration.test.ts +442 -0
  68. package/references/agent-inbox/test/mesh/mock-mesh.ts +125 -0
  69. package/references/agent-inbox/test/mesh/mock-meshpeer.ts +266 -0
  70. package/references/agent-inbox/test/mesh/type-mapper.test.ts +226 -0
  71. package/references/agent-inbox/test/message-router.test.ts +184 -0
  72. package/references/agent-inbox/test/push-notifier.test.ts +139 -0
  73. package/references/agent-inbox/test/registry/warm-registry.test.ts +171 -0
  74. package/references/agent-inbox/test/sqlite-prefix.test.ts +192 -0
  75. package/references/agent-inbox/test/sqlite-storage.test.ts +243 -0
  76. package/references/agent-inbox/test/storage.test.ts +196 -0
  77. package/references/agent-inbox/test/traceability.test.ts +123 -0
  78. package/references/agent-inbox/test/wake.test.ts +330 -0
  79. package/references/agent-inbox/tsconfig.json +20 -0
  80. package/references/agent-inbox/tsup.config.ts +10 -0
  81. package/references/agent-inbox/vitest.config.ts +8 -0
  82. package/references/minimem/.claude/settings.json +7 -0
  83. package/references/minimem/.sudocode/issues.jsonl +18 -0
  84. package/references/minimem/.sudocode/specs.jsonl +1 -0
  85. package/references/minimem/CLAUDE.md +329 -0
  86. package/references/minimem/README.md +565 -0
  87. package/references/minimem/claude-plugin/.claude-plugin/plugin.json +10 -0
  88. package/references/minimem/claude-plugin/.mcp.json +7 -0
  89. package/references/minimem/claude-plugin/README.md +158 -0
  90. package/references/minimem/claude-plugin/commands/recall.md +47 -0
  91. package/references/minimem/claude-plugin/commands/remember.md +41 -0
  92. package/references/minimem/claude-plugin/hooks/__tests__/hooks.test.ts +272 -0
  93. package/references/minimem/claude-plugin/hooks/hooks.json +27 -0
  94. package/references/minimem/claude-plugin/hooks/session-end.sh +86 -0
  95. package/references/minimem/claude-plugin/hooks/session-start.sh +85 -0
  96. package/references/minimem/claude-plugin/skills/memory/SKILL.md +108 -0
  97. package/references/minimem/media/banner.png +0 -0
  98. package/references/minimem/package-lock.json +5373 -0
  99. package/references/minimem/package.json +76 -0
  100. package/references/minimem/scripts/postbuild.js +49 -0
  101. package/references/minimem/src/__tests__/edge-cases.test.ts +371 -0
  102. package/references/minimem/src/__tests__/errors.test.ts +265 -0
  103. package/references/minimem/src/__tests__/helpers.ts +199 -0
  104. package/references/minimem/src/__tests__/internal.test.ts +407 -0
  105. package/references/minimem/src/__tests__/knowledge-frontmatter.test.ts +148 -0
  106. package/references/minimem/src/__tests__/knowledge.test.ts +148 -0
  107. package/references/minimem/src/__tests__/minimem.integration.test.ts +1127 -0
  108. package/references/minimem/src/__tests__/session.test.ts +190 -0
  109. package/references/minimem/src/cli/__tests__/commands.test.ts +760 -0
  110. package/references/minimem/src/cli/__tests__/contained-layout.test.ts +286 -0
  111. package/references/minimem/src/cli/commands/__tests__/conflicts.test.ts +141 -0
  112. package/references/minimem/src/cli/commands/append.ts +76 -0
  113. package/references/minimem/src/cli/commands/config.ts +262 -0
  114. package/references/minimem/src/cli/commands/conflicts.ts +415 -0
  115. package/references/minimem/src/cli/commands/daemon.ts +169 -0
  116. package/references/minimem/src/cli/commands/index.ts +12 -0
  117. package/references/minimem/src/cli/commands/init.ts +166 -0
  118. package/references/minimem/src/cli/commands/mcp.ts +221 -0
  119. package/references/minimem/src/cli/commands/push-pull.ts +213 -0
  120. package/references/minimem/src/cli/commands/search.ts +223 -0
  121. package/references/minimem/src/cli/commands/status.ts +84 -0
  122. package/references/minimem/src/cli/commands/store.ts +189 -0
  123. package/references/minimem/src/cli/commands/sync-init.ts +290 -0
  124. package/references/minimem/src/cli/commands/sync.ts +70 -0
  125. package/references/minimem/src/cli/commands/upsert.ts +197 -0
  126. package/references/minimem/src/cli/config.ts +611 -0
  127. package/references/minimem/src/cli/index.ts +299 -0
  128. package/references/minimem/src/cli/shared.ts +189 -0
  129. package/references/minimem/src/cli/sync/__tests__/central.test.ts +152 -0
  130. package/references/minimem/src/cli/sync/__tests__/conflicts.test.ts +209 -0
  131. package/references/minimem/src/cli/sync/__tests__/daemon.test.ts +118 -0
  132. package/references/minimem/src/cli/sync/__tests__/detection.test.ts +207 -0
  133. package/references/minimem/src/cli/sync/__tests__/integration.test.ts +476 -0
  134. package/references/minimem/src/cli/sync/__tests__/registry.test.ts +363 -0
  135. package/references/minimem/src/cli/sync/__tests__/state.test.ts +255 -0
  136. package/references/minimem/src/cli/sync/__tests__/validation.test.ts +193 -0
  137. package/references/minimem/src/cli/sync/__tests__/watcher.test.ts +178 -0
  138. package/references/minimem/src/cli/sync/central.ts +292 -0
  139. package/references/minimem/src/cli/sync/conflicts.ts +205 -0
  140. package/references/minimem/src/cli/sync/daemon.ts +407 -0
  141. package/references/minimem/src/cli/sync/detection.ts +138 -0
  142. package/references/minimem/src/cli/sync/index.ts +107 -0
  143. package/references/minimem/src/cli/sync/operations.ts +373 -0
  144. package/references/minimem/src/cli/sync/registry.ts +279 -0
  145. package/references/minimem/src/cli/sync/state.ts +358 -0
  146. package/references/minimem/src/cli/sync/validation.ts +206 -0
  147. package/references/minimem/src/cli/sync/watcher.ts +237 -0
  148. package/references/minimem/src/cli/version.ts +34 -0
  149. package/references/minimem/src/core/index.ts +9 -0
  150. package/references/minimem/src/core/indexer.ts +628 -0
  151. package/references/minimem/src/core/searcher.ts +221 -0
  152. package/references/minimem/src/db/schema.ts +183 -0
  153. package/references/minimem/src/db/sqlite-vec.ts +24 -0
  154. package/references/minimem/src/embeddings/__tests__/embeddings.test.ts +431 -0
  155. package/references/minimem/src/embeddings/batch-gemini.ts +392 -0
  156. package/references/minimem/src/embeddings/batch-openai.ts +409 -0
  157. package/references/minimem/src/embeddings/embeddings.ts +434 -0
  158. package/references/minimem/src/index.ts +132 -0
  159. package/references/minimem/src/internal.ts +299 -0
  160. package/references/minimem/src/minimem.ts +1291 -0
  161. package/references/minimem/src/search/__tests__/hybrid.test.ts +247 -0
  162. package/references/minimem/src/search/graph.ts +234 -0
  163. package/references/minimem/src/search/hybrid.ts +151 -0
  164. package/references/minimem/src/search/search.ts +256 -0
  165. package/references/minimem/src/server/__tests__/mcp.test.ts +347 -0
  166. package/references/minimem/src/server/__tests__/tools.test.ts +364 -0
  167. package/references/minimem/src/server/mcp.ts +326 -0
  168. package/references/minimem/src/server/tools.ts +720 -0
  169. package/references/minimem/src/session.ts +460 -0
  170. package/references/minimem/src/store/__tests__/manifest.test.ts +177 -0
  171. package/references/minimem/src/store/__tests__/materialize.test.ts +52 -0
  172. package/references/minimem/src/store/__tests__/store-graph.test.ts +228 -0
  173. package/references/minimem/src/store/index.ts +27 -0
  174. package/references/minimem/src/store/manifest.ts +203 -0
  175. package/references/minimem/src/store/materialize.ts +185 -0
  176. package/references/minimem/src/store/store-graph.ts +252 -0
  177. package/references/minimem/tsconfig.json +19 -0
  178. package/references/minimem/tsup.config.ts +26 -0
  179. package/references/minimem/vitest.config.ts +29 -0
  180. package/references/openteams/src/cli/generate.ts +23 -1
  181. package/references/openteams/src/generators/agent-prompt-generator.test.ts +94 -0
  182. package/references/openteams/src/generators/agent-prompt-generator.ts +42 -13
  183. package/references/openteams/src/generators/package-generator.ts +9 -1
  184. package/references/openteams/src/generators/skill-generator.test.ts +28 -0
  185. package/references/openteams/src/generators/skill-generator.ts +10 -4
  186. package/references/skill-tree/.claude/settings.json +6 -0
  187. package/references/skill-tree/.sudocode/issues.jsonl +19 -0
  188. package/references/skill-tree/.sudocode/specs.jsonl +3 -0
  189. package/references/skill-tree/CLAUDE.md +132 -0
  190. package/references/skill-tree/README.md +396 -0
  191. package/references/skill-tree/docs/GAPS_v1.md +221 -0
  192. package/references/skill-tree/docs/INTEGRATION_PLAN.md +467 -0
  193. package/references/skill-tree/docs/TODOS.md +91 -0
  194. package/references/skill-tree/docs/anthropic_skill_guide.md +1364 -0
  195. package/references/skill-tree/docs/design/federated-skill-trees.md +524 -0
  196. package/references/skill-tree/docs/design/multi-agent-sync.md +759 -0
  197. package/references/skill-tree/docs/scraper/BRAINSTORM.md +583 -0
  198. package/references/skill-tree/docs/scraper/POC_PLAN.md +420 -0
  199. package/references/skill-tree/docs/scraper/README.md +170 -0
  200. package/references/skill-tree/examples/basic-usage.ts +157 -0
  201. package/references/skill-tree/package-lock.json +1852 -0
  202. package/references/skill-tree/package.json +66 -0
  203. package/references/skill-tree/plan.md +78 -0
  204. package/references/skill-tree/scraper/README.md +123 -0
  205. package/references/skill-tree/scraper/docs/DESIGN.md +683 -0
  206. package/references/skill-tree/scraper/docs/PLAN.md +336 -0
  207. package/references/skill-tree/scraper/drizzle.config.ts +10 -0
  208. package/references/skill-tree/scraper/package-lock.json +6329 -0
  209. package/references/skill-tree/scraper/package.json +68 -0
  210. package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-description.md +7 -0
  211. package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-name.md +7 -0
  212. package/references/skill-tree/scraper/test/fixtures/minimal-skill/SKILL.md +27 -0
  213. package/references/skill-tree/scraper/test/fixtures/skill-json/SKILL.json +21 -0
  214. package/references/skill-tree/scraper/test/fixtures/skill-with-meta/SKILL.md +54 -0
  215. package/references/skill-tree/scraper/test/fixtures/skill-with-meta/_meta.json +24 -0
  216. package/references/skill-tree/scraper/test/fixtures/valid-skill/SKILL.md +93 -0
  217. package/references/skill-tree/scraper/test/fixtures/valid-skill/_meta.json +22 -0
  218. package/references/skill-tree/scraper/tsup.config.ts +14 -0
  219. package/references/skill-tree/scraper/vitest.config.ts +17 -0
  220. package/references/skill-tree/scripts/convert-to-vitest.ts +166 -0
  221. package/references/skill-tree/skills/skill-writer/SKILL.md +339 -0
  222. package/references/skill-tree/skills/skill-writer/references/examples.md +326 -0
  223. package/references/skill-tree/skills/skill-writer/references/patterns.md +210 -0
  224. package/references/skill-tree/skills/skill-writer/references/quality-checklist.md +123 -0
  225. package/references/skill-tree/test/run-all.ts +106 -0
  226. package/references/skill-tree/test/utils.ts +128 -0
  227. package/references/skill-tree/vitest.config.ts +16 -0
  228. package/references/swarmkit/src/commands/init/phases/configure.ts +0 -22
  229. package/references/swarmkit/src/commands/init/phases/global-setup.ts +5 -3
  230. package/references/swarmkit/src/commands/init/wizard.ts +2 -2
  231. package/references/swarmkit/src/packages/setup.test.ts +53 -7
  232. package/references/swarmkit/src/packages/setup.ts +37 -1
  233. package/scripts/bootstrap.mjs +26 -1
  234. package/scripts/generate-agents.mjs +5 -1
  235. package/scripts/map-hook.mjs +97 -64
  236. package/scripts/map-sidecar.mjs +179 -25
  237. package/scripts/team-loader.mjs +12 -41
  238. package/skills/swarm/SKILL.md +89 -25
  239. package/src/__tests__/agent-generator.test.mjs +6 -13
  240. package/src/__tests__/bootstrap.test.mjs +124 -1
  241. package/src/__tests__/config.test.mjs +200 -27
  242. package/src/__tests__/e2e-live-map.test.mjs +536 -0
  243. package/src/__tests__/e2e-mesh-sidecar.test.mjs +570 -0
  244. package/src/__tests__/e2e-native-task-hooks.test.mjs +376 -0
  245. package/src/__tests__/e2e-sidecar-bridge.test.mjs +477 -0
  246. package/src/__tests__/helpers.mjs +13 -0
  247. package/src/__tests__/inbox.test.mjs +22 -89
  248. package/src/__tests__/index.test.mjs +35 -9
  249. package/src/__tests__/integration.test.mjs +513 -0
  250. package/src/__tests__/map-events.test.mjs +514 -150
  251. package/src/__tests__/mesh-connection.test.mjs +308 -0
  252. package/src/__tests__/opentasks-client.test.mjs +517 -0
  253. package/src/__tests__/paths.test.mjs +185 -41
  254. package/src/__tests__/sidecar-client.test.mjs +35 -0
  255. package/src/__tests__/sidecar-server.test.mjs +124 -0
  256. package/src/__tests__/skilltree-client.test.mjs +80 -0
  257. package/src/agent-generator.mjs +104 -33
  258. package/src/bootstrap.mjs +150 -10
  259. package/src/config.mjs +81 -17
  260. package/src/context-output.mjs +58 -8
  261. package/src/inbox.mjs +9 -54
  262. package/src/index.mjs +39 -8
  263. package/src/map-connection.mjs +4 -3
  264. package/src/map-events.mjs +350 -80
  265. package/src/mesh-connection.mjs +148 -0
  266. package/src/opentasks-client.mjs +269 -0
  267. package/src/paths.mjs +182 -27
  268. package/src/sessionlog.mjs +14 -9
  269. package/src/sidecar-client.mjs +81 -27
  270. package/src/sidecar-server.mjs +175 -16
  271. package/src/skilltree-client.mjs +173 -0
  272. package/src/template.mjs +68 -4
  273. package/vitest.config.mjs +1 -0
@@ -0,0 +1,247 @@
1
+ import { describe, expect, it } from "vitest";
2
+
3
+ import { bm25RankToScore, buildFtsQuery, mergeHybridResults } from "../hybrid.js";
4
+
5
+ describe("buildFtsQuery", () => {
6
+ it("tokenizes and AND-joins", () => {
7
+ expect(buildFtsQuery("hello world")).toBe('"hello" AND "world"');
8
+ expect(buildFtsQuery("FOO_bar baz-1")).toBe('"FOO_bar" AND "baz" AND "1"');
9
+ });
10
+
11
+ it("returns null for empty/whitespace input", () => {
12
+ expect(buildFtsQuery(" ")).toBeNull();
13
+ expect(buildFtsQuery("")).toBeNull();
14
+ });
15
+
16
+ it("strips quotes from tokens", () => {
17
+ expect(buildFtsQuery('hello "world')).toBe('"hello" AND "world"');
18
+ });
19
+
20
+ it("handles single token", () => {
21
+ expect(buildFtsQuery("hello")).toBe('"hello"');
22
+ });
23
+
24
+ it("handles special characters", () => {
25
+ expect(buildFtsQuery("hello@world.com")).toBe('"hello" AND "world" AND "com"');
26
+ });
27
+ });
28
+
29
+ describe("bm25RankToScore", () => {
30
+ it("returns 1 for rank 0", () => {
31
+ expect(bm25RankToScore(0)).toBeCloseTo(1);
32
+ });
33
+
34
+ it("returns 0.5 for rank 1 or -1", () => {
35
+ // Both positive and negative 1 give same result (absolute value)
36
+ expect(bm25RankToScore(1)).toBeCloseTo(0.5);
37
+ expect(bm25RankToScore(-1)).toBeCloseTo(0.5);
38
+ });
39
+
40
+ it("is monotonically decreasing with absolute value", () => {
41
+ // Higher magnitude = lower score
42
+ expect(bm25RankToScore(10)).toBeLessThan(bm25RankToScore(1));
43
+ expect(bm25RankToScore(100)).toBeLessThan(bm25RankToScore(10));
44
+ });
45
+
46
+ it("treats negative ranks same as positive (uses absolute value)", () => {
47
+ // FTS5 BM25 ranks are negative, so -10 should give same score as 10
48
+ expect(bm25RankToScore(-10)).toBeCloseTo(bm25RankToScore(10));
49
+ expect(bm25RankToScore(-100)).toBeCloseTo(bm25RankToScore(100));
50
+ // -100 → abs = 100 → score = 1/101 ≈ 0.0099
51
+ expect(bm25RankToScore(-100)).toBeCloseTo(1 / 101);
52
+ });
53
+
54
+ it("handles infinity by returning 0", () => {
55
+ expect(bm25RankToScore(Infinity)).toBe(0);
56
+ expect(bm25RankToScore(-Infinity)).toBe(0);
57
+ });
58
+
59
+ it("handles NaN by returning 0", () => {
60
+ expect(bm25RankToScore(NaN)).toBe(0);
61
+ });
62
+ });
63
+
64
+ describe("mergeHybridResults", () => {
65
+ it("unions by id and combines weighted scores", () => {
66
+ const merged = mergeHybridResults({
67
+ vectorWeight: 0.7,
68
+ textWeight: 0.3,
69
+ vector: [
70
+ {
71
+ id: "a",
72
+ path: "memory/a.md",
73
+ startLine: 1,
74
+ endLine: 2,
75
+ source: "memory",
76
+ snippet: "vec-a",
77
+ vectorScore: 0.9,
78
+ },
79
+ ],
80
+ keyword: [
81
+ {
82
+ id: "b",
83
+ path: "memory/b.md",
84
+ startLine: 3,
85
+ endLine: 4,
86
+ source: "memory",
87
+ snippet: "kw-b",
88
+ textScore: 1.0,
89
+ },
90
+ ],
91
+ });
92
+
93
+ expect(merged).toHaveLength(2);
94
+ const a = merged.find((r) => r.path === "memory/a.md");
95
+ const b = merged.find((r) => r.path === "memory/b.md");
96
+ expect(a?.score).toBeCloseTo(0.7 * 0.9);
97
+ expect(b?.score).toBeCloseTo(0.3 * 1.0);
98
+ });
99
+
100
+ it("prefers keyword snippet when ids overlap", () => {
101
+ const merged = mergeHybridResults({
102
+ vectorWeight: 0.5,
103
+ textWeight: 0.5,
104
+ vector: [
105
+ {
106
+ id: "a",
107
+ path: "memory/a.md",
108
+ startLine: 1,
109
+ endLine: 2,
110
+ source: "memory",
111
+ snippet: "vec-a",
112
+ vectorScore: 0.2,
113
+ },
114
+ ],
115
+ keyword: [
116
+ {
117
+ id: "a",
118
+ path: "memory/a.md",
119
+ startLine: 1,
120
+ endLine: 2,
121
+ source: "memory",
122
+ snippet: "kw-a",
123
+ textScore: 1.0,
124
+ },
125
+ ],
126
+ });
127
+
128
+ expect(merged).toHaveLength(1);
129
+ expect(merged[0]?.snippet).toBe("kw-a");
130
+ expect(merged[0]?.score).toBeCloseTo(0.5 * 0.2 + 0.5 * 1.0);
131
+ });
132
+
133
+ it("sorts by score descending", () => {
134
+ const merged = mergeHybridResults({
135
+ vectorWeight: 1.0,
136
+ textWeight: 0.0,
137
+ vector: [
138
+ {
139
+ id: "low",
140
+ path: "memory/low.md",
141
+ startLine: 1,
142
+ endLine: 1,
143
+ source: "memory",
144
+ snippet: "low",
145
+ vectorScore: 0.3,
146
+ },
147
+ {
148
+ id: "high",
149
+ path: "memory/high.md",
150
+ startLine: 1,
151
+ endLine: 1,
152
+ source: "memory",
153
+ snippet: "high",
154
+ vectorScore: 0.9,
155
+ },
156
+ ],
157
+ keyword: [],
158
+ });
159
+
160
+ expect(merged[0]?.path).toBe("memory/high.md");
161
+ expect(merged[1]?.path).toBe("memory/low.md");
162
+ });
163
+
164
+ it("handles empty inputs", () => {
165
+ const merged = mergeHybridResults({
166
+ vectorWeight: 0.7,
167
+ textWeight: 0.3,
168
+ vector: [],
169
+ keyword: [],
170
+ });
171
+
172
+ expect(merged).toHaveLength(0);
173
+ });
174
+
175
+ it("handles vector-only results with normalized weights", () => {
176
+ const merged = mergeHybridResults({
177
+ vectorWeight: 0.7,
178
+ textWeight: 0.3,
179
+ vector: [
180
+ {
181
+ id: "a",
182
+ path: "memory/a.md",
183
+ startLine: 1,
184
+ endLine: 2,
185
+ source: "memory",
186
+ snippet: "vec-a",
187
+ vectorScore: 0.8,
188
+ },
189
+ ],
190
+ keyword: [],
191
+ });
192
+
193
+ expect(merged).toHaveLength(1);
194
+ // When keyword side is empty, vector weight normalizes to 1.0
195
+ expect(merged[0]?.score).toBeCloseTo(0.8);
196
+ });
197
+
198
+ it("handles keyword-only results with normalized weights", () => {
199
+ const merged = mergeHybridResults({
200
+ vectorWeight: 0.7,
201
+ textWeight: 0.3,
202
+ vector: [],
203
+ keyword: [
204
+ {
205
+ id: "a",
206
+ path: "memory/a.md",
207
+ startLine: 1,
208
+ endLine: 2,
209
+ source: "memory",
210
+ snippet: "kw-a",
211
+ textScore: 0.9,
212
+ },
213
+ ],
214
+ });
215
+
216
+ expect(merged).toHaveLength(1);
217
+ // When vector side is empty, text weight normalizes to 1.0
218
+ expect(merged[0]?.score).toBeCloseTo(0.9);
219
+ });
220
+
221
+ it("preserves all metadata fields", () => {
222
+ const merged = mergeHybridResults({
223
+ vectorWeight: 1.0,
224
+ textWeight: 0.0,
225
+ vector: [
226
+ {
227
+ id: "test",
228
+ path: "memory/test.md",
229
+ startLine: 5,
230
+ endLine: 10,
231
+ source: "sessions",
232
+ snippet: "test snippet",
233
+ vectorScore: 0.5,
234
+ },
235
+ ],
236
+ keyword: [],
237
+ });
238
+
239
+ expect(merged[0]).toMatchObject({
240
+ path: "memory/test.md",
241
+ startLine: 5,
242
+ endLine: 10,
243
+ source: "sessions",
244
+ snippet: "test snippet",
245
+ });
246
+ });
247
+ });
@@ -0,0 +1,234 @@
1
+ import type { DatabaseSync } from "node:sqlite";
2
+
3
+ /**
4
+ * A knowledge graph link between two nodes
5
+ */
6
+ export type GraphLink = {
7
+ fromId: string;
8
+ toId: string;
9
+ relation: string;
10
+ layer: string | null;
11
+ weight: number;
12
+ sourcePath: string | null;
13
+ };
14
+
15
+ /**
16
+ * A neighbor node discovered during graph traversal
17
+ */
18
+ export type GraphNeighbor = {
19
+ id: string;
20
+ depth: number;
21
+ link: GraphLink;
22
+ };
23
+
24
+ /**
25
+ * Get all outgoing links from a node.
26
+ */
27
+ export function getLinksFrom(
28
+ db: DatabaseSync,
29
+ fromId: string,
30
+ opts?: { relation?: string; layer?: string },
31
+ ): GraphLink[] {
32
+ let sql = `SELECT from_id, to_id, relation, layer, weight, source_path FROM knowledge_links WHERE from_id = ?`;
33
+ const params: (string | number)[] = [fromId];
34
+
35
+ if (opts?.relation) {
36
+ sql += ` AND relation = ?`;
37
+ params.push(opts.relation);
38
+ }
39
+ if (opts?.layer) {
40
+ sql += ` AND layer = ?`;
41
+ params.push(opts.layer);
42
+ }
43
+
44
+ const rows = db.prepare(sql).all(...params) as Array<{
45
+ from_id: string;
46
+ to_id: string;
47
+ relation: string;
48
+ layer: string | null;
49
+ weight: number;
50
+ source_path: string | null;
51
+ }>;
52
+
53
+ return rows.map(toGraphLink);
54
+ }
55
+
56
+ /**
57
+ * Get all incoming links to a node.
58
+ */
59
+ export function getLinksTo(
60
+ db: DatabaseSync,
61
+ toId: string,
62
+ opts?: { relation?: string; layer?: string },
63
+ ): GraphLink[] {
64
+ let sql = `SELECT from_id, to_id, relation, layer, weight, source_path FROM knowledge_links WHERE to_id = ?`;
65
+ const params: (string | number)[] = [toId];
66
+
67
+ if (opts?.relation) {
68
+ sql += ` AND relation = ?`;
69
+ params.push(opts.relation);
70
+ }
71
+ if (opts?.layer) {
72
+ sql += ` AND layer = ?`;
73
+ params.push(opts.layer);
74
+ }
75
+
76
+ const rows = db.prepare(sql).all(...params) as Array<{
77
+ from_id: string;
78
+ to_id: string;
79
+ relation: string;
80
+ layer: string | null;
81
+ weight: number;
82
+ source_path: string | null;
83
+ }>;
84
+
85
+ return rows.map(toGraphLink);
86
+ }
87
+
88
+ /**
89
+ * BFS traversal to find neighbors up to a given depth.
90
+ *
91
+ * @param db - Database handle
92
+ * @param startId - The starting node ID
93
+ * @param depth - Maximum traversal depth (default: 1)
94
+ * @param opts - Optional filters for relation and layer
95
+ * @returns Array of neighbor nodes with their depth and connecting link
96
+ */
97
+ export function getNeighbors(
98
+ db: DatabaseSync,
99
+ startId: string,
100
+ depth: number = 1,
101
+ opts?: { relation?: string; layer?: string },
102
+ ): GraphNeighbor[] {
103
+ const visited = new Set<string>([startId]);
104
+ const result: GraphNeighbor[] = [];
105
+ let frontier = [startId];
106
+
107
+ for (let d = 1; d <= depth; d++) {
108
+ const nextFrontier: string[] = [];
109
+
110
+ for (const nodeId of frontier) {
111
+ // Follow outgoing links
112
+ const outgoing = getLinksFrom(db, nodeId, opts);
113
+ for (const link of outgoing) {
114
+ if (!visited.has(link.toId)) {
115
+ visited.add(link.toId);
116
+ nextFrontier.push(link.toId);
117
+ result.push({ id: link.toId, depth: d, link });
118
+ }
119
+ }
120
+
121
+ // Follow incoming links
122
+ const incoming = getLinksTo(db, nodeId, opts);
123
+ for (const link of incoming) {
124
+ if (!visited.has(link.fromId)) {
125
+ visited.add(link.fromId);
126
+ nextFrontier.push(link.fromId);
127
+ result.push({ id: link.fromId, depth: d, link });
128
+ }
129
+ }
130
+ }
131
+
132
+ frontier = nextFrontier;
133
+ if (frontier.length === 0) break;
134
+ }
135
+
136
+ return result;
137
+ }
138
+
139
+ /**
140
+ * BFS shortest path between two nodes, max depth 3.
141
+ *
142
+ * @returns Array of links forming the path, or empty if no path found.
143
+ */
144
+ export function getPathBetween(
145
+ db: DatabaseSync,
146
+ fromId: string,
147
+ toId: string,
148
+ maxDepth: number = 3,
149
+ ): GraphLink[] {
150
+ if (fromId === toId) return [];
151
+
152
+ // BFS with parent tracking
153
+ const visited = new Set<string>([fromId]);
154
+ // Maps each visited node to the link that discovered it
155
+ const parentLink = new Map<string, GraphLink>();
156
+ let frontier = [fromId];
157
+
158
+ for (let d = 0; d < maxDepth; d++) {
159
+ const nextFrontier: string[] = [];
160
+
161
+ for (const nodeId of frontier) {
162
+ // Follow outgoing links
163
+ const outgoing = getLinksFrom(db, nodeId);
164
+ for (const link of outgoing) {
165
+ if (!visited.has(link.toId)) {
166
+ visited.add(link.toId);
167
+ parentLink.set(link.toId, link);
168
+ if (link.toId === toId) {
169
+ return reconstructPath(parentLink, fromId, toId);
170
+ }
171
+ nextFrontier.push(link.toId);
172
+ }
173
+ }
174
+
175
+ // Follow incoming links
176
+ const incoming = getLinksTo(db, nodeId);
177
+ for (const link of incoming) {
178
+ if (!visited.has(link.fromId)) {
179
+ visited.add(link.fromId);
180
+ parentLink.set(link.fromId, link);
181
+ if (link.fromId === toId) {
182
+ return reconstructPath(parentLink, fromId, toId);
183
+ }
184
+ nextFrontier.push(link.fromId);
185
+ }
186
+ }
187
+ }
188
+
189
+ frontier = nextFrontier;
190
+ if (frontier.length === 0) break;
191
+ }
192
+
193
+ return []; // No path found
194
+ }
195
+
196
+ /**
197
+ * Reconstruct the path from BFS parent map.
198
+ */
199
+ function reconstructPath(
200
+ parentLink: Map<string, GraphLink>,
201
+ fromId: string,
202
+ toId: string,
203
+ ): GraphLink[] {
204
+ const path: GraphLink[] = [];
205
+ let current = toId;
206
+
207
+ while (current !== fromId) {
208
+ const link = parentLink.get(current);
209
+ if (!link) break;
210
+ path.unshift(link);
211
+ // Determine which side led to `current`
212
+ current = link.toId === current ? link.fromId : link.toId;
213
+ }
214
+
215
+ return path;
216
+ }
217
+
218
+ function toGraphLink(row: {
219
+ from_id: string;
220
+ to_id: string;
221
+ relation: string;
222
+ layer: string | null;
223
+ weight: number;
224
+ source_path: string | null;
225
+ }): GraphLink {
226
+ return {
227
+ fromId: row.from_id,
228
+ toId: row.to_id,
229
+ relation: row.relation,
230
+ layer: row.layer,
231
+ weight: row.weight,
232
+ sourcePath: row.source_path,
233
+ };
234
+ }
@@ -0,0 +1,151 @@
1
+ export type HybridSource = string;
2
+
3
+ export type HybridVectorResult = {
4
+ id: string;
5
+ path: string;
6
+ startLine: number;
7
+ endLine: number;
8
+ source: HybridSource;
9
+ snippet: string;
10
+ vectorScore: number;
11
+ };
12
+
13
+ export type HybridKeywordResult = {
14
+ id: string;
15
+ path: string;
16
+ startLine: number;
17
+ endLine: number;
18
+ source: HybridSource;
19
+ snippet: string;
20
+ textScore: number;
21
+ };
22
+
23
+ export function buildFtsQuery(raw: string): string | null {
24
+ const tokens =
25
+ raw
26
+ .match(/[A-Za-z0-9_]+/g)
27
+ ?.map((t) => t.trim())
28
+ .filter(Boolean) ?? [];
29
+ if (tokens.length === 0) return null;
30
+ const quoted = tokens.map((t) => `"${t.replaceAll('"', "")}"`);
31
+ return quoted.join(" AND ");
32
+ }
33
+
34
+ /**
35
+ * Convert BM25 rank from SQLite FTS5 to a 0-1 score.
36
+ *
37
+ * FTS5 BM25 ranks are NEGATIVE numbers where more negative = better match.
38
+ * A rank of 0 means no match, -10 is better than -1.
39
+ *
40
+ * We use absolute value to convert to positive, then normalize to 0-1
41
+ * using the formula: score = 1 / (1 + absRank)
42
+ *
43
+ * Examples:
44
+ * - rank 0 (no match) -> score 1.0
45
+ * - rank -1 (weak match) -> score 0.5
46
+ * - rank -10 (strong match) -> score ~0.09
47
+ *
48
+ * Note: Higher absolute rank magnitude = better match = higher score after conversion.
49
+ */
50
+ export function bm25RankToScore(rank: number): number {
51
+ // Handle non-finite values (NaN, Infinity)
52
+ if (!Number.isFinite(rank)) {
53
+ return 0;
54
+ }
55
+
56
+ // BM25 ranks from FTS5 are negative (more negative = better match)
57
+ // Use absolute value to get the magnitude
58
+ const absRank = Math.abs(rank);
59
+
60
+ // Convert to 0-1 score where higher magnitude = higher score
61
+ // Using 1/(1+x) gives us a nice 0-1 range that decreases smoothly
62
+ return 1 / (1 + absRank);
63
+ }
64
+
65
+ export function mergeHybridResults(params: {
66
+ vector: HybridVectorResult[];
67
+ keyword: HybridKeywordResult[];
68
+ vectorWeight: number;
69
+ textWeight: number;
70
+ }): Array<{
71
+ path: string;
72
+ startLine: number;
73
+ endLine: number;
74
+ score: number;
75
+ snippet: string;
76
+ source: HybridSource;
77
+ }> {
78
+ const byId = new Map<
79
+ string,
80
+ {
81
+ id: string;
82
+ path: string;
83
+ startLine: number;
84
+ endLine: number;
85
+ source: HybridSource;
86
+ snippet: string;
87
+ vectorScore: number;
88
+ textScore: number;
89
+ }
90
+ >();
91
+
92
+ for (const r of params.vector) {
93
+ byId.set(r.id, {
94
+ id: r.id,
95
+ path: r.path,
96
+ startLine: r.startLine,
97
+ endLine: r.endLine,
98
+ source: r.source,
99
+ snippet: r.snippet,
100
+ vectorScore: r.vectorScore,
101
+ textScore: 0,
102
+ });
103
+ }
104
+
105
+ for (const r of params.keyword) {
106
+ const existing = byId.get(r.id);
107
+ if (existing) {
108
+ existing.textScore = r.textScore;
109
+ if (r.snippet && r.snippet.length > 0) existing.snippet = r.snippet;
110
+ } else {
111
+ byId.set(r.id, {
112
+ id: r.id,
113
+ path: r.path,
114
+ startLine: r.startLine,
115
+ endLine: r.endLine,
116
+ source: r.source,
117
+ snippet: r.snippet,
118
+ vectorScore: 0,
119
+ textScore: r.textScore,
120
+ });
121
+ }
122
+ }
123
+
124
+ // When one side of the hybrid search has no results, normalize weights
125
+ // so the available side scores at full strength. Without this, BM25-only
126
+ // results would be scaled to 0.3 * textScore which is too low to pass
127
+ // the default minScore threshold.
128
+ let vw = params.vectorWeight;
129
+ let tw = params.textWeight;
130
+ if (params.vector.length === 0 && params.keyword.length > 0) {
131
+ vw = 0;
132
+ tw = 1;
133
+ } else if (params.keyword.length === 0 && params.vector.length > 0) {
134
+ vw = 1;
135
+ tw = 0;
136
+ }
137
+
138
+ const merged = Array.from(byId.values()).map((entry) => {
139
+ const score = vw * entry.vectorScore + tw * entry.textScore;
140
+ return {
141
+ path: entry.path,
142
+ startLine: entry.startLine,
143
+ endLine: entry.endLine,
144
+ score,
145
+ snippet: entry.snippet,
146
+ source: entry.source,
147
+ };
148
+ });
149
+
150
+ return merged.sort((a, b) => b.score - a.score);
151
+ }