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,290 @@
1
+ /**
2
+ * minimem sync init - Initialize sync for a memory directory
3
+ */
4
+
5
+ import fs from "node:fs/promises";
6
+ import path from "node:path";
7
+
8
+ import {
9
+ resolveMemoryDir,
10
+ loadConfig,
11
+ saveConfig,
12
+ isInitialized,
13
+ formatPath,
14
+ getMachineId,
15
+ expandPath,
16
+ getDefaultSyncConfig,
17
+ exitWithError,
18
+ } from "../config.js";
19
+ import {
20
+ getCentralRepoPath,
21
+ initCentralRepo,
22
+ validateCentralRepo,
23
+ } from "../sync/central.js";
24
+ import {
25
+ readRegistry,
26
+ writeRegistry,
27
+ checkCollision,
28
+ addMapping,
29
+ } from "../sync/registry.js";
30
+ import { detectDirectoryType } from "../sync/detection.js";
31
+
32
+ export type SyncInitOptions = {
33
+ /** Memory directory to initialize sync for */
34
+ local?: string;
35
+ /** Path in central repo */
36
+ path?: string;
37
+ /** Use global ~/.minimem directory */
38
+ global?: boolean;
39
+ };
40
+
41
+ export type SyncInitCentralOptions = {
42
+ /** Force creation even if directory exists */
43
+ force?: boolean;
44
+ };
45
+
46
+ /**
47
+ * Initialize a central repository
48
+ */
49
+ export async function syncInitCentral(
50
+ repoPath: string,
51
+ options: SyncInitCentralOptions = {}
52
+ ): Promise<void> {
53
+ console.log(`Initializing central repository at ${formatPath(repoPath)}...`);
54
+
55
+ const result = await initCentralRepo(repoPath);
56
+
57
+ if (!result.success) {
58
+ exitWithError(result.message);
59
+ }
60
+
61
+ if (result.created) {
62
+ console.log(" Created new git repository");
63
+ console.log(" Created .gitignore");
64
+ console.log(" Created .minimem-registry.json");
65
+ console.log(" Created README.md");
66
+ } else {
67
+ console.log(" Configured existing directory");
68
+ }
69
+
70
+ console.log();
71
+ console.log("Central repository is ready!");
72
+ console.log(`Set as centralRepo in ~/.config/minimem/config.json`);
73
+ console.log();
74
+ console.log("Next steps:");
75
+ console.log(" 1. Initialize sync for a memory directory:");
76
+ console.log(" minimem sync init --path myproject/");
77
+ console.log();
78
+ console.log(" 2. Or from a different directory:");
79
+ console.log(" minimem sync init --local ~/my-memories --path mymemories/");
80
+ }
81
+
82
+ /**
83
+ * Initialize sync for a memory directory
84
+ */
85
+ export async function syncInit(options: SyncInitOptions): Promise<void> {
86
+ // Resolve the memory directory
87
+ const memoryDir = resolveMemoryDir({
88
+ dir: options.local,
89
+ global: options.global,
90
+ });
91
+
92
+ // Check if initialized
93
+ if (!(await isInitialized(memoryDir))) {
94
+ exitWithError(
95
+ `${formatPath(memoryDir)} is not initialized.`,
96
+ "Run 'minimem init' first."
97
+ );
98
+ }
99
+
100
+ // Check if --path is provided
101
+ if (!options.path) {
102
+ exitWithError(
103
+ "--path is required.",
104
+ "Example: minimem sync init --path myproject/"
105
+ );
106
+ }
107
+
108
+ const centralPath = options.path;
109
+
110
+ // Get central repo
111
+ const centralRepo = await getCentralRepoPath();
112
+ if (!centralRepo) {
113
+ exitWithError(
114
+ "No central repository configured.",
115
+ "First initialize a central repository: minimem sync init-central ~/memories-repo"
116
+ );
117
+ }
118
+
119
+ // Validate central repo
120
+ const validation = await validateCentralRepo(centralRepo);
121
+ if (!validation.valid) {
122
+ exitWithError(
123
+ `Central repository is invalid: ${validation.errors.join(", ")}`
124
+ );
125
+ }
126
+
127
+ if (validation.warnings.length > 0) {
128
+ console.log("Warnings about central repository:");
129
+ for (const warning of validation.warnings) {
130
+ console.log(` - ${warning}`);
131
+ }
132
+ console.log();
133
+ }
134
+
135
+ // Get machine ID
136
+ const machineId = await getMachineId();
137
+
138
+ // Check for collisions
139
+ const registry = await readRegistry(centralRepo);
140
+ const collisionResult = checkCollision(registry, centralPath, memoryDir, machineId);
141
+
142
+ if (collisionResult === "collision") {
143
+ const existingMapping = registry.mappings.find(
144
+ (m) => m.path === centralPath || m.path === `${centralPath}/`
145
+ );
146
+ let details = `Path '${centralPath}' is already mapped by another machine.`;
147
+ if (existingMapping) {
148
+ details += ` Machine: ${existingMapping.machineId}, Local: ${existingMapping.localPath}`;
149
+ }
150
+ exitWithError(details, "Choose a different path or remove the existing mapping.");
151
+ }
152
+
153
+ // Detect directory type
154
+ const dirType = await detectDirectoryType(memoryDir);
155
+
156
+ console.log(`Initializing sync for ${formatPath(memoryDir)}...`);
157
+ console.log(` Directory type: ${dirType}`);
158
+ console.log(` Central path: ${centralPath}`);
159
+ console.log(` Machine ID: ${machineId}`);
160
+ console.log();
161
+
162
+ // Update local config
163
+ const localConfig = await loadConfig(memoryDir);
164
+ const syncDefaults = getDefaultSyncConfig();
165
+
166
+ localConfig.sync = {
167
+ enabled: true,
168
+ path: centralPath,
169
+ include: localConfig.sync?.include ?? syncDefaults.include,
170
+ exclude: localConfig.sync?.exclude ?? syncDefaults.exclude,
171
+ };
172
+
173
+ await saveConfig(memoryDir, localConfig);
174
+ console.log(" Updated .minimem/config.json with sync settings");
175
+
176
+ // Update registry
177
+ const updatedRegistry = addMapping(registry, {
178
+ path: centralPath,
179
+ localPath: memoryDir,
180
+ machineId,
181
+ lastSync: new Date().toISOString(),
182
+ });
183
+
184
+ await writeRegistry(centralRepo, updatedRegistry);
185
+ console.log(" Registered mapping in central repository");
186
+
187
+ // Create central path directory if it doesn't exist
188
+ const centralDir = path.join(centralRepo, centralPath);
189
+ try {
190
+ await fs.mkdir(centralDir, { recursive: true });
191
+ } catch {
192
+ // Directory might already exist
193
+ }
194
+
195
+ console.log();
196
+ console.log("Sync initialized successfully!");
197
+ console.log();
198
+ console.log("Next steps:");
199
+ console.log(" Push local files to central:");
200
+ console.log(" minimem push");
201
+ console.log();
202
+ console.log(" Or pull from central:");
203
+ console.log(" minimem pull");
204
+ }
205
+
206
+ /**
207
+ * List all sync mappings
208
+ */
209
+ export async function syncList(): Promise<void> {
210
+ const centralRepo = await getCentralRepoPath();
211
+
212
+ if (!centralRepo) {
213
+ console.log("No central repository configured.");
214
+ console.log("Run: minimem sync init-central <path>");
215
+ return;
216
+ }
217
+
218
+ const machineId = await getMachineId();
219
+ const registry = await readRegistry(centralRepo);
220
+
221
+ console.log(`Central Repository: ${formatPath(centralRepo)}`);
222
+ console.log(`Machine ID: ${machineId}`);
223
+ console.log();
224
+
225
+ if (registry.mappings.length === 0) {
226
+ console.log("No sync mappings registered.");
227
+ console.log("Run: minimem sync init --path <name>/");
228
+ return;
229
+ }
230
+
231
+ console.log("Registered Mappings:");
232
+ console.log("-".repeat(80));
233
+
234
+ for (const mapping of registry.mappings) {
235
+ const isCurrentMachine = mapping.machineId === machineId;
236
+ const marker = isCurrentMachine ? " *" : " ";
237
+ const lastSync = new Date(mapping.lastSync).toLocaleString();
238
+
239
+ console.log(`${marker}Path: ${mapping.path}`);
240
+ console.log(` Local: ${mapping.localPath}`);
241
+ console.log(` Machine: ${mapping.machineId}`);
242
+ console.log(` Last Sync: ${lastSync}`);
243
+ console.log();
244
+ }
245
+
246
+ console.log("* = this machine");
247
+ }
248
+
249
+ /**
250
+ * Remove sync mapping for a directory
251
+ */
252
+ export async function syncRemove(options: { local?: string; global?: boolean }): Promise<void> {
253
+ const memoryDir = resolveMemoryDir({
254
+ dir: options.local,
255
+ global: options.global,
256
+ });
257
+
258
+ const centralRepo = await getCentralRepoPath();
259
+ if (!centralRepo) {
260
+ exitWithError("No central repository configured.");
261
+ }
262
+
263
+ const localConfig = await loadConfig(memoryDir);
264
+ if (!localConfig.sync?.path) {
265
+ exitWithError(`${formatPath(memoryDir)} is not configured for sync.`);
266
+ }
267
+
268
+ const centralPath = localConfig.sync.path;
269
+ const machineId = await getMachineId();
270
+
271
+ // Update local config
272
+ delete localConfig.sync;
273
+ await saveConfig(memoryDir, localConfig);
274
+ console.log(`Removed sync config from ${formatPath(memoryDir)}`);
275
+
276
+ // Update registry
277
+ const registry = await readRegistry(centralRepo);
278
+ const updatedRegistry = {
279
+ ...registry,
280
+ mappings: registry.mappings.filter(
281
+ (m) => !(m.path === centralPath && m.machineId === machineId)
282
+ ),
283
+ };
284
+ await writeRegistry(centralRepo, updatedRegistry);
285
+ console.log(`Removed mapping from central registry`);
286
+
287
+ console.log();
288
+ console.log("Note: Files in the central repository were NOT deleted.");
289
+ console.log(`They remain at: ${formatPath(path.join(centralRepo, centralPath))}`);
290
+ }
@@ -0,0 +1,70 @@
1
+ /**
2
+ * minimem sync - Force re-index memory files
3
+ */
4
+
5
+ import { Minimem } from "../../minimem.js";
6
+ import {
7
+ resolveMemoryDir,
8
+ exitWithError,
9
+ loadConfig,
10
+ buildMinimemConfig,
11
+ isInitialized,
12
+ formatPath,
13
+ } from "../config.js";
14
+
15
+ export type SyncOptions = {
16
+ dir?: string;
17
+ global?: boolean;
18
+ force?: boolean;
19
+ provider?: string;
20
+ };
21
+
22
+ export async function sync(options: SyncOptions): Promise<void> {
23
+ const memoryDir = resolveMemoryDir({ dir: options.dir, global: options.global });
24
+
25
+ // Check if initialized
26
+ if (!(await isInitialized(memoryDir))) {
27
+ exitWithError(
28
+ `${formatPath(memoryDir)} is not initialized.`,
29
+ `Run: minimem init${options.dir ? ` ${options.dir}` : ""}`
30
+ );
31
+ }
32
+
33
+ console.log(`Syncing ${formatPath(memoryDir)}...`);
34
+
35
+ // Load config and create Minimem instance
36
+ const cliConfig = await loadConfig(memoryDir);
37
+ const config = buildMinimemConfig(memoryDir, cliConfig, {
38
+ provider: options.provider,
39
+ watch: false,
40
+ });
41
+
42
+ // Add debug logging
43
+ config.debug = (message, data) => {
44
+ if (data) {
45
+ console.log(` ${message}`, data);
46
+ } else {
47
+ console.log(` ${message}`);
48
+ }
49
+ };
50
+
51
+ let minimem: Minimem | null = null;
52
+
53
+ try {
54
+ minimem = await Minimem.create(config);
55
+
56
+ const startTime = Date.now();
57
+ await minimem.sync({ force: options.force });
58
+ const duration = Date.now() - startTime;
59
+
60
+ const status = await minimem.status();
61
+
62
+ console.log();
63
+ console.log(`Sync complete in ${duration}ms`);
64
+ console.log(` Files: ${status.fileCount}`);
65
+ console.log(` Chunks: ${status.chunkCount}`);
66
+ console.log(` Cache: ${status.cacheCount} embeddings`);
67
+ } finally {
68
+ minimem?.close();
69
+ }
70
+ }
@@ -0,0 +1,197 @@
1
+ /**
2
+ * minimem upsert - Create or update a memory file
3
+ *
4
+ * File paths are relative to the memory directory:
5
+ * minimem upsert MEMORY.md "content" → {memoryDir}/MEMORY.md
6
+ * minimem upsert memory/notes.md "content" → {memoryDir}/memory/notes.md
7
+ * minimem upsert /abs/path.md "content" → /abs/path.md
8
+ */
9
+
10
+ import * as fs from "node:fs/promises";
11
+ import * as path from "node:path";
12
+ import { Minimem } from "../../minimem.js";
13
+ import {
14
+ resolveMemoryDir,
15
+ exitWithError,
16
+ note,
17
+ loadConfig,
18
+ buildMinimemConfig,
19
+ isInitialized,
20
+ formatPath,
21
+ } from "../config.js";
22
+ import {
23
+ addFrontmatter,
24
+ parseFrontmatter,
25
+ type SessionContext,
26
+ } from "../../session.js";
27
+
28
+ export type UpsertOptions = {
29
+ dir?: string;
30
+ global?: boolean;
31
+ provider?: string;
32
+ stdin?: boolean;
33
+ session?: string;
34
+ sessionSource?: string;
35
+ };
36
+
37
+ export async function upsert(
38
+ file: string,
39
+ content: string | undefined,
40
+ options: UpsertOptions,
41
+ ): Promise<void> {
42
+ const memoryDir = resolveMemoryDir({ dir: options.dir, global: options.global });
43
+
44
+ // Check if initialized
45
+ if (!(await isInitialized(memoryDir))) {
46
+ exitWithError(
47
+ `${formatPath(memoryDir)} is not initialized.`,
48
+ `Run: minimem init${options.dir ? ` ${options.dir}` : ""}`
49
+ );
50
+ }
51
+
52
+ // Get content from stdin if --stdin flag is set
53
+ let finalContent = content;
54
+ if (options.stdin) {
55
+ finalContent = await readStdin();
56
+ }
57
+
58
+ if (!finalContent) {
59
+ exitWithError(
60
+ "No content provided.",
61
+ "Use --stdin or provide content as argument."
62
+ );
63
+ }
64
+
65
+ // Build session context from explicit options
66
+ const session: SessionContext | undefined = options.session
67
+ ? {
68
+ id: options.session,
69
+ source: options.sessionSource,
70
+ project: process.cwd(),
71
+ }
72
+ : undefined;
73
+
74
+ // Resolve file path - all relative paths are relative to memoryDir
75
+ const filePath = resolveFilePath(file, memoryDir);
76
+
77
+ // Ensure the file is within the memory directory
78
+ const resolvedPath = path.resolve(filePath);
79
+ const resolvedMemoryDir = path.resolve(memoryDir);
80
+ if (!resolvedPath.startsWith(resolvedMemoryDir + path.sep) && resolvedPath !== resolvedMemoryDir) {
81
+ exitWithError(
82
+ "File path must be within the memory directory.",
83
+ `Memory dir: ${formatPath(memoryDir)}, File: ${formatPath(filePath)}`
84
+ );
85
+ }
86
+
87
+ // Ensure parent directory exists
88
+ const parentDir = path.dirname(filePath);
89
+ await fs.mkdir(parentDir, { recursive: true });
90
+
91
+ // Check if file exists (for reporting)
92
+ let isUpdate = false;
93
+ let existingContent: string | undefined;
94
+ try {
95
+ await fs.access(filePath);
96
+ isUpdate = true;
97
+ existingContent = await fs.readFile(filePath, "utf-8");
98
+ } catch {
99
+ // File doesn't exist, this is a create
100
+ }
101
+
102
+ // Add session frontmatter if session context exists and file is markdown
103
+ let contentToWrite = finalContent;
104
+ if (session && filePath.endsWith(".md")) {
105
+ if (isUpdate && existingContent) {
106
+ // For updates, preserve existing frontmatter but update session
107
+ const { frontmatter: existing, body } = parseFrontmatter(existingContent);
108
+ // Check if the new content already has frontmatter
109
+ const { frontmatter: newFm, body: newBody } = parseFrontmatter(finalContent);
110
+ if (newFm) {
111
+ // New content has frontmatter, merge with existing
112
+ contentToWrite = addFrontmatter(newBody, {
113
+ ...existing,
114
+ ...newFm,
115
+ session: { ...existing?.session, ...newFm.session, ...session },
116
+ });
117
+ } else {
118
+ // New content has no frontmatter, add session to new content
119
+ contentToWrite = addFrontmatter(finalContent, {
120
+ ...existing,
121
+ session: { ...existing?.session, ...session },
122
+ });
123
+ }
124
+ } else {
125
+ // New file - add frontmatter with session
126
+ const { frontmatter: existingFm, body } = parseFrontmatter(finalContent);
127
+ contentToWrite = addFrontmatter(body, {
128
+ ...existingFm,
129
+ session,
130
+ });
131
+ }
132
+ }
133
+
134
+ // Write content to file
135
+ await fs.writeFile(filePath, contentToWrite, "utf-8");
136
+
137
+ const relativePath = path.relative(memoryDir, filePath);
138
+ const action = isUpdate ? "Updated" : "Created";
139
+ console.log(`${action}: ${relativePath}`);
140
+ console.log(` in ${formatPath(memoryDir)}`);
141
+ if (options.session) {
142
+ console.log(` Session: ${options.session}`);
143
+ }
144
+
145
+ // Try to sync the index (requires embedding provider)
146
+ let minimem: Minimem | null = null;
147
+
148
+ try {
149
+ const cliConfig = await loadConfig(memoryDir);
150
+ const config = buildMinimemConfig(memoryDir, cliConfig, {
151
+ provider: options.provider,
152
+ watch: false,
153
+ });
154
+
155
+ minimem = await Minimem.create(config);
156
+ await minimem.sync();
157
+ console.log(" Index synced.");
158
+ } catch {
159
+ // Sync failed, but file was written successfully
160
+ note("Index not synced (run 'minimem sync' with API key to index).");
161
+ } finally {
162
+ minimem?.close();
163
+ }
164
+ }
165
+
166
+ /**
167
+ * Resolve file path to an absolute path.
168
+ *
169
+ * Simple rules:
170
+ * - Absolute paths are used as-is
171
+ * - Relative paths are relative to memoryDir
172
+ *
173
+ * Examples:
174
+ * MEMORY.md → {memoryDir}/MEMORY.md
175
+ * memory/notes.md → {memoryDir}/memory/notes.md
176
+ * memory/2024/jan.md → {memoryDir}/memory/2024/jan.md
177
+ * /abs/path.md → /abs/path.md
178
+ */
179
+ function resolveFilePath(file: string, memoryDir: string): string {
180
+ if (path.isAbsolute(file)) {
181
+ return file;
182
+ }
183
+ return path.join(memoryDir, file);
184
+ }
185
+
186
+ /**
187
+ * Read content from stdin
188
+ */
189
+ async function readStdin(): Promise<string> {
190
+ const chunks: Buffer[] = [];
191
+
192
+ return new Promise((resolve, reject) => {
193
+ process.stdin.on("data", (chunk) => chunks.push(chunk));
194
+ process.stdin.on("end", () => resolve(Buffer.concat(chunks).toString("utf-8")));
195
+ process.stdin.on("error", reject);
196
+ });
197
+ }