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,256 @@
1
+ import type { DatabaseSync } from "node:sqlite";
2
+
3
+ import { cosineSimilarity, parseEmbedding, truncateUtf16Safe, vectorToBlob } from "../internal.js";
4
+
5
+ export type SearchSource = string;
6
+
7
+ export type SearchRowResult = {
8
+ id: string;
9
+ path: string;
10
+ startLine: number;
11
+ endLine: number;
12
+ score: number;
13
+ snippet: string;
14
+ source: SearchSource;
15
+ };
16
+
17
+ /**
18
+ * Options for filtering search results by knowledge metadata
19
+ */
20
+ export type KnowledgeSearchOptions = {
21
+ /** Filter to chunks matching any of these domains */
22
+ domain?: string[];
23
+ /** Filter to chunks referencing any of these entities */
24
+ entities?: string[];
25
+ /** Minimum confidence threshold */
26
+ minConfidence?: number;
27
+ /** Filter to a specific knowledge type */
28
+ knowledgeType?: string;
29
+ };
30
+
31
+ /**
32
+ * Build SQL WHERE clause fragments for knowledge filters.
33
+ * Uses json_each() for array column filtering.
34
+ */
35
+ export function buildKnowledgeFilterSql(opts: KnowledgeSearchOptions): {
36
+ sql: string;
37
+ params: (string | number)[];
38
+ } {
39
+ const clauses: string[] = [];
40
+ const params: (string | number)[] = [];
41
+
42
+ if (opts.knowledgeType) {
43
+ clauses.push(` AND c.knowledge_type = ?`);
44
+ params.push(opts.knowledgeType);
45
+ }
46
+
47
+ if (opts.minConfidence !== undefined) {
48
+ clauses.push(` AND c.confidence >= ?`);
49
+ params.push(opts.minConfidence);
50
+ }
51
+
52
+ if (opts.domain && opts.domain.length > 0) {
53
+ // At least one of the provided domains must appear in the JSON array
54
+ const domainPlaceholders = opts.domain.map(() => "?").join(", ");
55
+ clauses.push(
56
+ ` AND EXISTS (SELECT 1 FROM json_each(c.domains) AS d WHERE d.value IN (${domainPlaceholders}))`,
57
+ );
58
+ params.push(...opts.domain);
59
+ }
60
+
61
+ if (opts.entities && opts.entities.length > 0) {
62
+ const entityPlaceholders = opts.entities.map(() => "?").join(", ");
63
+ clauses.push(
64
+ ` AND EXISTS (SELECT 1 FROM json_each(c.entities) AS e WHERE e.value IN (${entityPlaceholders}))`,
65
+ );
66
+ params.push(...opts.entities);
67
+ }
68
+
69
+ return { sql: clauses.join(""), params };
70
+ }
71
+
72
+ /**
73
+ * Perform a vector similarity search against indexed memory chunks.
74
+ *
75
+ * First attempts to use sqlite-vec for fast approximate nearest neighbor search.
76
+ * Falls back to brute-force cosine similarity over all chunks if the vector
77
+ * extension is unavailable.
78
+ *
79
+ * @returns Matching chunks sorted by descending similarity score (0-1 range).
80
+ */
81
+ export async function searchVector(params: {
82
+ db: DatabaseSync;
83
+ vectorTable: string;
84
+ providerModel: string;
85
+ queryVec: number[];
86
+ limit: number;
87
+ snippetMaxChars: number;
88
+ ensureVectorReady: (dimensions: number) => Promise<boolean>;
89
+ sourceFilterVec: { sql: string; params: SearchSource[] };
90
+ sourceFilterChunks: { sql: string; params: SearchSource[] };
91
+ }): Promise<SearchRowResult[]> {
92
+ if (params.queryVec.length === 0 || params.limit <= 0) return [];
93
+ if (await params.ensureVectorReady(params.queryVec.length)) {
94
+ const rows = params.db
95
+ .prepare(
96
+ `SELECT c.id, c.path, c.start_line, c.end_line, c.text,\n` +
97
+ ` c.source,\n` +
98
+ ` vec_distance_cosine(v.embedding, ?) AS dist\n` +
99
+ ` FROM ${params.vectorTable} v\n` +
100
+ ` JOIN chunks c ON c.id = v.id\n` +
101
+ ` WHERE c.model = ?${params.sourceFilterVec.sql}\n` +
102
+ ` ORDER BY dist ASC\n` +
103
+ ` LIMIT ?`,
104
+ )
105
+ .all(
106
+ vectorToBlob(params.queryVec),
107
+ params.providerModel,
108
+ ...params.sourceFilterVec.params,
109
+ params.limit,
110
+ ) as Array<{
111
+ id: string;
112
+ path: string;
113
+ start_line: number;
114
+ end_line: number;
115
+ text: string;
116
+ source: SearchSource;
117
+ dist: number;
118
+ }>;
119
+ return rows.map((row) => ({
120
+ id: row.id,
121
+ path: row.path,
122
+ startLine: row.start_line,
123
+ endLine: row.end_line,
124
+ score: 1 - row.dist,
125
+ snippet: truncateUtf16Safe(row.text, params.snippetMaxChars),
126
+ source: row.source,
127
+ }));
128
+ }
129
+
130
+ const candidates = listChunks({
131
+ db: params.db,
132
+ providerModel: params.providerModel,
133
+ sourceFilter: params.sourceFilterChunks,
134
+ });
135
+ const scored = candidates
136
+ .map((chunk) => ({
137
+ chunk,
138
+ score: cosineSimilarity(params.queryVec, chunk.embedding),
139
+ }))
140
+ .filter((entry) => Number.isFinite(entry.score));
141
+ return scored
142
+ .sort((a, b) => b.score - a.score)
143
+ .slice(0, params.limit)
144
+ .map((entry) => ({
145
+ id: entry.chunk.id,
146
+ path: entry.chunk.path,
147
+ startLine: entry.chunk.startLine,
148
+ endLine: entry.chunk.endLine,
149
+ score: entry.score,
150
+ snippet: truncateUtf16Safe(entry.chunk.text, params.snippetMaxChars),
151
+ source: entry.chunk.source,
152
+ }));
153
+ }
154
+
155
+ /**
156
+ * List all indexed chunks for a given embedding model and source filter.
157
+ * Used as a fallback when sqlite-vec is not available for vector search.
158
+ *
159
+ * @returns All matching chunks with their parsed embedding vectors.
160
+ */
161
+ export function listChunks(params: {
162
+ db: DatabaseSync;
163
+ providerModel: string;
164
+ sourceFilter: { sql: string; params: SearchSource[] };
165
+ }): Array<{
166
+ id: string;
167
+ path: string;
168
+ startLine: number;
169
+ endLine: number;
170
+ text: string;
171
+ embedding: number[];
172
+ source: SearchSource;
173
+ }> {
174
+ const rows = params.db
175
+ .prepare(
176
+ `SELECT id, path, start_line, end_line, text, embedding, source\n` +
177
+ ` FROM chunks\n` +
178
+ ` WHERE model = ?${params.sourceFilter.sql}`,
179
+ )
180
+ .all(params.providerModel, ...params.sourceFilter.params) as Array<{
181
+ id: string;
182
+ path: string;
183
+ start_line: number;
184
+ end_line: number;
185
+ text: string;
186
+ embedding: string;
187
+ source: SearchSource;
188
+ }>;
189
+
190
+ return rows.map((row) => ({
191
+ id: row.id,
192
+ path: row.path,
193
+ startLine: row.start_line,
194
+ endLine: row.end_line,
195
+ text: row.text,
196
+ embedding: parseEmbedding(row.embedding),
197
+ source: row.source,
198
+ }));
199
+ }
200
+
201
+ /**
202
+ * Perform a full-text keyword search using SQLite FTS5 with BM25 ranking.
203
+ *
204
+ * Tokenizes the query into quoted AND terms and runs them against the FTS index.
205
+ * Results are scored using BM25 rank converted to a 0-1 range.
206
+ *
207
+ * @returns Matching chunks sorted by BM25 relevance, with both score and textScore fields.
208
+ */
209
+ export async function searchKeyword(params: {
210
+ db: DatabaseSync;
211
+ ftsTable: string;
212
+ providerModel: string;
213
+ query: string;
214
+ limit: number;
215
+ snippetMaxChars: number;
216
+ sourceFilter: { sql: string; params: SearchSource[] };
217
+ buildFtsQuery: (raw: string) => string | null;
218
+ bm25RankToScore: (rank: number) => number;
219
+ }): Promise<Array<SearchRowResult & { textScore: number }>> {
220
+ if (params.limit <= 0) return [];
221
+ const ftsQuery = params.buildFtsQuery(params.query);
222
+ if (!ftsQuery) return [];
223
+
224
+ const rows = params.db
225
+ .prepare(
226
+ `SELECT id, path, source, start_line, end_line, text,\n` +
227
+ ` bm25(${params.ftsTable}) AS rank\n` +
228
+ ` FROM ${params.ftsTable}\n` +
229
+ ` WHERE ${params.ftsTable} MATCH ? AND model = ?${params.sourceFilter.sql}\n` +
230
+ ` ORDER BY rank ASC\n` +
231
+ ` LIMIT ?`,
232
+ )
233
+ .all(ftsQuery, params.providerModel, ...params.sourceFilter.params, params.limit) as Array<{
234
+ id: string;
235
+ path: string;
236
+ source: SearchSource;
237
+ start_line: number;
238
+ end_line: number;
239
+ text: string;
240
+ rank: number;
241
+ }>;
242
+
243
+ return rows.map((row) => {
244
+ const textScore = params.bm25RankToScore(row.rank);
245
+ return {
246
+ id: row.id,
247
+ path: row.path,
248
+ startLine: row.start_line,
249
+ endLine: row.end_line,
250
+ score: textScore,
251
+ textScore,
252
+ snippet: truncateUtf16Safe(row.text, params.snippetMaxChars),
253
+ source: row.source,
254
+ };
255
+ });
256
+ }
@@ -0,0 +1,347 @@
1
+ import { describe, expect, it, vi, beforeEach } from "vitest";
2
+
3
+ import { McpServer, createMcpServer, generateMcpConfig } from "../mcp.js";
4
+ import type { Minimem } from "../../minimem.js";
5
+
6
+ // Mock Minimem instance
7
+ const createMockMinimem = () => {
8
+ return {
9
+ search: vi.fn().mockResolvedValue([
10
+ {
11
+ path: "memory/test.md",
12
+ startLine: 1,
13
+ endLine: 5,
14
+ score: 0.85,
15
+ snippet: "Test content about architecture decisions",
16
+ },
17
+ ]),
18
+ readLines: vi.fn().mockResolvedValue({
19
+ content: "Full test content about architecture decisions\nLine 2\nLine 3\nLine 4\nLine 5",
20
+ startLine: 1,
21
+ endLine: 5,
22
+ }),
23
+ } as unknown as Minimem;
24
+ };
25
+
26
+ describe("McpServer", () => {
27
+ let server: McpServer;
28
+ let mockMinimem: ReturnType<typeof createMockMinimem>;
29
+
30
+ beforeEach(() => {
31
+ mockMinimem = createMockMinimem();
32
+ server = createMcpServer(mockMinimem as Minimem);
33
+ });
34
+
35
+ describe("initialize", () => {
36
+ it("responds with protocol version and capabilities", async () => {
37
+ const response = await server.handleRequest({
38
+ jsonrpc: "2.0",
39
+ id: 1,
40
+ method: "initialize",
41
+ params: { protocolVersion: "2024-11-05" },
42
+ });
43
+
44
+ expect(response.error).toBeUndefined();
45
+ expect(response.result).toMatchObject({
46
+ protocolVersion: "2024-11-05",
47
+ capabilities: {
48
+ tools: { listChanged: false },
49
+ },
50
+ serverInfo: {
51
+ name: "minimem",
52
+ version: expect.any(String),
53
+ },
54
+ });
55
+ });
56
+ });
57
+
58
+ describe("tools/list", () => {
59
+ it("returns memory_search and memory_get_details tools", async () => {
60
+ const response = await server.handleRequest({
61
+ jsonrpc: "2.0",
62
+ id: 2,
63
+ method: "tools/list",
64
+ });
65
+
66
+ expect(response.error).toBeUndefined();
67
+ const result = response.result as { tools: Array<{ name: string }> };
68
+ expect(result.tools).toHaveLength(5);
69
+ expect(result.tools.map((t) => t.name)).toEqual([
70
+ "memory_search",
71
+ "memory_get_details",
72
+ "knowledge_search",
73
+ "knowledge_graph",
74
+ "knowledge_path",
75
+ ]);
76
+ });
77
+
78
+ it("tools have valid schemas", async () => {
79
+ const response = await server.handleRequest({
80
+ jsonrpc: "2.0",
81
+ id: 3,
82
+ method: "tools/list",
83
+ });
84
+
85
+ const result = response.result as {
86
+ tools: Array<{
87
+ name: string;
88
+ description: string;
89
+ inputSchema: { type: string };
90
+ }>;
91
+ };
92
+
93
+ for (const tool of result.tools) {
94
+ expect(tool.name).toBeTruthy();
95
+ expect(tool.description).toBeTruthy();
96
+ expect(tool.inputSchema.type).toBe("object");
97
+ }
98
+ });
99
+ });
100
+
101
+ describe("tools/call", () => {
102
+ it("calls memory_search with compact detail by default", async () => {
103
+ const response = await server.handleRequest({
104
+ jsonrpc: "2.0",
105
+ id: 4,
106
+ method: "tools/call",
107
+ params: {
108
+ name: "memory_search",
109
+ arguments: { query: "test query" },
110
+ },
111
+ });
112
+
113
+ expect(response.error).toBeUndefined();
114
+ expect(mockMinimem.search).toHaveBeenCalledWith("test query", {
115
+ maxResults: 15,
116
+ minScore: undefined,
117
+ type: undefined,
118
+ });
119
+
120
+ const result = response.result as { content: Array<{ type: string; text: string }> };
121
+ expect(result.content[0].type).toBe("text");
122
+ expect(result.content[0].text).toContain("memory/test.md");
123
+ // Compact format: includes hint to use memory_get_details
124
+ expect(result.content[0].text).toContain("memory_get_details");
125
+ // Compact format: short preview, not full snippet
126
+ expect(result.content[0].text).toContain("85%");
127
+ });
128
+
129
+ it("calls memory_search with full detail", async () => {
130
+ const response = await server.handleRequest({
131
+ jsonrpc: "2.0",
132
+ id: "full-1",
133
+ method: "tools/call",
134
+ params: {
135
+ name: "memory_search",
136
+ arguments: { query: "test query", detail: "full" },
137
+ },
138
+ });
139
+
140
+ expect(response.error).toBeUndefined();
141
+ const result = response.result as { content: Array<{ type: string; text: string }> };
142
+ // Full format: contains the full snippet text
143
+ expect(result.content[0].text).toContain("Test content about architecture decisions");
144
+ // Full format: does NOT contain the compact hint
145
+ expect(result.content[0].text).not.toContain("Use memory_get_details");
146
+ });
147
+
148
+ it("calls memory_get_details tool", async () => {
149
+ const response = await server.handleRequest({
150
+ jsonrpc: "2.0",
151
+ id: "details-1",
152
+ method: "tools/call",
153
+ params: {
154
+ name: "memory_get_details",
155
+ arguments: {
156
+ results: [{ path: "memory/test.md", startLine: 1, endLine: 5 }],
157
+ },
158
+ },
159
+ });
160
+
161
+ expect(response.error).toBeUndefined();
162
+ expect(mockMinimem.readLines).toHaveBeenCalledWith("memory/test.md", {
163
+ from: 1,
164
+ lines: 5,
165
+ });
166
+
167
+ const result = response.result as { content: Array<{ type: string; text: string }> };
168
+ expect(result.content[0].text).toContain("Full test content about architecture decisions");
169
+ expect(result.content[0].text).toContain("memory/test.md:1-5");
170
+ });
171
+
172
+ it("memory_get_details returns error for empty results", async () => {
173
+ const response = await server.handleRequest({
174
+ jsonrpc: "2.0",
175
+ id: "details-2",
176
+ method: "tools/call",
177
+ params: {
178
+ name: "memory_get_details",
179
+ arguments: { results: [] },
180
+ },
181
+ });
182
+
183
+ expect(response.error).toBeUndefined();
184
+ const result = response.result as { content: Array<{ text: string }>; isError: boolean };
185
+ expect(result.isError).toBe(true);
186
+ });
187
+
188
+ it("returns error for unknown tool", async () => {
189
+ const response = await server.handleRequest({
190
+ jsonrpc: "2.0",
191
+ id: 5,
192
+ method: "tools/call",
193
+ params: {
194
+ name: "unknown_tool",
195
+ arguments: {},
196
+ },
197
+ });
198
+
199
+ expect(response.error).toBeUndefined();
200
+ const result = response.result as { content: Array<{ text: string }>; isError: boolean };
201
+ expect(result.isError).toBe(true);
202
+ expect(result.content[0].text).toContain("Unknown tool");
203
+ });
204
+
205
+ it("passes type filter via MCP to search", async () => {
206
+ const response = await server.handleRequest({
207
+ jsonrpc: "2.0",
208
+ id: "type-1",
209
+ method: "tools/call",
210
+ params: {
211
+ name: "memory_search",
212
+ arguments: { query: "API design", type: "decision" },
213
+ },
214
+ });
215
+
216
+ expect(response.error).toBeUndefined();
217
+ expect(mockMinimem.search).toHaveBeenCalledWith("API design", {
218
+ maxResults: 15,
219
+ minScore: undefined,
220
+ type: "decision",
221
+ });
222
+ });
223
+
224
+ it("passes detail and type together", async () => {
225
+ const response = await server.handleRequest({
226
+ jsonrpc: "2.0",
227
+ id: "combo-1",
228
+ method: "tools/call",
229
+ params: {
230
+ name: "memory_search",
231
+ arguments: { query: "bugs", detail: "full", type: "bugfix" },
232
+ },
233
+ });
234
+
235
+ expect(response.error).toBeUndefined();
236
+ expect(mockMinimem.search).toHaveBeenCalledWith("bugs", {
237
+ maxResults: 15,
238
+ minScore: undefined,
239
+ type: "bugfix",
240
+ });
241
+ const result = response.result as { content: Array<{ text: string }> };
242
+ // Full mode — should have the snippet, not the hint
243
+ expect(result.content[0].text).toContain("Test content about architecture decisions");
244
+ expect(result.content[0].text).not.toContain("Use memory_get_details");
245
+ });
246
+
247
+ it("memory_get_details handles file not found gracefully", async () => {
248
+ // Override readLines to return null
249
+ mockMinimem.readLines = vi.fn().mockResolvedValue(null);
250
+
251
+ const response = await server.handleRequest({
252
+ jsonrpc: "2.0",
253
+ id: "details-notfound",
254
+ method: "tools/call",
255
+ params: {
256
+ name: "memory_get_details",
257
+ arguments: {
258
+ results: [{ path: "memory/missing.md", startLine: 1, endLine: 5 }],
259
+ },
260
+ },
261
+ });
262
+
263
+ expect(response.error).toBeUndefined();
264
+ const result = response.result as { content: Array<{ text: string }> };
265
+ expect(result.content[0].text).toContain("(not found)");
266
+ expect(result.content[0].text).toContain("memory/missing.md:1-5");
267
+ });
268
+
269
+ it("memory_search returns no results message", async () => {
270
+ mockMinimem.search = vi.fn().mockResolvedValue([]);
271
+
272
+ const response = await server.handleRequest({
273
+ jsonrpc: "2.0",
274
+ id: "empty-1",
275
+ method: "tools/call",
276
+ params: {
277
+ name: "memory_search",
278
+ arguments: { query: "something that does not exist" },
279
+ },
280
+ });
281
+
282
+ expect(response.error).toBeUndefined();
283
+ const result = response.result as { content: Array<{ text: string }> };
284
+ expect(result.content[0].text).toBe("No results found.");
285
+ });
286
+
287
+ it("returns error when tool name missing", async () => {
288
+ const response = await server.handleRequest({
289
+ jsonrpc: "2.0",
290
+ id: 6,
291
+ method: "tools/call",
292
+ params: { arguments: {} },
293
+ });
294
+
295
+ expect(response.error).toBeDefined();
296
+ expect(response.error?.message).toContain("Missing tool name");
297
+ });
298
+ });
299
+
300
+ describe("error handling", () => {
301
+ it("returns method not found for unknown methods", async () => {
302
+ const response = await server.handleRequest({
303
+ jsonrpc: "2.0",
304
+ id: 7,
305
+ method: "unknown/method",
306
+ });
307
+
308
+ expect(response.error).toBeDefined();
309
+ expect(response.error?.code).toBe(-32601);
310
+ expect(response.error?.message).toContain("Method not found");
311
+ });
312
+
313
+ it("handles ping method", async () => {
314
+ const response = await server.handleRequest({
315
+ jsonrpc: "2.0",
316
+ id: 8,
317
+ method: "ping",
318
+ });
319
+
320
+ expect(response.error).toBeUndefined();
321
+ expect(response.result).toEqual({});
322
+ });
323
+ });
324
+ });
325
+
326
+ describe("generateMcpConfig", () => {
327
+ it("generates valid config for Claude Desktop", () => {
328
+ const config = generateMcpConfig({
329
+ serverPath: "/path/to/server.js",
330
+ memoryDir: "/home/user/memory",
331
+ });
332
+
333
+ expect(config.command).toBe("node");
334
+ expect(config.args).toEqual(["/path/to/server.js"]);
335
+ expect(config.env?.MEMORY_DIR).toBe("/home/user/memory");
336
+ });
337
+
338
+ it("includes embedding provider when specified", () => {
339
+ const config = generateMcpConfig({
340
+ serverPath: "/path/to/server.js",
341
+ memoryDir: "/home/user/memory",
342
+ embeddingProvider: "openai",
343
+ });
344
+
345
+ expect(config.env?.EMBEDDING_PROVIDER).toBe("openai");
346
+ });
347
+ });