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,364 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+
3
+ import {
4
+ getToolDefinitions,
5
+ MEMORY_TOOLS,
6
+ MEMORY_SEARCH_TOOL,
7
+ MEMORY_GET_DETAILS_TOOL,
8
+ MemoryToolExecutor,
9
+ type ToolDefinition,
10
+ type MemoryInstance,
11
+ } from "../tools.js";
12
+ import type { Minimem } from "../../minimem.js";
13
+
14
+ describe("Tool definitions", () => {
15
+ it("exports memory_search and memory_get_details tools", () => {
16
+ expect(MEMORY_TOOLS).toHaveLength(5);
17
+ expect(MEMORY_TOOLS.map((t) => t.name)).toEqual([
18
+ "memory_search",
19
+ "memory_get_details",
20
+ "knowledge_search",
21
+ "knowledge_graph",
22
+ "knowledge_path",
23
+ ]);
24
+ });
25
+
26
+ it("getToolDefinitions returns all tools", () => {
27
+ const tools = getToolDefinitions();
28
+ expect(tools).toEqual(MEMORY_TOOLS);
29
+ });
30
+ });
31
+
32
+ describe("memory_search tool", () => {
33
+ it("has correct schema", () => {
34
+ expect(MEMORY_SEARCH_TOOL.name).toBe("memory_search");
35
+ expect(MEMORY_SEARCH_TOOL.inputSchema.type).toBe("object");
36
+ expect(MEMORY_SEARCH_TOOL.inputSchema.required).toContain("query");
37
+ expect(MEMORY_SEARCH_TOOL.inputSchema.properties.query.type).toBe("string");
38
+ expect(MEMORY_SEARCH_TOOL.inputSchema.properties.maxResults.type).toBe("number");
39
+ expect(MEMORY_SEARCH_TOOL.inputSchema.properties.minScore.type).toBe("number");
40
+ });
41
+
42
+ it("has detail parameter with compact and full options", () => {
43
+ const detail = MEMORY_SEARCH_TOOL.inputSchema.properties.detail;
44
+ expect(detail.type).toBe("string");
45
+ expect(detail.enum).toEqual(["compact", "full"]);
46
+ });
47
+
48
+ it("has description", () => {
49
+ expect(MEMORY_SEARCH_TOOL.description).toBeTruthy();
50
+ expect(MEMORY_SEARCH_TOOL.description.length).toBeGreaterThan(20);
51
+ });
52
+ });
53
+
54
+ describe("memory_get_details tool", () => {
55
+ it("has correct schema", () => {
56
+ expect(MEMORY_GET_DETAILS_TOOL.name).toBe("memory_get_details");
57
+ expect(MEMORY_GET_DETAILS_TOOL.inputSchema.type).toBe("object");
58
+ expect(MEMORY_GET_DETAILS_TOOL.inputSchema.required).toContain("results");
59
+ expect(MEMORY_GET_DETAILS_TOOL.inputSchema.properties.results.type).toBe("array");
60
+ });
61
+
62
+ it("has description mentioning two-step approach", () => {
63
+ expect(MEMORY_GET_DETAILS_TOOL.description).toContain("two-step");
64
+ });
65
+ });
66
+
67
+ describe("Tool schema compatibility", () => {
68
+ const validateToolSchema = (tool: ToolDefinition) => {
69
+ // MCP/OpenAI/Anthropic require type: "object" at top level
70
+ expect(tool.inputSchema.type).toBe("object");
71
+
72
+ // All tools need name and description
73
+ expect(tool.name).toBeTruthy();
74
+ expect(tool.description).toBeTruthy();
75
+
76
+ // Properties should be an object
77
+ expect(typeof tool.inputSchema.properties).toBe("object");
78
+
79
+ // Required should be an array (or undefined)
80
+ if (tool.inputSchema.required !== undefined) {
81
+ expect(Array.isArray(tool.inputSchema.required)).toBe(true);
82
+ }
83
+ };
84
+
85
+ it("all tools have valid schemas", () => {
86
+ for (const tool of MEMORY_TOOLS) {
87
+ validateToolSchema(tool);
88
+ }
89
+ });
90
+
91
+ it("tool names are snake_case", () => {
92
+ for (const tool of MEMORY_TOOLS) {
93
+ expect(tool.name).toMatch(/^[a-z][a-z0-9_]*$/);
94
+ }
95
+ });
96
+
97
+ it("property types are valid JSON Schema types", () => {
98
+ const validTypes = ["string", "number", "boolean", "array", "object", "null"];
99
+ for (const tool of MEMORY_TOOLS) {
100
+ for (const prop of Object.values(tool.inputSchema.properties)) {
101
+ expect(validTypes).toContain(prop.type);
102
+ }
103
+ }
104
+ });
105
+ });
106
+
107
+ describe("memory_search tool schema", () => {
108
+ it("has type parameter for observation filtering", () => {
109
+ const type = MEMORY_SEARCH_TOOL.inputSchema.properties.type;
110
+ expect(type).toBeDefined();
111
+ expect(type.type).toBe("string");
112
+ expect(type.description).toContain("type");
113
+ });
114
+
115
+ it("has directories parameter", () => {
116
+ const dirs = MEMORY_SEARCH_TOOL.inputSchema.properties.directories;
117
+ expect(dirs).toBeDefined();
118
+ expect(dirs.type).toBe("array");
119
+ });
120
+ });
121
+
122
+ // Helper to create mock Minimem for executor tests
123
+ function createMockMinimem(searchResults: Array<{
124
+ path: string;
125
+ startLine: number;
126
+ endLine: number;
127
+ score: number;
128
+ snippet: string;
129
+ }> = []) {
130
+ return {
131
+ search: vi.fn().mockResolvedValue(searchResults),
132
+ readLines: vi.fn().mockImplementation((path: string, opts?: { from?: number; lines?: number }) => {
133
+ // Simulate returning content for known paths
134
+ if (path === "memory/test.md") {
135
+ return Promise.resolve({
136
+ content: "Full content of test.md\nLine 2\nLine 3",
137
+ startLine: opts?.from ?? 1,
138
+ endLine: (opts?.from ?? 1) + (opts?.lines ?? 3) - 1,
139
+ });
140
+ }
141
+ return Promise.resolve(null);
142
+ }),
143
+ } as unknown as Minimem;
144
+ }
145
+
146
+ describe("MemoryToolExecutor", () => {
147
+ describe("memorySearch with type filter", () => {
148
+ it("passes type parameter to search", async () => {
149
+ const mock = createMockMinimem([
150
+ { path: "memory/test.md", startLine: 1, endLine: 5, score: 0.85, snippet: "Decision about API" },
151
+ ]);
152
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
153
+
154
+ await executor.execute("memory_search", {
155
+ query: "API design",
156
+ type: "decision",
157
+ });
158
+
159
+ expect(mock.search).toHaveBeenCalledWith("API design", {
160
+ maxResults: 15,
161
+ minScore: undefined,
162
+ type: "decision",
163
+ });
164
+ });
165
+
166
+ it("does not pass type when not specified", async () => {
167
+ const mock = createMockMinimem([]);
168
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
169
+
170
+ await executor.execute("memory_search", { query: "test" });
171
+
172
+ expect(mock.search).toHaveBeenCalledWith("test", {
173
+ maxResults: 15,
174
+ minScore: undefined,
175
+ type: undefined,
176
+ });
177
+ });
178
+ });
179
+
180
+ describe("memorySearch compact format", () => {
181
+ it("returns compact format by default with preview and hint", async () => {
182
+ const mock = createMockMinimem([
183
+ { path: "memory/test.md", startLine: 1, endLine: 5, score: 0.9, snippet: "# Architecture Decision\nWe chose TypeScript." },
184
+ ]);
185
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
186
+
187
+ const result = await executor.execute("memory_search", { query: "architecture" });
188
+
189
+ const text = result.content[0].text;
190
+ // Compact: uses [0] indexing, percentage without decimal, short preview
191
+ expect(text).toContain("[0]");
192
+ expect(text).toContain("90%");
193
+ expect(text).toContain("memory/test.md:1-5");
194
+ // Should show a heading-based preview
195
+ expect(text).toContain("Architecture Decision");
196
+ // Should include hint
197
+ expect(text).toContain("memory_get_details");
198
+ });
199
+
200
+ it("truncates long previews to ~80 chars with ellipsis", async () => {
201
+ const longSnippet = "A".repeat(120);
202
+ const mock = createMockMinimem([
203
+ { path: "test.md", startLine: 1, endLine: 1, score: 0.8, snippet: longSnippet },
204
+ ]);
205
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
206
+
207
+ const result = await executor.execute("memory_search", { query: "test" });
208
+
209
+ const text = result.content[0].text;
210
+ // Preview should be truncated — the line with the preview should be ~80 chars + prefix
211
+ const previewMatch = text.match(/"([^"]+)"/);
212
+ expect(previewMatch).toBeTruthy();
213
+ expect(previewMatch![1].length).toBeLessThanOrEqual(80);
214
+ expect(previewMatch![1]).toContain("...");
215
+ });
216
+
217
+ it("shows (empty) for empty snippet", async () => {
218
+ const mock = createMockMinimem([
219
+ { path: "test.md", startLine: 1, endLine: 1, score: 0.5, snippet: "" },
220
+ ]);
221
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
222
+
223
+ const result = await executor.execute("memory_search", { query: "test" });
224
+ expect(result.content[0].text).toContain("(empty)");
225
+ });
226
+
227
+ it("returns no results message when empty", async () => {
228
+ const mock = createMockMinimem([]);
229
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
230
+
231
+ const result = await executor.execute("memory_search", { query: "nonexistent" });
232
+ expect(result.content[0].text).toBe("No results found.");
233
+ });
234
+ });
235
+
236
+ describe("memorySearch full format", () => {
237
+ it("returns full snippets when detail=full", async () => {
238
+ const mock = createMockMinimem([
239
+ { path: "test.md", startLine: 1, endLine: 5, score: 0.85, snippet: "Full detailed content here\nWith multiple lines" },
240
+ ]);
241
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
242
+
243
+ const result = await executor.execute("memory_search", { query: "test", detail: "full" });
244
+
245
+ const text = result.content[0].text;
246
+ expect(text).toContain("Full detailed content here");
247
+ expect(text).toContain("With multiple lines");
248
+ // Full mode uses [1] indexing and .1 decimal precision
249
+ expect(text).toContain("[1]");
250
+ expect(text).toContain("85.0%");
251
+ // No hint about memory_get_details
252
+ expect(text).not.toContain("Use memory_get_details");
253
+ });
254
+ });
255
+
256
+ describe("memorySearch multi-directory", () => {
257
+ it("shows directory source when searching multiple dirs", async () => {
258
+ const mock1 = createMockMinimem([
259
+ { path: "MEMORY.md", startLine: 1, endLine: 3, score: 0.9, snippet: "From project A" },
260
+ ]);
261
+ const mock2 = createMockMinimem([
262
+ { path: "MEMORY.md", startLine: 1, endLine: 3, score: 0.8, snippet: "From project B" },
263
+ ]);
264
+ const executor = new MemoryToolExecutor([
265
+ { minimem: mock1, memoryDir: "/projectA", name: "projectA" },
266
+ { minimem: mock2, memoryDir: "/projectB", name: "projectB" },
267
+ ]);
268
+
269
+ const result = await executor.execute("memory_search", { query: "test", detail: "full" });
270
+ const text = result.content[0].text;
271
+ expect(text).toContain("[projectA]");
272
+ expect(text).toContain("[projectB]");
273
+ expect(text).toContain("Searched 2 directories");
274
+ });
275
+
276
+ it("returns error when directory filter matches nothing", async () => {
277
+ const mock = createMockMinimem([]);
278
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test", name: "test" });
279
+
280
+ const result = await executor.execute("memory_search", {
281
+ query: "test",
282
+ directories: ["nonexistent"],
283
+ });
284
+ expect(result.isError).toBe(true);
285
+ expect(result.content[0].text).toContain("No matching directories");
286
+ });
287
+ });
288
+
289
+ describe("memoryGetDetails", () => {
290
+ it("returns full text for requested chunks", async () => {
291
+ const mock = createMockMinimem();
292
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
293
+
294
+ const result = await executor.execute("memory_get_details", {
295
+ results: [{ path: "memory/test.md", startLine: 1, endLine: 3 }],
296
+ });
297
+
298
+ expect(mock.readLines).toHaveBeenCalledWith("memory/test.md", { from: 1, lines: 3 });
299
+ const text = result.content[0].text;
300
+ expect(text).toContain("memory/test.md:1-3");
301
+ expect(text).toContain("Full content of test.md");
302
+ });
303
+
304
+ it("returns (not found) for non-existent files", async () => {
305
+ const mock = createMockMinimem();
306
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
307
+
308
+ const result = await executor.execute("memory_get_details", {
309
+ results: [{ path: "memory/nonexistent.md", startLine: 1, endLine: 5 }],
310
+ });
311
+
312
+ const text = result.content[0].text;
313
+ expect(text).toContain("(not found)");
314
+ expect(text).toContain("memory/nonexistent.md:1-5");
315
+ });
316
+
317
+ it("handles multiple results with mixed found/not-found", async () => {
318
+ const mock = createMockMinimem();
319
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
320
+
321
+ const result = await executor.execute("memory_get_details", {
322
+ results: [
323
+ { path: "memory/test.md", startLine: 1, endLine: 3 },
324
+ { path: "memory/missing.md", startLine: 1, endLine: 5 },
325
+ ],
326
+ });
327
+
328
+ const text = result.content[0].text;
329
+ expect(text).toContain("Full content of test.md");
330
+ expect(text).toContain("(not found)");
331
+ });
332
+
333
+ it("returns error for empty results array", async () => {
334
+ const mock = createMockMinimem();
335
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
336
+
337
+ const result = await executor.execute("memory_get_details", { results: [] });
338
+ expect(result.isError).toBe(true);
339
+ });
340
+
341
+ it("returns error when directory filter matches nothing", async () => {
342
+ const mock = createMockMinimem();
343
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test", name: "test" });
344
+
345
+ const result = await executor.execute("memory_get_details", {
346
+ results: [{ path: "test.md", startLine: 1, endLine: 1 }],
347
+ directories: ["nonexistent"],
348
+ });
349
+ expect(result.isError).toBe(true);
350
+ expect(result.content[0].text).toContain("No matching directories");
351
+ });
352
+ });
353
+
354
+ describe("unknown tools", () => {
355
+ it("returns error for unknown tool name", async () => {
356
+ const mock = createMockMinimem();
357
+ const executor = new MemoryToolExecutor({ minimem: mock, memoryDir: "/test" });
358
+
359
+ const result = await executor.execute("nonexistent_tool", {});
360
+ expect(result.isError).toBe(true);
361
+ expect(result.content[0].text).toContain("Unknown tool");
362
+ });
363
+ });
364
+ });
@@ -0,0 +1,326 @@
1
+ /**
2
+ * MCP (Model Context Protocol) Server for Minimem
3
+ *
4
+ * Provides memory tools via JSON-RPC 2.0 over stdio.
5
+ * Compatible with Claude Desktop, Cursor, and other MCP clients.
6
+ *
7
+ * Usage:
8
+ * import { Minimem } from "minimem";
9
+ * import { createMcpServer, runMcpServer } from "minimem/mcp";
10
+ *
11
+ * const minimem = await Minimem.create({ ... });
12
+ * const server = createMcpServer(minimem);
13
+ * await runMcpServer(server); // Runs over stdio
14
+ */
15
+
16
+ import * as readline from "node:readline";
17
+ import type { Minimem } from "../minimem.js";
18
+ import {
19
+ MEMORY_TOOLS,
20
+ type ToolDefinition,
21
+ type ToolResult,
22
+ type MemoryInstance,
23
+ MemoryToolExecutor,
24
+ } from "./tools.js";
25
+
26
+ const PROTOCOL_VERSION = "2024-11-05";
27
+ const SERVER_NAME = "minimem";
28
+ const SERVER_VERSION = "0.1.0";
29
+
30
+ /**
31
+ * JSON-RPC 2.0 request
32
+ */
33
+ type JsonRpcRequest = {
34
+ jsonrpc: "2.0";
35
+ id: string | number;
36
+ method: string;
37
+ params?: Record<string, unknown>;
38
+ };
39
+
40
+ /**
41
+ * JSON-RPC 2.0 response
42
+ */
43
+ type JsonRpcResponse = {
44
+ jsonrpc: "2.0";
45
+ id: string | number | null;
46
+ result?: unknown;
47
+ error?: {
48
+ code: number;
49
+ message: string;
50
+ data?: unknown;
51
+ };
52
+ };
53
+
54
+ /**
55
+ * JSON-RPC 2.0 notification (no id)
56
+ */
57
+ type JsonRpcNotification = {
58
+ jsonrpc: "2.0";
59
+ method: string;
60
+ params?: Record<string, unknown>;
61
+ };
62
+
63
+ /**
64
+ * MCP Server capabilities
65
+ */
66
+ type ServerCapabilities = {
67
+ tools?: {
68
+ listChanged?: boolean;
69
+ };
70
+ };
71
+
72
+ /**
73
+ * MCP Server info
74
+ */
75
+ type ServerInfo = {
76
+ name: string;
77
+ version: string;
78
+ };
79
+
80
+ /**
81
+ * MCP Initialize result
82
+ */
83
+ type InitializeResult = {
84
+ protocolVersion: string;
85
+ capabilities: ServerCapabilities;
86
+ serverInfo: ServerInfo;
87
+ };
88
+
89
+ /**
90
+ * MCP Tool in list format
91
+ */
92
+ type McpTool = {
93
+ name: string;
94
+ description: string;
95
+ inputSchema: {
96
+ type: "object";
97
+ properties: Record<string, unknown>;
98
+ required?: string[];
99
+ };
100
+ };
101
+
102
+ /**
103
+ * MCP Server implementation
104
+ */
105
+ export class McpServer {
106
+ private executor: MemoryToolExecutor;
107
+ private initialized = false;
108
+
109
+ constructor(instances: Minimem | MemoryInstance | MemoryInstance[]) {
110
+ this.executor = new MemoryToolExecutor(instances);
111
+ }
112
+
113
+ /**
114
+ * Handle a JSON-RPC request and return a response
115
+ */
116
+ async handleRequest(request: JsonRpcRequest): Promise<JsonRpcResponse> {
117
+ try {
118
+ const result = await this.dispatch(request.method, request.params);
119
+ return {
120
+ jsonrpc: "2.0",
121
+ id: request.id,
122
+ result,
123
+ };
124
+ } catch (err) {
125
+ const message = err instanceof Error ? err.message : String(err);
126
+ const code = err instanceof McpError ? err.code : -32603;
127
+ return {
128
+ jsonrpc: "2.0",
129
+ id: request.id,
130
+ error: { code, message },
131
+ };
132
+ }
133
+ }
134
+
135
+ /**
136
+ * Dispatch a method call
137
+ */
138
+ private async dispatch(
139
+ method: string,
140
+ params?: Record<string, unknown>,
141
+ ): Promise<unknown> {
142
+ switch (method) {
143
+ case "initialize":
144
+ return this.initialize(params);
145
+ case "initialized":
146
+ // Notification, no response needed
147
+ return {};
148
+ case "tools/list":
149
+ return this.listTools();
150
+ case "tools/call":
151
+ return this.callTool(params);
152
+ case "ping":
153
+ return {};
154
+ default:
155
+ throw new McpError(-32601, `Method not found: ${method}`);
156
+ }
157
+ }
158
+
159
+ /**
160
+ * Handle initialize request
161
+ */
162
+ private initialize(
163
+ params?: Record<string, unknown>,
164
+ ): InitializeResult {
165
+ this.initialized = true;
166
+ return {
167
+ protocolVersion: PROTOCOL_VERSION,
168
+ capabilities: {
169
+ tools: {
170
+ listChanged: false,
171
+ },
172
+ },
173
+ serverInfo: {
174
+ name: SERVER_NAME,
175
+ version: SERVER_VERSION,
176
+ },
177
+ };
178
+ }
179
+
180
+ /**
181
+ * List available tools
182
+ */
183
+ private listTools(): { tools: McpTool[] } {
184
+ const tools: McpTool[] = MEMORY_TOOLS.map((tool) => ({
185
+ name: tool.name,
186
+ description: tool.description,
187
+ inputSchema: tool.inputSchema,
188
+ }));
189
+ return { tools };
190
+ }
191
+
192
+ /**
193
+ * Call a tool
194
+ */
195
+ private async callTool(
196
+ params?: Record<string, unknown>,
197
+ ): Promise<{ content: Array<{ type: string; text: string }>; isError?: boolean }> {
198
+ if (!params?.name || typeof params.name !== "string") {
199
+ throw new McpError(-32602, "Missing tool name");
200
+ }
201
+
202
+ const toolName = params.name;
203
+ const toolParams = (params.arguments ?? {}) as Record<string, unknown>;
204
+
205
+ const result = await this.executor.execute(toolName, toolParams);
206
+ return result;
207
+ }
208
+ }
209
+
210
+ /**
211
+ * Custom error class for MCP errors
212
+ */
213
+ class McpError extends Error {
214
+ constructor(
215
+ public code: number,
216
+ message: string,
217
+ ) {
218
+ super(message);
219
+ this.name = "McpError";
220
+ }
221
+ }
222
+
223
+ /**
224
+ * Create an MCP server for the given Minimem instance(s)
225
+ */
226
+ export function createMcpServer(
227
+ instances: Minimem | MemoryInstance | MemoryInstance[],
228
+ ): McpServer {
229
+ return new McpServer(instances);
230
+ }
231
+
232
+ /**
233
+ * Run the MCP server over stdio
234
+ */
235
+ export async function runMcpServer(server: McpServer): Promise<void> {
236
+ const rl = readline.createInterface({
237
+ input: process.stdin,
238
+ output: process.stdout,
239
+ terminal: false,
240
+ });
241
+
242
+ const send = (message: JsonRpcResponse | JsonRpcNotification) => {
243
+ const json = JSON.stringify(message);
244
+ process.stdout.write(json + "\n");
245
+ };
246
+
247
+ rl.on("line", async (line) => {
248
+ if (!line.trim()) return;
249
+
250
+ try {
251
+ const request = JSON.parse(line) as JsonRpcRequest;
252
+
253
+ if (request.jsonrpc !== "2.0") {
254
+ send({
255
+ jsonrpc: "2.0",
256
+ id: request.id ?? null,
257
+ error: { code: -32600, message: "Invalid JSON-RPC version" },
258
+ });
259
+ return;
260
+ }
261
+
262
+ // Handle notification (no id)
263
+ if (request.id === undefined) {
264
+ await server.handleRequest({ ...request, id: 0 });
265
+ return;
266
+ }
267
+
268
+ const response = await server.handleRequest(request);
269
+ send(response);
270
+ } catch (err) {
271
+ const message = err instanceof Error ? err.message : String(err);
272
+ send({
273
+ jsonrpc: "2.0",
274
+ id: null,
275
+ error: { code: -32700, message: `Parse error: ${message}` },
276
+ });
277
+ }
278
+ });
279
+
280
+ rl.on("close", () => {
281
+ process.exit(0);
282
+ });
283
+
284
+ // Keep the process alive
285
+ await new Promise(() => {});
286
+ }
287
+
288
+ /**
289
+ * MCP server configuration for claude_desktop_config.json
290
+ *
291
+ * Example:
292
+ * {
293
+ * "mcpServers": {
294
+ * "minimem": {
295
+ * "command": "node",
296
+ * "args": ["path/to/your/mcp-server.js"],
297
+ * "env": {
298
+ * "MEMORY_DIR": "/path/to/memory"
299
+ * }
300
+ * }
301
+ * }
302
+ * }
303
+ */
304
+ export type McpServerConfig = {
305
+ command: string;
306
+ args: string[];
307
+ env?: Record<string, string>;
308
+ };
309
+
310
+ /**
311
+ * Generate MCP server config for Claude Desktop
312
+ */
313
+ export function generateMcpConfig(opts: {
314
+ serverPath: string;
315
+ memoryDir: string;
316
+ embeddingProvider?: "openai" | "gemini" | "local" | "auto";
317
+ }): McpServerConfig {
318
+ return {
319
+ command: "node",
320
+ args: [opts.serverPath],
321
+ env: {
322
+ MEMORY_DIR: opts.memoryDir,
323
+ ...(opts.embeddingProvider ? { EMBEDDING_PROVIDER: opts.embeddingProvider } : {}),
324
+ },
325
+ };
326
+ }