macro-agent 0.1.1 → 0.1.2

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 (406) hide show
  1. package/.sudocode/issues.jsonl +28 -0
  2. package/.sudocode/specs.jsonl +4 -0
  3. package/CLAUDE.md +9 -3
  4. package/dist/agent/agent-manager.d.ts.map +1 -1
  5. package/dist/agent/agent-manager.js +111 -48
  6. package/dist/agent/agent-manager.js.map +1 -1
  7. package/dist/agent/types.d.ts +7 -0
  8. package/dist/agent/types.d.ts.map +1 -1
  9. package/dist/agent/types.js.map +1 -1
  10. package/dist/api/server.d.ts +5 -1
  11. package/dist/api/server.d.ts.map +1 -1
  12. package/dist/api/server.js +100 -3
  13. package/dist/api/server.js.map +1 -1
  14. package/dist/api/types.d.ts +1 -1
  15. package/dist/api/types.d.ts.map +1 -1
  16. package/dist/cli/acp.d.ts.map +1 -1
  17. package/dist/cli/acp.js +71 -1
  18. package/dist/cli/acp.js.map +1 -1
  19. package/dist/cli/index.js +5 -1
  20. package/dist/cli/index.js.map +1 -1
  21. package/dist/cli/mcp.js +27 -8
  22. package/dist/cli/mcp.js.map +1 -1
  23. package/dist/config/project-config.d.ts +13 -2
  24. package/dist/config/project-config.d.ts.map +1 -1
  25. package/dist/config/project-config.js +12 -2
  26. package/dist/config/project-config.js.map +1 -1
  27. package/dist/index.d.ts +1 -0
  28. package/dist/index.d.ts.map +1 -1
  29. package/dist/index.js +2 -0
  30. package/dist/index.js.map +1 -1
  31. package/dist/lifecycle/handlers/index.d.ts +7 -3
  32. package/dist/lifecycle/handlers/index.d.ts.map +1 -1
  33. package/dist/lifecycle/handlers/index.js +25 -8
  34. package/dist/lifecycle/handlers/index.js.map +1 -1
  35. package/dist/lifecycle/types.d.ts +2 -0
  36. package/dist/lifecycle/types.d.ts.map +1 -1
  37. package/dist/lifecycle/types.js.map +1 -1
  38. package/dist/map/adapter/extensions/index.d.ts +4 -1
  39. package/dist/map/adapter/extensions/index.d.ts.map +1 -1
  40. package/dist/map/adapter/extensions/index.js +27 -0
  41. package/dist/map/adapter/extensions/index.js.map +1 -1
  42. package/dist/map/adapter/extensions/streams.d.ts +95 -0
  43. package/dist/map/adapter/extensions/streams.d.ts.map +1 -0
  44. package/dist/map/adapter/extensions/streams.js +515 -0
  45. package/dist/map/adapter/extensions/streams.js.map +1 -0
  46. package/dist/map/adapter/index.d.ts +1 -1
  47. package/dist/map/adapter/index.d.ts.map +1 -1
  48. package/dist/map/adapter/index.js +3 -1
  49. package/dist/map/adapter/index.js.map +1 -1
  50. package/dist/map/adapter/types.d.ts +1 -1
  51. package/dist/map/adapter/types.d.ts.map +1 -1
  52. package/dist/mcp/mcp-server.d.ts +2 -0
  53. package/dist/mcp/mcp-server.d.ts.map +1 -1
  54. package/dist/mcp/mcp-server.js +12 -3
  55. package/dist/mcp/mcp-server.js.map +1 -1
  56. package/dist/mcp/tools/done.d.ts.map +1 -1
  57. package/dist/mcp/tools/done.js +18 -0
  58. package/dist/mcp/tools/done.js.map +1 -1
  59. package/dist/roles/builtin/coordinator.d.ts.map +1 -1
  60. package/dist/roles/builtin/coordinator.js +2 -1
  61. package/dist/roles/builtin/coordinator.js.map +1 -1
  62. package/dist/roles/builtin/integrator.d.ts.map +1 -1
  63. package/dist/roles/builtin/integrator.js +2 -1
  64. package/dist/roles/builtin/integrator.js.map +1 -1
  65. package/dist/roles/builtin/worker.d.ts.map +1 -1
  66. package/dist/roles/builtin/worker.js +3 -1
  67. package/dist/roles/builtin/worker.js.map +1 -1
  68. package/dist/roles/capabilities.d.ts +6 -0
  69. package/dist/roles/capabilities.d.ts.map +1 -1
  70. package/dist/roles/capabilities.js +10 -0
  71. package/dist/roles/capabilities.js.map +1 -1
  72. package/dist/roles/config-loader.d.ts +1 -1
  73. package/dist/roles/config-loader.d.ts.map +1 -1
  74. package/dist/roles/config-loader.js +3 -2
  75. package/dist/roles/config-loader.js.map +1 -1
  76. package/dist/roles/types.d.ts +3 -1
  77. package/dist/roles/types.d.ts.map +1 -1
  78. package/dist/server/combined-server.d.ts +8 -1
  79. package/dist/server/combined-server.d.ts.map +1 -1
  80. package/dist/server/combined-server.js +6 -2
  81. package/dist/server/combined-server.js.map +1 -1
  82. package/dist/store/event-store.d.ts.map +1 -1
  83. package/dist/store/event-store.js +12 -5
  84. package/dist/store/event-store.js.map +1 -1
  85. package/dist/store/instance.d.ts +1 -1
  86. package/dist/store/instance.d.ts.map +1 -1
  87. package/dist/store/instance.js +2 -2
  88. package/dist/store/instance.js.map +1 -1
  89. package/dist/store/types/agents.d.ts +5 -0
  90. package/dist/store/types/agents.d.ts.map +1 -1
  91. package/dist/task/backend/opentasks/daemon-manager.d.ts.map +1 -1
  92. package/dist/task/backend/opentasks/daemon-manager.js +1 -1
  93. package/dist/task/backend/opentasks/daemon-manager.js.map +1 -1
  94. package/dist/teams/index.d.ts +3 -1
  95. package/dist/teams/index.d.ts.map +1 -1
  96. package/dist/teams/index.js +2 -0
  97. package/dist/teams/index.js.map +1 -1
  98. package/dist/teams/seed-defaults.d.ts +20 -0
  99. package/dist/teams/seed-defaults.d.ts.map +1 -0
  100. package/dist/teams/seed-defaults.js +71 -0
  101. package/dist/teams/seed-defaults.js.map +1 -0
  102. package/dist/teams/team-loader.d.ts +6 -2
  103. package/dist/teams/team-loader.d.ts.map +1 -1
  104. package/dist/teams/team-loader.js +154 -162
  105. package/dist/teams/team-loader.js.map +1 -1
  106. package/dist/teams/team-manager.d.ts +112 -0
  107. package/dist/teams/team-manager.d.ts.map +1 -0
  108. package/dist/teams/team-manager.js +305 -0
  109. package/dist/teams/team-manager.js.map +1 -0
  110. package/dist/teams/team-runtime.d.ts +125 -19
  111. package/dist/teams/team-runtime.d.ts.map +1 -1
  112. package/dist/teams/team-runtime.js +527 -119
  113. package/dist/teams/team-runtime.js.map +1 -1
  114. package/dist/teams/types.d.ts +41 -151
  115. package/dist/teams/types.d.ts.map +1 -1
  116. package/dist/teams/types.js +2 -3
  117. package/dist/teams/types.js.map +1 -1
  118. package/docs/teams.md +73 -0
  119. package/package.json +2 -1
  120. package/references/minimem/.claude/settings.json +7 -0
  121. package/references/minimem/.sudocode/issues.jsonl +18 -0
  122. package/references/minimem/.sudocode/specs.jsonl +1 -0
  123. package/references/minimem/CLAUDE.md +310 -0
  124. package/references/minimem/README.md +562 -0
  125. package/references/minimem/claude-plugin/.claude-plugin/plugin.json +10 -0
  126. package/references/minimem/claude-plugin/.mcp.json +7 -0
  127. package/references/minimem/claude-plugin/README.md +158 -0
  128. package/references/minimem/claude-plugin/commands/recall.md +47 -0
  129. package/references/minimem/claude-plugin/commands/remember.md +41 -0
  130. package/references/minimem/claude-plugin/hooks/__tests__/hooks.test.ts +272 -0
  131. package/references/minimem/claude-plugin/hooks/hooks.json +27 -0
  132. package/references/minimem/claude-plugin/hooks/session-end.sh +86 -0
  133. package/references/minimem/claude-plugin/hooks/session-start.sh +85 -0
  134. package/references/minimem/claude-plugin/skills/memory/SKILL.md +108 -0
  135. package/references/minimem/media/banner.png +0 -0
  136. package/references/minimem/package-lock.json +5373 -0
  137. package/references/minimem/package.json +72 -0
  138. package/references/minimem/scripts/postbuild.js +35 -0
  139. package/references/minimem/src/__tests__/edge-cases.test.ts +371 -0
  140. package/references/minimem/src/__tests__/errors.test.ts +265 -0
  141. package/references/minimem/src/__tests__/helpers.ts +199 -0
  142. package/references/minimem/src/__tests__/internal.test.ts +407 -0
  143. package/references/minimem/src/__tests__/knowledge.test.ts +287 -0
  144. package/references/minimem/src/__tests__/minimem.integration.test.ts +1127 -0
  145. package/references/minimem/src/__tests__/session.test.ts +190 -0
  146. package/references/minimem/src/cli/__tests__/commands.test.ts +759 -0
  147. package/references/minimem/src/cli/commands/__tests__/conflicts.test.ts +141 -0
  148. package/references/minimem/src/cli/commands/append.ts +76 -0
  149. package/references/minimem/src/cli/commands/config.ts +262 -0
  150. package/references/minimem/src/cli/commands/conflicts.ts +413 -0
  151. package/references/minimem/src/cli/commands/daemon.ts +169 -0
  152. package/references/minimem/src/cli/commands/index.ts +12 -0
  153. package/references/minimem/src/cli/commands/init.ts +88 -0
  154. package/references/minimem/src/cli/commands/mcp.ts +177 -0
  155. package/references/minimem/src/cli/commands/push-pull.ts +213 -0
  156. package/references/minimem/src/cli/commands/search.ts +158 -0
  157. package/references/minimem/src/cli/commands/status.ts +84 -0
  158. package/references/minimem/src/cli/commands/sync-init.ts +290 -0
  159. package/references/minimem/src/cli/commands/sync.ts +70 -0
  160. package/references/minimem/src/cli/commands/upsert.ts +197 -0
  161. package/references/minimem/src/cli/config.ts +584 -0
  162. package/references/minimem/src/cli/index.ts +264 -0
  163. package/references/minimem/src/cli/shared.ts +161 -0
  164. package/references/minimem/src/cli/sync/__tests__/central.test.ts +152 -0
  165. package/references/minimem/src/cli/sync/__tests__/conflicts.test.ts +209 -0
  166. package/references/minimem/src/cli/sync/__tests__/daemon.test.ts +118 -0
  167. package/references/minimem/src/cli/sync/__tests__/detection.test.ts +207 -0
  168. package/references/minimem/src/cli/sync/__tests__/integration.test.ts +476 -0
  169. package/references/minimem/src/cli/sync/__tests__/registry.test.ts +363 -0
  170. package/references/minimem/src/cli/sync/__tests__/state.test.ts +255 -0
  171. package/references/minimem/src/cli/sync/__tests__/validation.test.ts +193 -0
  172. package/references/minimem/src/cli/sync/__tests__/watcher.test.ts +178 -0
  173. package/references/minimem/src/cli/sync/central.ts +292 -0
  174. package/references/minimem/src/cli/sync/conflicts.ts +204 -0
  175. package/references/minimem/src/cli/sync/daemon.ts +407 -0
  176. package/references/minimem/src/cli/sync/detection.ts +138 -0
  177. package/references/minimem/src/cli/sync/index.ts +107 -0
  178. package/references/minimem/src/cli/sync/operations.ts +373 -0
  179. package/references/minimem/src/cli/sync/registry.ts +279 -0
  180. package/references/minimem/src/cli/sync/state.ts +355 -0
  181. package/references/minimem/src/cli/sync/validation.ts +206 -0
  182. package/references/minimem/src/cli/sync/watcher.ts +234 -0
  183. package/references/minimem/src/cli/version.ts +34 -0
  184. package/references/minimem/src/core/index.ts +9 -0
  185. package/references/minimem/src/core/indexer.ts +628 -0
  186. package/references/minimem/src/core/searcher.ts +221 -0
  187. package/references/minimem/src/db/schema.ts +183 -0
  188. package/references/minimem/src/db/sqlite-vec.ts +24 -0
  189. package/references/minimem/src/embeddings/__tests__/embeddings.test.ts +431 -0
  190. package/references/minimem/src/embeddings/batch-gemini.ts +392 -0
  191. package/references/minimem/src/embeddings/batch-openai.ts +409 -0
  192. package/references/minimem/src/embeddings/embeddings.ts +434 -0
  193. package/references/minimem/src/index.ts +109 -0
  194. package/references/minimem/src/internal.ts +299 -0
  195. package/references/minimem/src/minimem.ts +1276 -0
  196. package/references/minimem/src/search/__tests__/hybrid.test.ts +247 -0
  197. package/references/minimem/src/search/graph.ts +234 -0
  198. package/references/minimem/src/search/hybrid.ts +151 -0
  199. package/references/minimem/src/search/search.ts +256 -0
  200. package/references/minimem/src/server/__tests__/mcp.test.ts +341 -0
  201. package/references/minimem/src/server/__tests__/tools.test.ts +364 -0
  202. package/references/minimem/src/server/mcp.ts +326 -0
  203. package/references/minimem/src/server/tools.ts +720 -0
  204. package/references/minimem/src/session.ts +460 -0
  205. package/references/minimem/tsconfig.json +19 -0
  206. package/references/minimem/tsup.config.ts +26 -0
  207. package/references/minimem/vitest.config.ts +24 -0
  208. package/references/openteams/.claude/settings.json +6 -0
  209. package/references/openteams/README.md +1 -0
  210. package/references/openteams/SKILL.md +341 -0
  211. package/references/openteams/design.md +411 -0
  212. package/references/openteams/examples/bmad-method/prompts/analyst/ROLE.md +16 -0
  213. package/references/openteams/examples/bmad-method/prompts/analyst/SOUL.md +5 -0
  214. package/references/openteams/examples/bmad-method/prompts/architect/ROLE.md +24 -0
  215. package/references/openteams/examples/bmad-method/prompts/architect/SOUL.md +5 -0
  216. package/references/openteams/examples/bmad-method/prompts/developer/ROLE.md +25 -0
  217. package/references/openteams/examples/bmad-method/prompts/developer/SOUL.md +5 -0
  218. package/references/openteams/examples/bmad-method/prompts/master/ROLE.md +21 -0
  219. package/references/openteams/examples/bmad-method/prompts/master/SOUL.md +5 -0
  220. package/references/openteams/examples/bmad-method/prompts/pm/ROLE.md +20 -0
  221. package/references/openteams/examples/bmad-method/prompts/pm/SOUL.md +5 -0
  222. package/references/openteams/examples/bmad-method/prompts/qa/ROLE.md +17 -0
  223. package/references/openteams/examples/bmad-method/prompts/qa/SOUL.md +5 -0
  224. package/references/openteams/examples/bmad-method/prompts/quick-flow-dev/ROLE.md +23 -0
  225. package/references/openteams/examples/bmad-method/prompts/quick-flow-dev/SOUL.md +5 -0
  226. package/references/openteams/examples/bmad-method/prompts/scrum-master/ROLE.md +27 -0
  227. package/references/openteams/examples/bmad-method/prompts/scrum-master/SOUL.md +5 -0
  228. package/references/openteams/examples/bmad-method/prompts/tech-writer/ROLE.md +21 -0
  229. package/references/openteams/examples/bmad-method/prompts/tech-writer/SOUL.md +5 -0
  230. package/references/openteams/examples/bmad-method/prompts/ux-designer/ROLE.md +16 -0
  231. package/references/openteams/examples/bmad-method/prompts/ux-designer/SOUL.md +5 -0
  232. package/references/openteams/examples/bmad-method/roles/analyst.yaml +9 -0
  233. package/references/openteams/examples/bmad-method/roles/architect.yaml +9 -0
  234. package/references/openteams/examples/bmad-method/roles/developer.yaml +8 -0
  235. package/references/openteams/examples/bmad-method/roles/master.yaml +8 -0
  236. package/references/openteams/examples/bmad-method/roles/pm.yaml +9 -0
  237. package/references/openteams/examples/bmad-method/roles/qa.yaml +8 -0
  238. package/references/openteams/examples/bmad-method/roles/quick-flow-dev.yaml +8 -0
  239. package/references/openteams/examples/bmad-method/roles/scrum-master.yaml +9 -0
  240. package/references/openteams/examples/bmad-method/roles/tech-writer.yaml +8 -0
  241. package/references/openteams/examples/bmad-method/roles/ux-designer.yaml +8 -0
  242. package/references/openteams/examples/bmad-method/team.yaml +161 -0
  243. package/references/openteams/examples/get-shit-done/prompts/codebase-mapper/ROLE.md +17 -0
  244. package/references/openteams/examples/get-shit-done/prompts/codebase-mapper/SOUL.md +5 -0
  245. package/references/openteams/examples/get-shit-done/prompts/debugger/ROLE.md +25 -0
  246. package/references/openteams/examples/get-shit-done/prompts/debugger/SOUL.md +5 -0
  247. package/references/openteams/examples/get-shit-done/prompts/executor/ROLE.md +34 -0
  248. package/references/openteams/examples/get-shit-done/prompts/executor/SOUL.md +5 -0
  249. package/references/openteams/examples/get-shit-done/prompts/integration-checker/ROLE.md +18 -0
  250. package/references/openteams/examples/get-shit-done/prompts/integration-checker/SOUL.md +3 -0
  251. package/references/openteams/examples/get-shit-done/prompts/orchestrator/ROLE.md +42 -0
  252. package/references/openteams/examples/get-shit-done/prompts/orchestrator/SOUL.md +5 -0
  253. package/references/openteams/examples/get-shit-done/prompts/phase-researcher/ROLE.md +15 -0
  254. package/references/openteams/examples/get-shit-done/prompts/phase-researcher/SOUL.md +3 -0
  255. package/references/openteams/examples/get-shit-done/prompts/plan-checker/ROLE.md +17 -0
  256. package/references/openteams/examples/get-shit-done/prompts/plan-checker/SOUL.md +3 -0
  257. package/references/openteams/examples/get-shit-done/prompts/planner/ROLE.md +28 -0
  258. package/references/openteams/examples/get-shit-done/prompts/planner/SOUL.md +5 -0
  259. package/references/openteams/examples/get-shit-done/prompts/project-researcher/ROLE.md +16 -0
  260. package/references/openteams/examples/get-shit-done/prompts/project-researcher/SOUL.md +3 -0
  261. package/references/openteams/examples/get-shit-done/prompts/research-synthesizer/ROLE.md +13 -0
  262. package/references/openteams/examples/get-shit-done/prompts/research-synthesizer/SOUL.md +3 -0
  263. package/references/openteams/examples/get-shit-done/prompts/roadmapper/ROLE.md +14 -0
  264. package/references/openteams/examples/get-shit-done/prompts/roadmapper/SOUL.md +3 -0
  265. package/references/openteams/examples/get-shit-done/prompts/verifier/ROLE.md +19 -0
  266. package/references/openteams/examples/get-shit-done/prompts/verifier/SOUL.md +5 -0
  267. package/references/openteams/examples/get-shit-done/roles/codebase-mapper.yaml +8 -0
  268. package/references/openteams/examples/get-shit-done/roles/debugger.yaml +8 -0
  269. package/references/openteams/examples/get-shit-done/roles/executor.yaml +8 -0
  270. package/references/openteams/examples/get-shit-done/roles/integration-checker.yaml +8 -0
  271. package/references/openteams/examples/get-shit-done/roles/orchestrator.yaml +9 -0
  272. package/references/openteams/examples/get-shit-done/roles/phase-researcher.yaml +7 -0
  273. package/references/openteams/examples/get-shit-done/roles/plan-checker.yaml +8 -0
  274. package/references/openteams/examples/get-shit-done/roles/planner.yaml +8 -0
  275. package/references/openteams/examples/get-shit-done/roles/project-researcher.yaml +8 -0
  276. package/references/openteams/examples/get-shit-done/roles/research-synthesizer.yaml +7 -0
  277. package/references/openteams/examples/get-shit-done/roles/roadmapper.yaml +7 -0
  278. package/references/openteams/examples/get-shit-done/roles/verifier.yaml +8 -0
  279. package/references/openteams/examples/get-shit-done/team.yaml +154 -0
  280. package/references/openteams/package-lock.json +2181 -0
  281. package/references/openteams/package.json +48 -0
  282. package/references/openteams/schema/role.schema.json +125 -0
  283. package/references/openteams/schema/team.schema.json +284 -0
  284. package/references/openteams/src/cli/agent.ts +104 -0
  285. package/references/openteams/src/cli/cli.test.ts +381 -0
  286. package/references/openteams/src/cli/generate.ts +220 -0
  287. package/references/openteams/src/cli/message.ts +241 -0
  288. package/references/openteams/src/cli/task.ts +154 -0
  289. package/references/openteams/src/cli/team.ts +104 -0
  290. package/references/openteams/src/cli/template.ts +207 -0
  291. package/references/openteams/src/cli.ts +45 -0
  292. package/references/openteams/src/db/database.test.ts +185 -0
  293. package/references/openteams/src/db/database.ts +240 -0
  294. package/references/openteams/src/generators/agent-prompt-generator.test.ts +332 -0
  295. package/references/openteams/src/generators/agent-prompt-generator.ts +521 -0
  296. package/references/openteams/src/generators/package-generator.test.ts +129 -0
  297. package/references/openteams/src/generators/package-generator.ts +102 -0
  298. package/references/openteams/src/generators/skill-generator.test.ts +246 -0
  299. package/references/openteams/src/generators/skill-generator.ts +374 -0
  300. package/references/openteams/src/index.ts +104 -0
  301. package/references/openteams/src/services/agent-service.test.ts +158 -0
  302. package/references/openteams/src/services/agent-service.ts +84 -0
  303. package/references/openteams/src/services/communication-service.test.ts +455 -0
  304. package/references/openteams/src/services/communication-service.ts +371 -0
  305. package/references/openteams/src/services/message-service.test.ts +342 -0
  306. package/references/openteams/src/services/message-service.ts +203 -0
  307. package/references/openteams/src/services/task-service.test.ts +434 -0
  308. package/references/openteams/src/services/task-service.ts +239 -0
  309. package/references/openteams/src/services/team-service.test.ts +181 -0
  310. package/references/openteams/src/services/team-service.ts +139 -0
  311. package/references/openteams/src/services/template-service.test.ts +306 -0
  312. package/references/openteams/src/services/template-service.ts +182 -0
  313. package/references/openteams/src/spawner/acp-factory.ts +96 -0
  314. package/references/openteams/src/spawner/interface.ts +31 -0
  315. package/references/openteams/src/spawner/mock.test.ts +93 -0
  316. package/references/openteams/src/spawner/mock.ts +59 -0
  317. package/references/openteams/src/template/loader.test.ts +1319 -0
  318. package/references/openteams/src/template/loader.ts +698 -0
  319. package/references/openteams/src/template/types.ts +200 -0
  320. package/references/openteams/src/types.ts +205 -0
  321. package/references/openteams/tsconfig.json +18 -0
  322. package/references/openteams/vitest.config.ts +9 -0
  323. package/references/skill-tree/.claude/settings.json +6 -0
  324. package/references/skill-tree/.sudocode/issues.jsonl +11 -0
  325. package/references/skill-tree/.sudocode/specs.jsonl +1 -0
  326. package/references/skill-tree/CLAUDE.md +150 -0
  327. package/references/skill-tree/README.md +324 -0
  328. package/references/skill-tree/docs/GAPS_v1.md +221 -0
  329. package/references/skill-tree/docs/INTEGRATION_PLAN.md +467 -0
  330. package/references/skill-tree/docs/TODOS.md +91 -0
  331. package/references/skill-tree/docs/anthropic_skill_guide.md +1364 -0
  332. package/references/skill-tree/docs/design/federated-skill-trees.md +524 -0
  333. package/references/skill-tree/docs/design/multi-agent-sync.md +759 -0
  334. package/references/skill-tree/docs/scraper/BRAINSTORM.md +583 -0
  335. package/references/skill-tree/docs/scraper/POC_PLAN.md +420 -0
  336. package/references/skill-tree/docs/scraper/README.md +170 -0
  337. package/references/skill-tree/examples/basic-usage.ts +190 -0
  338. package/references/skill-tree/package-lock.json +1509 -0
  339. package/references/skill-tree/package.json +66 -0
  340. package/references/skill-tree/scraper/README.md +123 -0
  341. package/references/skill-tree/scraper/docs/DESIGN.md +683 -0
  342. package/references/skill-tree/scraper/docs/PLAN.md +336 -0
  343. package/references/skill-tree/scraper/drizzle.config.ts +10 -0
  344. package/references/skill-tree/scraper/package-lock.json +6329 -0
  345. package/references/skill-tree/scraper/package.json +68 -0
  346. package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-description.md +7 -0
  347. package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-name.md +7 -0
  348. package/references/skill-tree/scraper/test/fixtures/minimal-skill/SKILL.md +27 -0
  349. package/references/skill-tree/scraper/test/fixtures/skill-json/SKILL.json +21 -0
  350. package/references/skill-tree/scraper/test/fixtures/skill-with-meta/SKILL.md +54 -0
  351. package/references/skill-tree/scraper/test/fixtures/skill-with-meta/_meta.json +24 -0
  352. package/references/skill-tree/scraper/test/fixtures/valid-skill/SKILL.md +93 -0
  353. package/references/skill-tree/scraper/test/fixtures/valid-skill/_meta.json +22 -0
  354. package/references/skill-tree/scraper/tsup.config.ts +14 -0
  355. package/references/skill-tree/scraper/vitest.config.ts +17 -0
  356. package/references/skill-tree/scripts/convert-to-vitest.ts +166 -0
  357. package/references/skill-tree/skills/skill-writer/SKILL.md +339 -0
  358. package/references/skill-tree/skills/skill-writer/references/examples.md +326 -0
  359. package/references/skill-tree/skills/skill-writer/references/patterns.md +210 -0
  360. package/references/skill-tree/skills/skill-writer/references/quality-checklist.md +123 -0
  361. package/references/skill-tree/test/run-all.ts +106 -0
  362. package/references/skill-tree/test/utils.ts +128 -0
  363. package/references/skill-tree/vitest.config.ts +16 -0
  364. package/src/agent/agent-manager.ts +143 -72
  365. package/src/agent/types.ts +9 -0
  366. package/src/api/__tests__/server.test.ts +203 -4
  367. package/src/api/server.ts +130 -5
  368. package/src/api/types.ts +3 -1
  369. package/src/cli/acp.ts +68 -1
  370. package/src/cli/index.ts +5 -1
  371. package/src/cli/mcp.ts +27 -13
  372. package/src/config/project-config.ts +27 -3
  373. package/src/index.ts +3 -0
  374. package/src/lifecycle/__tests__/handlers.test.ts +53 -0
  375. package/src/lifecycle/handlers/index.ts +25 -8
  376. package/src/lifecycle/types.ts +3 -0
  377. package/src/map/adapter/__tests__/stream-extensions.test.ts +494 -0
  378. package/src/map/adapter/extensions/index.ts +36 -0
  379. package/src/map/adapter/extensions/streams.ts +839 -0
  380. package/src/map/adapter/index.ts +5 -0
  381. package/src/map/adapter/types.ts +8 -1
  382. package/src/mcp/mcp-server.ts +14 -3
  383. package/src/mcp/tools/done.ts +19 -0
  384. package/src/roles/builtin/coordinator.ts +2 -0
  385. package/src/roles/builtin/integrator.ts +2 -0
  386. package/src/roles/builtin/worker.ts +3 -0
  387. package/src/roles/capabilities.ts +11 -0
  388. package/src/roles/config-loader.ts +3 -2
  389. package/src/roles/types.ts +7 -0
  390. package/src/server/combined-server.ts +15 -1
  391. package/src/store/__tests__/event-store-oob.test.ts +109 -0
  392. package/src/store/event-store.ts +13 -3
  393. package/src/store/instance.ts +2 -2
  394. package/src/store/types/agents.ts +5 -0
  395. package/src/task/backend/__tests__/memory-pull-mode.test.ts +153 -0
  396. package/src/task/backend/opentasks/daemon-manager.ts +4 -1
  397. package/src/teams/CLAUDE.md +180 -0
  398. package/src/teams/__tests__/e2e/workspace-isolation.e2e.test.ts +1263 -0
  399. package/src/teams/__tests__/team-manager.test.ts +814 -0
  400. package/src/teams/__tests__/team-system.test.ts +1291 -8
  401. package/src/teams/index.ts +21 -3
  402. package/src/teams/seed-defaults.ts +79 -0
  403. package/src/teams/team-loader.ts +200 -234
  404. package/src/teams/team-manager.ts +387 -0
  405. package/src/teams/team-runtime.ts +590 -121
  406. package/src/teams/types.ts +99 -200
@@ -0,0 +1,363 @@
1
+ /**
2
+ * Tests for registry system
3
+ */
4
+
5
+ import fs from "node:fs/promises";
6
+ import os from "node:os";
7
+ import path from "node:path";
8
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
9
+
10
+ import {
11
+ createEmptyRegistry,
12
+ readRegistry,
13
+ writeRegistry,
14
+ checkCollision,
15
+ addMapping,
16
+ removeMapping,
17
+ findMapping,
18
+ getMachineMappings,
19
+ updateLastSync,
20
+ normalizeRepoPath,
21
+ compressPath,
22
+ normalizePath,
23
+ type Registry,
24
+ } from "../registry.js";
25
+
26
+ describe("Registry System", () => {
27
+ let tempDir: string;
28
+
29
+ beforeEach(async () => {
30
+ tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-registry-test-"));
31
+ });
32
+
33
+ afterEach(async () => {
34
+ await fs.rm(tempDir, { recursive: true, force: true });
35
+ });
36
+
37
+ describe("createEmptyRegistry", () => {
38
+ it("should create registry with empty mappings", () => {
39
+ const registry = createEmptyRegistry();
40
+ expect(registry.version).toBe(1);
41
+ expect(registry.mappings).toEqual([]);
42
+ });
43
+ });
44
+
45
+ describe("readRegistry / writeRegistry", () => {
46
+ it("should return empty registry when file does not exist", async () => {
47
+ const registry = await readRegistry(tempDir);
48
+ expect(registry.mappings).toEqual([]);
49
+ });
50
+
51
+ it("should read and write registry correctly", async () => {
52
+ const registry: Registry = {
53
+ version: 1,
54
+ mappings: [
55
+ {
56
+ path: "global/",
57
+ localPath: "~/.minimem",
58
+ machineId: "test-machine-1234",
59
+ lastSync: "2024-01-15T10:30:00.000Z",
60
+ },
61
+ ],
62
+ };
63
+
64
+ await writeRegistry(tempDir, registry);
65
+ const loaded = await readRegistry(tempDir);
66
+
67
+ expect(loaded.version).toBe(1);
68
+ expect(loaded.mappings.length).toBe(1);
69
+ expect(loaded.mappings[0].path).toBe("global/");
70
+ expect(loaded.mappings[0].machineId).toBe("test-machine-1234");
71
+ });
72
+
73
+ it("should handle malformed registry file", async () => {
74
+ await fs.writeFile(
75
+ path.join(tempDir, ".minimem-registry.json"),
76
+ "{ invalid json"
77
+ );
78
+
79
+ const registry = await readRegistry(tempDir);
80
+ expect(registry.mappings).toEqual([]);
81
+ });
82
+ });
83
+
84
+ describe("normalizeRepoPath", () => {
85
+ it("should add trailing slash", () => {
86
+ expect(normalizeRepoPath("global")).toBe("global/");
87
+ });
88
+
89
+ it("should preserve existing trailing slash", () => {
90
+ expect(normalizeRepoPath("global/")).toBe("global/");
91
+ });
92
+
93
+ it("should remove multiple trailing slashes", () => {
94
+ expect(normalizeRepoPath("global///")).toBe("global/");
95
+ });
96
+
97
+ it("should handle root path", () => {
98
+ expect(normalizeRepoPath("/")).toBe("/");
99
+ });
100
+ });
101
+
102
+ describe("compressPath / normalizePath", () => {
103
+ it("should compress home directory to ~", () => {
104
+ const homePath = path.join(os.homedir(), "test", "path");
105
+ const compressed = compressPath(homePath);
106
+ expect(compressed).toBe("~/test/path");
107
+ });
108
+
109
+ it("should expand ~ to home directory", () => {
110
+ const expanded = normalizePath("~/test/path");
111
+ expect(expanded).toBe(path.join(os.homedir(), "test", "path"));
112
+ });
113
+
114
+ it("should handle paths without home", () => {
115
+ const absPath = "/tmp/test/path";
116
+ expect(compressPath(absPath)).toBe(absPath);
117
+ expect(normalizePath(absPath)).toBe(absPath);
118
+ });
119
+ });
120
+
121
+ describe("checkCollision", () => {
122
+ it("should return 'none' for new path", () => {
123
+ const registry = createEmptyRegistry();
124
+ const result = checkCollision(registry, "newpath/", "/tmp/test", "machine-1");
125
+ expect(result).toBe("none");
126
+ });
127
+
128
+ it("should return 'same-machine' for existing path on same machine", () => {
129
+ const registry: Registry = {
130
+ version: 1,
131
+ mappings: [
132
+ {
133
+ path: "global/",
134
+ localPath: "~/.minimem",
135
+ machineId: "machine-1",
136
+ lastSync: "2024-01-15T10:30:00.000Z",
137
+ },
138
+ ],
139
+ };
140
+
141
+ const result = checkCollision(registry, "global/", "/other/path", "machine-1");
142
+ expect(result).toBe("same-machine");
143
+ });
144
+
145
+ it("should return 'collision' for existing path on different machine", () => {
146
+ const registry: Registry = {
147
+ version: 1,
148
+ mappings: [
149
+ {
150
+ path: "global/",
151
+ localPath: "~/.minimem",
152
+ machineId: "machine-1",
153
+ lastSync: "2024-01-15T10:30:00.000Z",
154
+ },
155
+ ],
156
+ };
157
+
158
+ const result = checkCollision(registry, "global/", "/other/path", "machine-2");
159
+ expect(result).toBe("collision");
160
+ });
161
+
162
+ it("should handle path normalization in collision check", () => {
163
+ const registry: Registry = {
164
+ version: 1,
165
+ mappings: [
166
+ {
167
+ path: "global/",
168
+ localPath: "~/.minimem",
169
+ machineId: "machine-1",
170
+ lastSync: "2024-01-15T10:30:00.000Z",
171
+ },
172
+ ],
173
+ };
174
+
175
+ // Without trailing slash should still match
176
+ const result = checkCollision(registry, "global", "/other/path", "machine-2");
177
+ expect(result).toBe("collision");
178
+ });
179
+ });
180
+
181
+ describe("addMapping", () => {
182
+ it("should add new mapping", () => {
183
+ const registry = createEmptyRegistry();
184
+ const updated = addMapping(registry, {
185
+ path: "global/",
186
+ localPath: "~/.minimem",
187
+ machineId: "machine-1",
188
+ lastSync: "2024-01-15T10:30:00.000Z",
189
+ });
190
+
191
+ expect(updated.mappings.length).toBe(1);
192
+ expect(updated.mappings[0].path).toBe("global/");
193
+ });
194
+
195
+ it("should update existing mapping for same machine", () => {
196
+ const registry: Registry = {
197
+ version: 1,
198
+ mappings: [
199
+ {
200
+ path: "global/",
201
+ localPath: "~/.minimem",
202
+ machineId: "machine-1",
203
+ lastSync: "2024-01-15T10:30:00.000Z",
204
+ },
205
+ ],
206
+ };
207
+
208
+ // Use absolute path that will be compressed to ~/new-path
209
+ const newPath = path.join(os.homedir(), "new-path");
210
+ const updated = addMapping(registry, {
211
+ path: "global/",
212
+ localPath: newPath,
213
+ machineId: "machine-1",
214
+ lastSync: "2024-01-16T10:30:00.000Z",
215
+ });
216
+
217
+ expect(updated.mappings.length).toBe(1);
218
+ expect(updated.mappings[0].localPath).toBe("~/new-path");
219
+ });
220
+
221
+ it("should compress local paths", () => {
222
+ const registry = createEmptyRegistry();
223
+ const homePath = path.join(os.homedir(), "test");
224
+ const updated = addMapping(registry, {
225
+ path: "test/",
226
+ localPath: homePath,
227
+ machineId: "machine-1",
228
+ lastSync: "2024-01-15T10:30:00.000Z",
229
+ });
230
+
231
+ expect(updated.mappings[0].localPath).toBe("~/test");
232
+ });
233
+ });
234
+
235
+ describe("removeMapping", () => {
236
+ it("should remove existing mapping", () => {
237
+ const registry: Registry = {
238
+ version: 1,
239
+ mappings: [
240
+ {
241
+ path: "global/",
242
+ localPath: "~/.minimem",
243
+ machineId: "machine-1",
244
+ lastSync: "2024-01-15T10:30:00.000Z",
245
+ },
246
+ {
247
+ path: "work/",
248
+ localPath: "~/work",
249
+ machineId: "machine-1",
250
+ lastSync: "2024-01-15T10:30:00.000Z",
251
+ },
252
+ ],
253
+ };
254
+
255
+ const updated = removeMapping(registry, "global/", "machine-1");
256
+ expect(updated.mappings.length).toBe(1);
257
+ expect(updated.mappings[0].path).toBe("work/");
258
+ });
259
+
260
+ it("should not affect other machines' mappings", () => {
261
+ const registry: Registry = {
262
+ version: 1,
263
+ mappings: [
264
+ {
265
+ path: "global/",
266
+ localPath: "~/.minimem",
267
+ machineId: "machine-1",
268
+ lastSync: "2024-01-15T10:30:00.000Z",
269
+ },
270
+ {
271
+ path: "global/",
272
+ localPath: "~/.minimem",
273
+ machineId: "machine-2",
274
+ lastSync: "2024-01-15T10:30:00.000Z",
275
+ },
276
+ ],
277
+ };
278
+
279
+ const updated = removeMapping(registry, "global/", "machine-1");
280
+ expect(updated.mappings.length).toBe(1);
281
+ expect(updated.mappings[0].machineId).toBe("machine-2");
282
+ });
283
+ });
284
+
285
+ describe("findMapping", () => {
286
+ it("should find existing mapping", () => {
287
+ const registry: Registry = {
288
+ version: 1,
289
+ mappings: [
290
+ {
291
+ path: "global/",
292
+ localPath: "~/.minimem",
293
+ machineId: "machine-1",
294
+ lastSync: "2024-01-15T10:30:00.000Z",
295
+ },
296
+ ],
297
+ };
298
+
299
+ const found = findMapping(registry, "global/", "machine-1");
300
+ expect(found).toBeTruthy();
301
+ expect(found!.localPath).toBe("~/.minimem");
302
+ });
303
+
304
+ it("should return undefined for non-existent mapping", () => {
305
+ const registry = createEmptyRegistry();
306
+ const found = findMapping(registry, "global/", "machine-1");
307
+ expect(found).toBeUndefined();
308
+ });
309
+ });
310
+
311
+ describe("getMachineMappings", () => {
312
+ it("should return all mappings for a machine", () => {
313
+ const registry: Registry = {
314
+ version: 1,
315
+ mappings: [
316
+ {
317
+ path: "global/",
318
+ localPath: "~/.minimem",
319
+ machineId: "machine-1",
320
+ lastSync: "2024-01-15T10:30:00.000Z",
321
+ },
322
+ {
323
+ path: "work/",
324
+ localPath: "~/work",
325
+ machineId: "machine-1",
326
+ lastSync: "2024-01-15T10:30:00.000Z",
327
+ },
328
+ {
329
+ path: "other/",
330
+ localPath: "~/other",
331
+ machineId: "machine-2",
332
+ lastSync: "2024-01-15T10:30:00.000Z",
333
+ },
334
+ ],
335
+ };
336
+
337
+ const mappings = getMachineMappings(registry, "machine-1");
338
+ expect(mappings.length).toBe(2);
339
+ });
340
+ });
341
+
342
+ describe("updateLastSync", () => {
343
+ it("should update lastSync timestamp", () => {
344
+ const registry: Registry = {
345
+ version: 1,
346
+ mappings: [
347
+ {
348
+ path: "global/",
349
+ localPath: "~/.minimem",
350
+ machineId: "machine-1",
351
+ lastSync: "2024-01-15T10:30:00.000Z",
352
+ },
353
+ ],
354
+ };
355
+
356
+ const updated = updateLastSync(registry, "global/", "machine-1");
357
+ expect(updated.mappings[0].lastSync).not.toBe("2024-01-15T10:30:00.000Z");
358
+ // Should be a valid ISO date
359
+ const date = new Date(updated.mappings[0].lastSync);
360
+ expect(isNaN(date.getTime())).toBe(false);
361
+ });
362
+ });
363
+ });
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Tests for sync state tracking
3
+ *
4
+ * Uses simplified 2-way comparison (local vs remote).
5
+ * No lastSyncedHash - conflict resolution is last-write-wins.
6
+ */
7
+
8
+ import fs from "node:fs/promises";
9
+ import os from "node:os";
10
+ import path from "node:path";
11
+ import { afterEach, beforeEach, describe, it, expect } from "vitest";
12
+
13
+ import {
14
+ createEmptySyncState,
15
+ loadSyncState,
16
+ saveSyncState,
17
+ computeFileHash,
18
+ computeContentHash,
19
+ listSyncableFiles,
20
+ getFileSyncStatus,
21
+ updateSyncStateAfterSync,
22
+ removeFileFromSyncState,
23
+ type SyncState,
24
+ } from "../state.js";
25
+
26
+ describe("Sync State", () => {
27
+ let tempDir: string;
28
+
29
+ beforeEach(async () => {
30
+ tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-state-test-"));
31
+ });
32
+
33
+ afterEach(async () => {
34
+ await fs.rm(tempDir, { recursive: true, force: true });
35
+ });
36
+
37
+ describe("createEmptySyncState", () => {
38
+ it("should create empty state with central path", () => {
39
+ const state = createEmptySyncState("myproject/");
40
+ expect(state.version).toBe(2);
41
+ expect(state.lastSync).toBe(null);
42
+ expect(state.centralPath).toBe("myproject/");
43
+ expect(state.files).toEqual({});
44
+ });
45
+ });
46
+
47
+ describe("loadSyncState / saveSyncState", () => {
48
+ it("should return empty state when file does not exist", async () => {
49
+ const state = await loadSyncState(tempDir, "test/");
50
+ expect(state.lastSync).toBe(null);
51
+ expect(state.files).toEqual({});
52
+ });
53
+
54
+ it("should save and load state correctly", async () => {
55
+ await fs.mkdir(path.join(tempDir, ".minimem"), { recursive: true });
56
+
57
+ const state: SyncState = {
58
+ version: 2,
59
+ lastSync: "2024-01-15T10:30:00.000Z",
60
+ centralPath: "myproject/",
61
+ files: {
62
+ "MEMORY.md": {
63
+ localHash: "abc123",
64
+ remoteHash: "abc123",
65
+ lastModified: "2024-01-15T10:30:00.000Z",
66
+ },
67
+ },
68
+ };
69
+
70
+ await saveSyncState(tempDir, state);
71
+ const loaded = await loadSyncState(tempDir, "myproject/");
72
+
73
+ expect(loaded.lastSync).toBe("2024-01-15T10:30:00.000Z");
74
+ expect(loaded.files["MEMORY.md"].localHash).toBe("abc123");
75
+ });
76
+
77
+ it("should migrate v1 state to v2", async () => {
78
+ await fs.mkdir(path.join(tempDir, ".minimem"), { recursive: true });
79
+
80
+ // Write v1 state with lastSyncedHash
81
+ const v1State = {
82
+ version: 1,
83
+ lastSync: "2024-01-15T10:30:00.000Z",
84
+ centralPath: "myproject/",
85
+ files: {
86
+ "MEMORY.md": {
87
+ localHash: "abc123",
88
+ remoteHash: "abc123",
89
+ lastSyncedHash: "abc123",
90
+ lastModified: "2024-01-15T10:30:00.000Z",
91
+ },
92
+ },
93
+ };
94
+
95
+ await fs.writeFile(
96
+ path.join(tempDir, ".minimem", "sync-state.json"),
97
+ JSON.stringify(v1State, null, 2)
98
+ );
99
+
100
+ const loaded = await loadSyncState(tempDir, "myproject/");
101
+
102
+ expect(loaded.version).toBe(2);
103
+ // lastSyncedHash should be removed during migration
104
+ expect((loaded.files["MEMORY.md"] as unknown as { lastSyncedHash?: string }).lastSyncedHash).toBeUndefined();
105
+ });
106
+ });
107
+
108
+ describe("computeFileHash / computeContentHash", () => {
109
+ it("should compute consistent hash for file", async () => {
110
+ const filePath = path.join(tempDir, "test.txt");
111
+ await fs.writeFile(filePath, "Hello, World!");
112
+
113
+ const hash1 = await computeFileHash(filePath);
114
+ const hash2 = await computeFileHash(filePath);
115
+
116
+ expect(hash1).toBe(hash2);
117
+ expect(hash1.length).toBe(64); // SHA-256 hex
118
+ });
119
+
120
+ it("should compute consistent hash for content", () => {
121
+ const hash1 = computeContentHash("Hello, World!");
122
+ const hash2 = computeContentHash("Hello, World!");
123
+
124
+ expect(hash1).toBe(hash2);
125
+ });
126
+
127
+ it("should produce different hashes for different content", () => {
128
+ const hash1 = computeContentHash("Hello");
129
+ const hash2 = computeContentHash("World");
130
+
131
+ expect(hash1).not.toBe(hash2);
132
+ });
133
+ });
134
+
135
+ describe("listSyncableFiles", () => {
136
+ beforeEach(async () => {
137
+ // Create test file structure
138
+ await fs.mkdir(path.join(tempDir, "memory"), { recursive: true });
139
+ await fs.mkdir(path.join(tempDir, ".minimem"), { recursive: true });
140
+ await fs.writeFile(path.join(tempDir, "MEMORY.md"), "# Memory");
141
+ await fs.writeFile(path.join(tempDir, "memory", "notes.md"), "# Notes");
142
+ await fs.writeFile(path.join(tempDir, "memory", "draft.txt"), "Draft");
143
+ await fs.writeFile(path.join(tempDir, ".minimem", "config.json"), "{}");
144
+ await fs.writeFile(path.join(tempDir, "other.txt"), "Other");
145
+ });
146
+
147
+ it("should list files matching include patterns", async () => {
148
+ const files = await listSyncableFiles(
149
+ tempDir,
150
+ ["MEMORY.md", "memory/**/*.md"],
151
+ []
152
+ );
153
+
154
+ expect(files.includes("MEMORY.md")).toBeTruthy();
155
+ expect(files.includes("memory/notes.md")).toBeTruthy();
156
+ expect(!files.includes("memory/draft.txt")).toBeTruthy(); // .txt not matched
157
+ expect(!files.includes("other.txt")).toBeTruthy(); // not matched
158
+ });
159
+
160
+ it("should exclude .minimem directory", async () => {
161
+ const files = await listSyncableFiles(tempDir, ["**/*"], []);
162
+
163
+ expect(!files.some((f) => f.includes(".minimem"))).toBeTruthy();
164
+ });
165
+
166
+ it("should apply exclude patterns", async () => {
167
+ const files = await listSyncableFiles(
168
+ tempDir,
169
+ ["**/*.md"],
170
+ ["memory/**"]
171
+ );
172
+
173
+ expect(files.includes("MEMORY.md")).toBeTruthy();
174
+ expect(!files.includes("memory/notes.md")).toBeTruthy(); // excluded
175
+ });
176
+
177
+ it("should return sorted list", async () => {
178
+ const files = await listSyncableFiles(
179
+ tempDir,
180
+ ["MEMORY.md", "memory/**/*.md"],
181
+ []
182
+ );
183
+
184
+ const sorted = [...files].sort();
185
+ expect(files).toEqual(sorted);
186
+ });
187
+ });
188
+
189
+ describe("getFileSyncStatus", () => {
190
+ // 2-way comparison: only localHash and remoteHash
191
+
192
+ it("should return unchanged when hashes match", () => {
193
+ const status = getFileSyncStatus("abc", "abc");
194
+ expect(status).toBe("unchanged");
195
+ });
196
+
197
+ it("should return local-modified when hashes differ", () => {
198
+ const status = getFileSyncStatus("local", "remote");
199
+ expect(status).toBe("local-modified");
200
+ });
201
+
202
+ it("should return local-only for file only on local", () => {
203
+ const status = getFileSyncStatus("abc", null);
204
+ expect(status).toBe("local-only");
205
+ });
206
+
207
+ it("should return remote-only for file only on remote", () => {
208
+ const status = getFileSyncStatus(null, "abc");
209
+ expect(status).toBe("remote-only");
210
+ });
211
+
212
+ it("should return unchanged when both null", () => {
213
+ const status = getFileSyncStatus(null, null);
214
+ expect(status).toBe("unchanged");
215
+ });
216
+ });
217
+
218
+ describe("updateSyncStateAfterSync", () => {
219
+ it("should update file entry and lastSync", () => {
220
+ const state = createEmptySyncState("test/");
221
+ const updated = updateSyncStateAfterSync(state, "MEMORY.md", "newhash");
222
+
223
+ expect(updated.lastSync).toBeTruthy();
224
+ expect(updated.files["MEMORY.md"].localHash).toBe("newhash");
225
+ expect(updated.files["MEMORY.md"].remoteHash).toBe("newhash");
226
+ });
227
+ });
228
+
229
+ describe("removeFileFromSyncState", () => {
230
+ it("should remove file entry", () => {
231
+ const state: SyncState = {
232
+ version: 2,
233
+ lastSync: "2024-01-15T10:30:00.000Z",
234
+ centralPath: "test/",
235
+ files: {
236
+ "MEMORY.md": {
237
+ localHash: "abc",
238
+ remoteHash: "abc",
239
+ lastModified: "2024-01-15T10:30:00.000Z",
240
+ },
241
+ "notes.md": {
242
+ localHash: "def",
243
+ remoteHash: "def",
244
+ lastModified: "2024-01-15T10:30:00.000Z",
245
+ },
246
+ },
247
+ };
248
+
249
+ const updated = removeFileFromSyncState(state, "MEMORY.md");
250
+
251
+ expect(!("MEMORY.md" in updated.files)).toBeTruthy();
252
+ expect("notes.md" in updated.files).toBeTruthy();
253
+ });
254
+ });
255
+ });