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,265 @@
1
+ /**
2
+ * Error path tests
3
+ *
4
+ * Tests error handling and recovery for:
5
+ * - Embedding API failures and timeouts
6
+ * - File system errors
7
+ * - Invalid configurations
8
+ * - Concurrent sync operations
9
+ */
10
+
11
+ import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
12
+ import fs from "node:fs/promises";
13
+ import os from "node:os";
14
+ import path from "node:path";
15
+
16
+ import { listMemoryFiles, logError, ensureDir } from "../internal.js";
17
+ import { parseFrontmatter, serializeFrontmatter, addFrontmatter } from "../session.js";
18
+ import { createEmbeddingProvider, type EmbeddingProviderOptions } from "../embeddings/embeddings.js";
19
+
20
+ describe("Error path: listMemoryFiles", () => {
21
+ let tempDir: string;
22
+
23
+ beforeEach(async () => {
24
+ tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-err-"));
25
+ });
26
+
27
+ afterEach(async () => {
28
+ await fs.rm(tempDir, { recursive: true, force: true });
29
+ });
30
+
31
+ it("throws when both MEMORY.md and memory.md exist as separate files", async () => {
32
+ await fs.writeFile(path.join(tempDir, "MEMORY.md"), "# Upper");
33
+ await fs.writeFile(path.join(tempDir, "memory.md"), "# Lower");
34
+
35
+ // On case-insensitive filesystems (macOS HFS+/APFS), both names resolve
36
+ // to the same file, so the error won't be thrown. Skip in that case.
37
+ const upperReal = await fs.realpath(path.join(tempDir, "MEMORY.md"));
38
+ const lowerReal = await fs.realpath(path.join(tempDir, "memory.md"));
39
+ if (upperReal === lowerReal) {
40
+ // Case-insensitive FS — can't have two distinct files
41
+ const files = await listMemoryFiles(tempDir);
42
+ expect(files).toHaveLength(1);
43
+ return;
44
+ }
45
+
46
+ await expect(listMemoryFiles(tempDir)).rejects.toThrow(
47
+ /Both MEMORY\.md and memory\.md exist/,
48
+ );
49
+ });
50
+
51
+ it("returns empty array when directory has no memory files", async () => {
52
+ const files = await listMemoryFiles(tempDir);
53
+ expect(files).toHaveLength(0);
54
+ });
55
+
56
+ it("handles non-existent directory gracefully", async () => {
57
+ const badDir = path.join(tempDir, "does-not-exist");
58
+ const files = await listMemoryFiles(badDir);
59
+ expect(files).toHaveLength(0);
60
+ });
61
+
62
+ it("only returns one file when both MEMORY.md and memory.md resolve to same path", async () => {
63
+ // On case-insensitive filesystems, MEMORY.md and memory.md are the same file.
64
+ // On case-sensitive filesystems, this test verifies the single-file path.
65
+ await fs.writeFile(path.join(tempDir, "MEMORY.md"), "# Test");
66
+
67
+ const files = await listMemoryFiles(tempDir);
68
+ expect(files).toHaveLength(1);
69
+ expect(files[0]).toContain("MEMORY.md");
70
+ });
71
+ });
72
+
73
+ describe("Error path: logError", () => {
74
+ it("does nothing when debug is undefined", () => {
75
+ // Should not throw
76
+ logError("test", new Error("something broke"));
77
+ });
78
+
79
+ it("logs error message when debug is provided", () => {
80
+ const messages: string[] = [];
81
+ const debug = (msg: string) => messages.push(msg);
82
+
83
+ logError("testContext", new Error("something broke"), debug);
84
+
85
+ expect(messages).toHaveLength(1);
86
+ expect(messages[0]).toContain("[testContext]");
87
+ expect(messages[0]).toContain("something broke");
88
+ });
89
+
90
+ it("handles non-Error objects", () => {
91
+ const messages: string[] = [];
92
+ const debug = (msg: string) => messages.push(msg);
93
+
94
+ logError("testContext", "string error", debug);
95
+ expect(messages[0]).toContain("string error");
96
+
97
+ logError("testContext", 42, debug);
98
+ expect(messages[1]).toContain("42");
99
+
100
+ logError("testContext", null, debug);
101
+ expect(messages[2]).toContain("null");
102
+ });
103
+ });
104
+
105
+ describe("Error path: ensureDir", () => {
106
+ let tempDir: string;
107
+
108
+ beforeEach(async () => {
109
+ tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-err-"));
110
+ });
111
+
112
+ afterEach(async () => {
113
+ await fs.rm(tempDir, { recursive: true, force: true });
114
+ });
115
+
116
+ it("creates nested directories", () => {
117
+ const nested = path.join(tempDir, "a", "b", "c");
118
+ const result = ensureDir(nested);
119
+ expect(result).toBe(nested);
120
+ });
121
+
122
+ it("succeeds when directory already exists", () => {
123
+ ensureDir(tempDir);
124
+ const result = ensureDir(tempDir); // Call again
125
+ expect(result).toBe(tempDir);
126
+ });
127
+
128
+ it("logs non-EEXIST errors through debug", () => {
129
+ const messages: string[] = [];
130
+ const debug = (msg: string) => messages.push(msg);
131
+
132
+ // Attempt to create a directory with an invalid path (empty string)
133
+ // This may or may not fail depending on OS, so we just verify it doesn't throw
134
+ ensureDir(path.join(tempDir, "valid"), debug);
135
+ // No error logged for successful creation
136
+ });
137
+ });
138
+
139
+ describe("Error path: createEmbeddingProvider", () => {
140
+ it("returns no-op provider for 'none' without type coercion", async () => {
141
+ const options: EmbeddingProviderOptions = { provider: "none" };
142
+ const result = await createEmbeddingProvider(options);
143
+
144
+ expect(result.provider.id).toBe("none");
145
+ expect(result.provider.model).toBe("bm25-only");
146
+ expect(result.requestedProvider).toBe("none");
147
+ });
148
+
149
+ it("falls back to BM25-only when no API keys available in auto mode", async () => {
150
+ // Clear any API keys
151
+ const origOpenAi = process.env.OPENAI_API_KEY;
152
+ const origGoogle = process.env.GOOGLE_API_KEY;
153
+ const origGemini = process.env.GEMINI_API_KEY;
154
+ delete process.env.OPENAI_API_KEY;
155
+ delete process.env.GOOGLE_API_KEY;
156
+ delete process.env.GEMINI_API_KEY;
157
+
158
+ try {
159
+ const options: EmbeddingProviderOptions = { provider: "auto" };
160
+ const result = await createEmbeddingProvider(options);
161
+
162
+ expect(result.provider.id).toBe("none");
163
+ expect(result.requestedProvider).toBe("auto");
164
+ expect(result.fallbackFrom).toBe("auto");
165
+ expect(result.fallbackReason).toContain("BM25");
166
+ } finally {
167
+ // Restore
168
+ if (origOpenAi) process.env.OPENAI_API_KEY = origOpenAi;
169
+ if (origGoogle) process.env.GOOGLE_API_KEY = origGoogle;
170
+ if (origGemini) process.env.GEMINI_API_KEY = origGemini;
171
+ }
172
+ });
173
+
174
+ it("throws when openai explicitly requested without API key", async () => {
175
+ const origKey = process.env.OPENAI_API_KEY;
176
+ delete process.env.OPENAI_API_KEY;
177
+
178
+ try {
179
+ const options: EmbeddingProviderOptions = { provider: "openai" };
180
+ await expect(createEmbeddingProvider(options)).rejects.toThrow(/API key/);
181
+ } finally {
182
+ if (origKey) process.env.OPENAI_API_KEY = origKey;
183
+ }
184
+ });
185
+
186
+ it("throws when gemini explicitly requested without API key", async () => {
187
+ const origGoogle = process.env.GOOGLE_API_KEY;
188
+ const origGemini = process.env.GEMINI_API_KEY;
189
+ delete process.env.GOOGLE_API_KEY;
190
+ delete process.env.GEMINI_API_KEY;
191
+
192
+ try {
193
+ const options: EmbeddingProviderOptions = { provider: "gemini" };
194
+ await expect(createEmbeddingProvider(options)).rejects.toThrow(/API key/);
195
+ } finally {
196
+ if (origGoogle) process.env.GOOGLE_API_KEY = origGoogle;
197
+ if (origGemini) process.env.GEMINI_API_KEY = origGemini;
198
+ }
199
+ });
200
+
201
+ it("no-op provider returns empty vectors", async () => {
202
+ const options: EmbeddingProviderOptions = { provider: "none" };
203
+ const result = await createEmbeddingProvider(options);
204
+
205
+ const queryResult = await result.provider.embedQuery("test");
206
+ expect(queryResult).toEqual([]);
207
+
208
+ const batchResult = await result.provider.embedBatch(["test1", "test2"]);
209
+ expect(batchResult).toEqual([[], []]);
210
+ });
211
+ });
212
+
213
+ describe("Error path: parseFrontmatter", () => {
214
+ it("returns undefined frontmatter for content without frontmatter", () => {
215
+ const { frontmatter, body } = parseFrontmatter("# Just a heading\n\nSome content.");
216
+ expect(frontmatter).toBeUndefined();
217
+ expect(body).toBe("# Just a heading\n\nSome content.");
218
+ });
219
+
220
+ it("returns undefined frontmatter for malformed YAML", () => {
221
+ const content = "---\n{{{invalid: yaml: }}}\n---\nBody";
222
+ const { frontmatter, body } = parseFrontmatter(content);
223
+ // Malformed YAML should fall through to body
224
+ expect(body).toBeTruthy();
225
+ });
226
+
227
+ it("handles empty frontmatter block", () => {
228
+ const content = "---\n\n---\nBody content";
229
+ const { frontmatter, body } = parseFrontmatter(content);
230
+ expect(body).toBe("Body content");
231
+ });
232
+
233
+ it("handles content that starts with --- but has no closing ---", () => {
234
+ const content = "---\nkey: value\nMore content without closing";
235
+ const { frontmatter, body } = parseFrontmatter(content);
236
+ // No closing --- means no frontmatter
237
+ expect(frontmatter).toBeUndefined();
238
+ expect(body).toBe(content);
239
+ });
240
+ });
241
+
242
+ describe("Error path: addFrontmatter", () => {
243
+ it("creates frontmatter on content without any", () => {
244
+ const result = addFrontmatter("# Hello", { tags: ["test"] });
245
+ expect(result).toContain("---");
246
+ expect(result).toContain("tags: [test]");
247
+ expect(result).toContain("# Hello");
248
+ });
249
+
250
+ it("merges with existing frontmatter", () => {
251
+ const existing = "---\ncreated: 2024-01-01T00:00:00Z\n---\n# Hello";
252
+ const result = addFrontmatter(existing, { tags: ["new"] });
253
+ expect(result).toContain("tags: [new]");
254
+ expect(result).toContain("created: 2024-01-01T00:00:00Z");
255
+ expect(result).toContain("# Hello");
256
+ });
257
+
258
+ it("preserves session data when adding other frontmatter", () => {
259
+ const existing = "---\nsession:\n id: abc123\n source: test\n---\n# Hello";
260
+ const result = addFrontmatter(existing, { tags: ["new"] });
261
+ expect(result).toContain("id: abc123");
262
+ expect(result).toContain("source: test");
263
+ expect(result).toContain("tags: [new]");
264
+ });
265
+ });
@@ -0,0 +1,199 @@
1
+ /**
2
+ * Shared test utilities for minimem tests
3
+ *
4
+ * These helpers provide deterministic, API-free testing of embedding
5
+ * and search functionality.
6
+ */
7
+
8
+ import fs from "node:fs/promises";
9
+ import os from "node:os";
10
+ import path from "node:path";
11
+ import { mock } from "node:test";
12
+
13
+ /**
14
+ * Create a deterministic embedding based on keyword presence.
15
+ *
16
+ * Returns a normalized vector where dimensions correspond to keyword
17
+ * frequencies. This allows predictable test results without API calls.
18
+ *
19
+ * @param text - The text to embed
20
+ * @param dims - Number of dimensions (default: 128)
21
+ * @returns Normalized embedding vector
22
+ */
23
+ export function createDeterministicEmbedding(text: string, dims = 128): number[] {
24
+ const lower = text.toLowerCase();
25
+ const keywords = [
26
+ "project", "meeting", "todo", "bug", "feature", "api", "database", "user",
27
+ "test", "deploy", "config", "error", "fix", "update", "review", "design",
28
+ "alpha", "beta", "gamma", "delta", "epsilon", "zeta", "eta", "theta",
29
+ "important", "urgent", "note", "remember", "decision", "action", "plan", "goal",
30
+ ];
31
+
32
+ const vec = new Array(dims).fill(0);
33
+
34
+ // Set values based on keyword presence
35
+ keywords.forEach((keyword, i) => {
36
+ if (i < dims) {
37
+ const count = (lower.match(new RegExp(keyword, "g")) || []).length;
38
+ vec[i] = count * 0.5;
39
+ }
40
+ });
41
+
42
+ // Add some variation based on text length and characters
43
+ for (let i = 32; i < dims; i++) {
44
+ vec[i] = (lower.charCodeAt(i % lower.length) || 0) / 1000;
45
+ }
46
+
47
+ // Normalize
48
+ const magnitude = Math.sqrt(vec.reduce((sum, v) => sum + v * v, 0)) || 1;
49
+ return vec.map(v => v / magnitude);
50
+ }
51
+
52
+ /**
53
+ * Create a mock fetch function that returns deterministic embeddings.
54
+ *
55
+ * Handles OpenAI and Gemini embedding API endpoints.
56
+ */
57
+ export function createMockFetch() {
58
+ return mock.fn(async (url: string | URL, init?: RequestInit) => {
59
+ const urlStr = url.toString();
60
+ const body = init?.body ? JSON.parse(init.body as string) : {};
61
+
62
+ // Handle OpenAI embeddings endpoint
63
+ if (urlStr.includes("/embeddings")) {
64
+ const inputs = Array.isArray(body.input) ? body.input : [body.input];
65
+ const data = inputs.map((text: string, index: number) => ({
66
+ object: "embedding",
67
+ index,
68
+ embedding: createDeterministicEmbedding(text),
69
+ }));
70
+
71
+ return {
72
+ ok: true,
73
+ status: 200,
74
+ json: async () => ({ object: "list", data, model: body.model }),
75
+ text: async () => JSON.stringify({ object: "list", data, model: body.model }),
76
+ };
77
+ }
78
+
79
+ // Handle Gemini embeddings endpoint
80
+ if (urlStr.includes("embedContent") || urlStr.includes("batchEmbedContents")) {
81
+ const text = body.content?.parts?.[0]?.text || body.requests?.[0]?.content?.parts?.[0]?.text || "";
82
+ const embedding = createDeterministicEmbedding(text);
83
+
84
+ if (urlStr.includes("batchEmbedContents")) {
85
+ const requests = body.requests || [];
86
+ return {
87
+ ok: true,
88
+ status: 200,
89
+ json: async () => ({
90
+ embeddings: requests.map((req: { content?: { parts?: { text?: string }[] } }) => ({
91
+ values: createDeterministicEmbedding(req.content?.parts?.[0]?.text || ""),
92
+ })),
93
+ }),
94
+ text: async () => JSON.stringify({ embeddings: [] }),
95
+ };
96
+ }
97
+
98
+ return {
99
+ ok: true,
100
+ status: 200,
101
+ json: async () => ({ embedding: { values: embedding } }),
102
+ text: async () => JSON.stringify({ embedding: { values: embedding } }),
103
+ };
104
+ }
105
+
106
+ // Default: return error
107
+ return {
108
+ ok: false,
109
+ status: 404,
110
+ json: async () => ({ error: "Not found" }),
111
+ text: async () => "Not found",
112
+ };
113
+ });
114
+ }
115
+
116
+ /**
117
+ * Capture console output for testing CLI commands.
118
+ *
119
+ * Returns an object with logs and errors arrays, plus a restore function.
120
+ */
121
+ export function captureConsole() {
122
+ const logs: string[] = [];
123
+ const errors: string[] = [];
124
+ const originalLog = console.log;
125
+ const originalError = console.error;
126
+
127
+ console.log = (...args: unknown[]) => {
128
+ logs.push(args.map(a => String(a)).join(" "));
129
+ };
130
+ console.error = (...args: unknown[]) => {
131
+ errors.push(args.map(a => String(a)).join(" "));
132
+ };
133
+
134
+ return {
135
+ logs,
136
+ errors,
137
+ restore: () => {
138
+ console.log = originalLog;
139
+ console.error = originalError;
140
+ },
141
+ };
142
+ }
143
+
144
+ /**
145
+ * Create a temporary directory for test isolation.
146
+ *
147
+ * @param prefix - Directory name prefix (default: "minimem-test-")
148
+ * @returns Object with dir path and cleanup function
149
+ */
150
+ export async function createTempDir(prefix = "minimem-test-"): Promise<{
151
+ dir: string;
152
+ cleanup: () => Promise<void>;
153
+ }> {
154
+ const dir = await fs.mkdtemp(path.join(os.tmpdir(), prefix));
155
+ return {
156
+ dir,
157
+ cleanup: async () => {
158
+ await fs.rm(dir, { recursive: true, force: true });
159
+ },
160
+ };
161
+ }
162
+
163
+ /**
164
+ * Create a temporary directory with basic memory structure.
165
+ *
166
+ * Creates the directory with MEMORY.md and memory/ subdirectory.
167
+ *
168
+ * @param prefix - Directory name prefix
169
+ * @returns Object with dir path, cleanup function, and paths to common files
170
+ */
171
+ export async function createTempMemoryDir(prefix = "minimem-test-"): Promise<{
172
+ dir: string;
173
+ memoryFile: string;
174
+ memorySubdir: string;
175
+ cleanup: () => Promise<void>;
176
+ }> {
177
+ const dir = await fs.mkdtemp(path.join(os.tmpdir(), prefix));
178
+ const memoryFile = path.join(dir, "MEMORY.md");
179
+ const memorySubdir = path.join(dir, "memory");
180
+
181
+ await fs.mkdir(memorySubdir, { recursive: true });
182
+ await fs.writeFile(memoryFile, "# Memory\n\n");
183
+
184
+ return {
185
+ dir,
186
+ memoryFile,
187
+ memorySubdir,
188
+ cleanup: async () => {
189
+ await fs.rm(dir, { recursive: true, force: true });
190
+ },
191
+ };
192
+ }
193
+
194
+ /**
195
+ * Wait for a specified number of milliseconds.
196
+ */
197
+ export function sleep(ms: number): Promise<void> {
198
+ return new Promise(resolve => setTimeout(resolve, ms));
199
+ }