opentasks 0.0.6 → 0.0.7

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 (660) hide show
  1. package/README.md +68 -0
  2. package/dist/__tests__/cli-tools.test.d.ts +8 -0
  3. package/dist/__tests__/cli-tools.test.d.ts.map +1 -0
  4. package/dist/__tests__/cli-tools.test.js +546 -0
  5. package/dist/__tests__/cli-tools.test.js.map +1 -0
  6. package/dist/__tests__/cli.test.d.ts +5 -0
  7. package/dist/__tests__/cli.test.d.ts.map +1 -0
  8. package/dist/__tests__/cli.test.js +77 -0
  9. package/dist/__tests__/cli.test.js.map +1 -0
  10. package/dist/__tests__/p1-p3-gaps.test.d.ts +2 -0
  11. package/dist/__tests__/p1-p3-gaps.test.d.ts.map +1 -0
  12. package/dist/__tests__/p1-p3-gaps.test.js +463 -0
  13. package/dist/__tests__/p1-p3-gaps.test.js.map +1 -0
  14. package/dist/cli.d.ts +1 -0
  15. package/dist/cli.d.ts.map +1 -1
  16. package/dist/cli.js +64 -0
  17. package/dist/cli.js.map +1 -1
  18. package/dist/client/__tests__/client-crud.test.d.ts +7 -0
  19. package/dist/client/__tests__/client-crud.test.d.ts.map +1 -0
  20. package/dist/client/__tests__/client-crud.test.js +404 -0
  21. package/dist/client/__tests__/client-crud.test.js.map +1 -0
  22. package/dist/client/__tests__/client.test.d.ts +5 -0
  23. package/dist/client/__tests__/client.test.d.ts.map +1 -0
  24. package/dist/client/__tests__/client.test.js +518 -0
  25. package/dist/client/__tests__/client.test.js.map +1 -0
  26. package/dist/client/client.d.ts +47 -1
  27. package/dist/client/client.d.ts.map +1 -1
  28. package/dist/client/client.js +71 -0
  29. package/dist/client/client.js.map +1 -1
  30. package/dist/client/index.d.ts +1 -1
  31. package/dist/client/index.d.ts.map +1 -1
  32. package/dist/config/__tests__/defaults.test.d.ts +2 -0
  33. package/dist/config/__tests__/defaults.test.d.ts.map +1 -0
  34. package/dist/config/__tests__/defaults.test.js +57 -0
  35. package/dist/config/__tests__/defaults.test.js.map +1 -0
  36. package/dist/config/__tests__/env.test.d.ts +2 -0
  37. package/dist/config/__tests__/env.test.d.ts.map +1 -0
  38. package/dist/config/__tests__/env.test.js +136 -0
  39. package/dist/config/__tests__/env.test.js.map +1 -0
  40. package/dist/config/__tests__/index.test.d.ts +2 -0
  41. package/dist/config/__tests__/index.test.d.ts.map +1 -0
  42. package/dist/config/__tests__/index.test.js +113 -0
  43. package/dist/config/__tests__/index.test.js.map +1 -0
  44. package/dist/config/__tests__/loader.test.d.ts +2 -0
  45. package/dist/config/__tests__/loader.test.d.ts.map +1 -0
  46. package/dist/config/__tests__/loader.test.js +128 -0
  47. package/dist/config/__tests__/loader.test.js.map +1 -0
  48. package/dist/config/__tests__/merge.test.d.ts +2 -0
  49. package/dist/config/__tests__/merge.test.d.ts.map +1 -0
  50. package/dist/config/__tests__/merge.test.js +79 -0
  51. package/dist/config/__tests__/merge.test.js.map +1 -0
  52. package/dist/config/__tests__/schema.test.d.ts +2 -0
  53. package/dist/config/__tests__/schema.test.d.ts.map +1 -0
  54. package/dist/config/__tests__/schema.test.js +300 -0
  55. package/dist/config/__tests__/schema.test.js.map +1 -0
  56. package/dist/config/schema.d.ts +178 -4
  57. package/dist/config/schema.d.ts.map +1 -1
  58. package/dist/config/schema.js +109 -7
  59. package/dist/config/schema.js.map +1 -1
  60. package/dist/context-files/context-files.d.ts +72 -0
  61. package/dist/context-files/context-files.d.ts.map +1 -0
  62. package/dist/context-files/context-files.js +145 -0
  63. package/dist/context-files/context-files.js.map +1 -0
  64. package/dist/context-files/index.d.ts +16 -0
  65. package/dist/context-files/index.d.ts.map +1 -0
  66. package/dist/context-files/index.js +14 -0
  67. package/dist/context-files/index.js.map +1 -0
  68. package/dist/context-files/resolver.d.ts +43 -0
  69. package/dist/context-files/resolver.d.ts.map +1 -0
  70. package/dist/context-files/resolver.js +127 -0
  71. package/dist/context-files/resolver.js.map +1 -0
  72. package/dist/context-files/types.d.ts +94 -0
  73. package/dist/context-files/types.d.ts.map +1 -0
  74. package/dist/context-files/types.js +10 -0
  75. package/dist/context-files/types.js.map +1 -0
  76. package/dist/context-files/watcher-integration.d.ts +47 -0
  77. package/dist/context-files/watcher-integration.d.ts.map +1 -0
  78. package/dist/context-files/watcher-integration.js +47 -0
  79. package/dist/context-files/watcher-integration.js.map +1 -0
  80. package/dist/core/__tests__/conditional-redirects.test.d.ts +2 -0
  81. package/dist/core/__tests__/conditional-redirects.test.d.ts.map +1 -0
  82. package/dist/core/__tests__/conditional-redirects.test.js +83 -0
  83. package/dist/core/__tests__/conditional-redirects.test.js.map +1 -0
  84. package/dist/core/__tests__/connections.test.d.ts +2 -0
  85. package/dist/core/__tests__/connections.test.d.ts.map +1 -0
  86. package/dist/core/__tests__/connections.test.js +158 -0
  87. package/dist/core/__tests__/connections.test.js.map +1 -0
  88. package/dist/core/__tests__/hash.test.d.ts +2 -0
  89. package/dist/core/__tests__/hash.test.d.ts.map +1 -0
  90. package/dist/core/__tests__/hash.test.js +139 -0
  91. package/dist/core/__tests__/hash.test.js.map +1 -0
  92. package/dist/core/__tests__/id.test.d.ts +2 -0
  93. package/dist/core/__tests__/id.test.d.ts.map +1 -0
  94. package/dist/core/__tests__/id.test.js +142 -0
  95. package/dist/core/__tests__/id.test.js.map +1 -0
  96. package/dist/core/__tests__/location.test.d.ts +2 -0
  97. package/dist/core/__tests__/location.test.d.ts.map +1 -0
  98. package/dist/core/__tests__/location.test.js +77 -0
  99. package/dist/core/__tests__/location.test.js.map +1 -0
  100. package/dist/core/__tests__/merge-driver.test.d.ts +2 -0
  101. package/dist/core/__tests__/merge-driver.test.d.ts.map +1 -0
  102. package/dist/core/__tests__/merge-driver.test.js +218 -0
  103. package/dist/core/__tests__/merge-driver.test.js.map +1 -0
  104. package/dist/core/__tests__/redirects.test.d.ts +2 -0
  105. package/dist/core/__tests__/redirects.test.d.ts.map +1 -0
  106. package/dist/core/__tests__/redirects.test.js +123 -0
  107. package/dist/core/__tests__/redirects.test.js.map +1 -0
  108. package/dist/core/__tests__/resolve-location-target.test.d.ts +8 -0
  109. package/dist/core/__tests__/resolve-location-target.test.d.ts.map +1 -0
  110. package/dist/core/__tests__/resolve-location-target.test.js +303 -0
  111. package/dist/core/__tests__/resolve-location-target.test.js.map +1 -0
  112. package/dist/core/__tests__/uri.test.d.ts +2 -0
  113. package/dist/core/__tests__/uri.test.d.ts.map +1 -0
  114. package/dist/core/__tests__/uri.test.js +159 -0
  115. package/dist/core/__tests__/uri.test.js.map +1 -0
  116. package/dist/core/__tests__/worktree.test.d.ts +2 -0
  117. package/dist/core/__tests__/worktree.test.d.ts.map +1 -0
  118. package/dist/core/__tests__/worktree.test.js +120 -0
  119. package/dist/core/__tests__/worktree.test.js.map +1 -0
  120. package/dist/core/merge-driver.d.ts +5 -0
  121. package/dist/core/merge-driver.d.ts.map +1 -1
  122. package/dist/core/merge-driver.js +35 -0
  123. package/dist/core/merge-driver.js.map +1 -1
  124. package/dist/daemon/__tests__/flush.test.d.ts +5 -0
  125. package/dist/daemon/__tests__/flush.test.d.ts.map +1 -0
  126. package/dist/daemon/__tests__/flush.test.js +213 -0
  127. package/dist/daemon/__tests__/flush.test.js.map +1 -0
  128. package/dist/daemon/__tests__/integration.test.d.ts +7 -0
  129. package/dist/daemon/__tests__/integration.test.d.ts.map +1 -0
  130. package/dist/daemon/__tests__/integration.test.js +276 -0
  131. package/dist/daemon/__tests__/integration.test.js.map +1 -0
  132. package/dist/daemon/__tests__/ipc.test.d.ts +5 -0
  133. package/dist/daemon/__tests__/ipc.test.d.ts.map +1 -0
  134. package/dist/daemon/__tests__/ipc.test.js +314 -0
  135. package/dist/daemon/__tests__/ipc.test.js.map +1 -0
  136. package/dist/daemon/__tests__/lifecycle.test.d.ts +5 -0
  137. package/dist/daemon/__tests__/lifecycle.test.d.ts.map +1 -0
  138. package/dist/daemon/__tests__/lifecycle.test.js +301 -0
  139. package/dist/daemon/__tests__/lifecycle.test.js.map +1 -0
  140. package/dist/daemon/__tests__/lock.test.d.ts +5 -0
  141. package/dist/daemon/__tests__/lock.test.d.ts.map +1 -0
  142. package/dist/daemon/__tests__/lock.test.js +192 -0
  143. package/dist/daemon/__tests__/lock.test.js.map +1 -0
  144. package/dist/daemon/__tests__/methods/graph.test.d.ts +5 -0
  145. package/dist/daemon/__tests__/methods/graph.test.d.ts.map +1 -0
  146. package/dist/daemon/__tests__/methods/graph.test.js +309 -0
  147. package/dist/daemon/__tests__/methods/graph.test.js.map +1 -0
  148. package/dist/daemon/__tests__/methods/provider.test.d.ts +7 -0
  149. package/dist/daemon/__tests__/methods/provider.test.d.ts.map +1 -0
  150. package/dist/daemon/__tests__/methods/provider.test.js +181 -0
  151. package/dist/daemon/__tests__/methods/provider.test.js.map +1 -0
  152. package/dist/daemon/__tests__/methods/tools.test.d.ts +5 -0
  153. package/dist/daemon/__tests__/methods/tools.test.d.ts.map +1 -0
  154. package/dist/daemon/__tests__/methods/tools.test.js +587 -0
  155. package/dist/daemon/__tests__/methods/tools.test.js.map +1 -0
  156. package/dist/daemon/__tests__/multi-location.test.d.ts +8 -0
  157. package/dist/daemon/__tests__/multi-location.test.d.ts.map +1 -0
  158. package/dist/daemon/__tests__/multi-location.test.js +669 -0
  159. package/dist/daemon/__tests__/multi-location.test.js.map +1 -0
  160. package/dist/daemon/__tests__/registry.test.d.ts +5 -0
  161. package/dist/daemon/__tests__/registry.test.d.ts.map +1 -0
  162. package/dist/daemon/__tests__/registry.test.js +208 -0
  163. package/dist/daemon/__tests__/registry.test.js.map +1 -0
  164. package/dist/daemon/__tests__/watcher.test.d.ts +5 -0
  165. package/dist/daemon/__tests__/watcher.test.d.ts.map +1 -0
  166. package/dist/daemon/__tests__/watcher.test.js +234 -0
  167. package/dist/daemon/__tests__/watcher.test.js.map +1 -0
  168. package/dist/daemon/index.d.ts +6 -4
  169. package/dist/daemon/index.d.ts.map +1 -1
  170. package/dist/daemon/index.js +3 -2
  171. package/dist/daemon/index.js.map +1 -1
  172. package/dist/daemon/ipc.d.ts.map +1 -1
  173. package/dist/daemon/ipc.js +8 -1
  174. package/dist/daemon/ipc.js.map +1 -1
  175. package/dist/daemon/lifecycle.d.ts.map +1 -1
  176. package/dist/daemon/lifecycle.js +138 -25
  177. package/dist/daemon/lifecycle.js.map +1 -1
  178. package/dist/daemon/location-state.d.ts +12 -6
  179. package/dist/daemon/location-state.d.ts.map +1 -1
  180. package/dist/daemon/location-state.js +50 -20
  181. package/dist/daemon/location-state.js.map +1 -1
  182. package/dist/daemon/methods/__tests__/graph.test.d.ts +5 -0
  183. package/dist/daemon/methods/__tests__/graph.test.d.ts.map +1 -0
  184. package/dist/daemon/methods/__tests__/graph.test.js +274 -0
  185. package/dist/daemon/methods/__tests__/graph.test.js.map +1 -0
  186. package/dist/daemon/methods/__tests__/provider.test.d.ts +5 -0
  187. package/dist/daemon/methods/__tests__/provider.test.d.ts.map +1 -0
  188. package/dist/daemon/methods/__tests__/provider.test.js +184 -0
  189. package/dist/daemon/methods/__tests__/provider.test.js.map +1 -0
  190. package/dist/daemon/methods/__tests__/tools.test.d.ts +5 -0
  191. package/dist/daemon/methods/__tests__/tools.test.d.ts.map +1 -0
  192. package/dist/daemon/methods/__tests__/tools.test.js +295 -0
  193. package/dist/daemon/methods/__tests__/tools.test.js.map +1 -0
  194. package/dist/daemon/methods/context-files.d.ts +14 -0
  195. package/dist/daemon/methods/context-files.d.ts.map +1 -0
  196. package/dist/daemon/methods/context-files.js +95 -0
  197. package/dist/daemon/methods/context-files.js.map +1 -0
  198. package/dist/daemon/methods/provider.d.ts.map +1 -1
  199. package/dist/daemon/methods/provider.js +15 -0
  200. package/dist/daemon/methods/provider.js.map +1 -1
  201. package/dist/daemon/methods/tools.d.ts +1 -1
  202. package/dist/daemon/methods/tools.d.ts.map +1 -1
  203. package/dist/daemon/methods/tools.js +3 -2
  204. package/dist/daemon/methods/tools.js.map +1 -1
  205. package/dist/daemon/sessionlog-linker.d.ts +71 -0
  206. package/dist/daemon/sessionlog-linker.d.ts.map +1 -0
  207. package/dist/daemon/sessionlog-linker.js +472 -0
  208. package/dist/daemon/sessionlog-linker.js.map +1 -0
  209. package/dist/daemon/sessionlog-watcher.d.ts +79 -0
  210. package/dist/daemon/sessionlog-watcher.d.ts.map +1 -0
  211. package/dist/daemon/sessionlog-watcher.js +289 -0
  212. package/dist/daemon/sessionlog-watcher.js.map +1 -0
  213. package/dist/graph/__tests__/EdgeTypeRegistry.test.d.ts +2 -0
  214. package/dist/graph/__tests__/EdgeTypeRegistry.test.d.ts.map +1 -0
  215. package/dist/graph/__tests__/EdgeTypeRegistry.test.js +212 -0
  216. package/dist/graph/__tests__/EdgeTypeRegistry.test.js.map +1 -0
  217. package/dist/graph/__tests__/FederatedGraph.test.d.ts +2 -0
  218. package/dist/graph/__tests__/FederatedGraph.test.d.ts.map +1 -0
  219. package/dist/graph/__tests__/FederatedGraph.test.js +661 -0
  220. package/dist/graph/__tests__/FederatedGraph.test.js.map +1 -0
  221. package/dist/graph/__tests__/GraphologyAdapter.test.d.ts +2 -0
  222. package/dist/graph/__tests__/GraphologyAdapter.test.d.ts.map +1 -0
  223. package/dist/graph/__tests__/GraphologyAdapter.test.js +326 -0
  224. package/dist/graph/__tests__/GraphologyAdapter.test.js.map +1 -0
  225. package/dist/graph/__tests__/HydratingFederatedGraph.test.d.ts +2 -0
  226. package/dist/graph/__tests__/HydratingFederatedGraph.test.d.ts.map +1 -0
  227. package/dist/graph/__tests__/HydratingFederatedGraph.test.js +587 -0
  228. package/dist/graph/__tests__/HydratingFederatedGraph.test.js.map +1 -0
  229. package/dist/graph/__tests__/debounce.test.d.ts +5 -0
  230. package/dist/graph/__tests__/debounce.test.d.ts.map +1 -0
  231. package/dist/graph/__tests__/debounce.test.js +195 -0
  232. package/dist/graph/__tests__/debounce.test.js.map +1 -0
  233. package/dist/graph/__tests__/edge-cases.test.d.ts +8 -0
  234. package/dist/graph/__tests__/edge-cases.test.d.ts.map +1 -0
  235. package/dist/graph/__tests__/edge-cases.test.js +472 -0
  236. package/dist/graph/__tests__/edge-cases.test.js.map +1 -0
  237. package/dist/graph/__tests__/expansion.test.d.ts +2 -0
  238. package/dist/graph/__tests__/expansion.test.d.ts.map +1 -0
  239. package/dist/graph/__tests__/expansion.test.js +105 -0
  240. package/dist/graph/__tests__/expansion.test.js.map +1 -0
  241. package/dist/graph/__tests__/provider-store.test.d.ts +5 -0
  242. package/dist/graph/__tests__/provider-store.test.d.ts.map +1 -0
  243. package/dist/graph/__tests__/provider-store.test.js +791 -0
  244. package/dist/graph/__tests__/provider-store.test.js.map +1 -0
  245. package/dist/graph/__tests__/query.test.d.ts +5 -0
  246. package/dist/graph/__tests__/query.test.d.ts.map +1 -0
  247. package/dist/graph/__tests__/query.test.js +774 -0
  248. package/dist/graph/__tests__/query.test.js.map +1 -0
  249. package/dist/graph/__tests__/store.test.d.ts +5 -0
  250. package/dist/graph/__tests__/store.test.d.ts.map +1 -0
  251. package/dist/graph/__tests__/store.test.js +489 -0
  252. package/dist/graph/__tests__/store.test.js.map +1 -0
  253. package/dist/graph/__tests__/sync.test.d.ts +5 -0
  254. package/dist/graph/__tests__/sync.test.d.ts.map +1 -0
  255. package/dist/graph/__tests__/sync.test.js +129 -0
  256. package/dist/graph/__tests__/sync.test.js.map +1 -0
  257. package/dist/graph/__tests__/validation.test.d.ts +2 -0
  258. package/dist/graph/__tests__/validation.test.d.ts.map +1 -0
  259. package/dist/graph/__tests__/validation.test.js +521 -0
  260. package/dist/graph/__tests__/validation.test.js.map +1 -0
  261. package/dist/graph/index.d.ts +1 -1
  262. package/dist/graph/index.d.ts.map +1 -1
  263. package/dist/graph/index.js.map +1 -1
  264. package/dist/graph/provider-store.d.ts +78 -4
  265. package/dist/graph/provider-store.d.ts.map +1 -1
  266. package/dist/graph/provider-store.js +579 -55
  267. package/dist/graph/provider-store.js.map +1 -1
  268. package/dist/graph/store.d.ts.map +1 -1
  269. package/dist/graph/store.js +3 -0
  270. package/dist/graph/store.js.map +1 -1
  271. package/dist/graph/types.d.ts +16 -0
  272. package/dist/graph/types.d.ts.map +1 -1
  273. package/dist/graph/types.js.map +1 -1
  274. package/dist/index.d.ts +4 -2
  275. package/dist/index.d.ts.map +1 -1
  276. package/dist/index.js +9 -3
  277. package/dist/index.js.map +1 -1
  278. package/dist/materialization/git-archive-store.js +2 -2
  279. package/dist/materialization/git-archive-store.js.map +1 -1
  280. package/dist/materialization/git-remote-store.js +2 -2
  281. package/dist/materialization/git-remote-store.js.map +1 -1
  282. package/dist/materialization/snapshot.js +4 -4
  283. package/dist/materialization/snapshot.js.map +1 -1
  284. package/dist/mcp/index.d.ts +8 -0
  285. package/dist/mcp/index.d.ts.map +1 -0
  286. package/dist/mcp/index.js +8 -0
  287. package/dist/mcp/index.js.map +1 -0
  288. package/dist/mcp/server.d.ts +39 -0
  289. package/dist/mcp/server.d.ts.map +1 -0
  290. package/dist/mcp/server.js +677 -0
  291. package/dist/mcp/server.js.map +1 -0
  292. package/dist/mcp/stdio.d.ts +14 -0
  293. package/dist/mcp/stdio.d.ts.map +1 -0
  294. package/dist/mcp/stdio.js +19 -0
  295. package/dist/mcp/stdio.js.map +1 -0
  296. package/dist/providers/__tests__/beads.test.d.ts +5 -0
  297. package/dist/providers/__tests__/beads.test.d.ts.map +1 -0
  298. package/dist/providers/__tests__/beads.test.js +591 -0
  299. package/dist/providers/__tests__/beads.test.js.map +1 -0
  300. package/dist/providers/__tests__/claude-tasks.test.d.ts +5 -0
  301. package/dist/providers/__tests__/claude-tasks.test.d.ts.map +1 -0
  302. package/dist/providers/__tests__/claude-tasks.test.js +392 -0
  303. package/dist/providers/__tests__/claude-tasks.test.js.map +1 -0
  304. package/dist/providers/__tests__/from-config.test.d.ts +5 -0
  305. package/dist/providers/__tests__/from-config.test.d.ts.map +1 -0
  306. package/dist/providers/__tests__/from-config.test.js +152 -0
  307. package/dist/providers/__tests__/from-config.test.js.map +1 -0
  308. package/dist/providers/__tests__/materialization.test.d.ts +5 -0
  309. package/dist/providers/__tests__/materialization.test.d.ts.map +1 -0
  310. package/dist/providers/__tests__/materialization.test.js +407 -0
  311. package/dist/providers/__tests__/materialization.test.js.map +1 -0
  312. package/dist/providers/__tests__/native.test.d.ts +5 -0
  313. package/dist/providers/__tests__/native.test.d.ts.map +1 -0
  314. package/dist/providers/__tests__/native.test.js +566 -0
  315. package/dist/providers/__tests__/native.test.js.map +1 -0
  316. package/dist/providers/__tests__/registry.test.d.ts +5 -0
  317. package/dist/providers/__tests__/registry.test.d.ts.map +1 -0
  318. package/dist/providers/__tests__/registry.test.js +183 -0
  319. package/dist/providers/__tests__/registry.test.js.map +1 -0
  320. package/dist/providers/beads.d.ts.map +1 -1
  321. package/dist/providers/beads.js +17 -1
  322. package/dist/providers/beads.js.map +1 -1
  323. package/dist/providers/claude-tasks-fs.d.ts +22 -0
  324. package/dist/providers/claude-tasks-fs.d.ts.map +1 -0
  325. package/dist/providers/claude-tasks-fs.js +158 -0
  326. package/dist/providers/claude-tasks-fs.js.map +1 -0
  327. package/dist/providers/claude-tasks.d.ts +13 -4
  328. package/dist/providers/claude-tasks.d.ts.map +1 -1
  329. package/dist/providers/claude-tasks.js +318 -5
  330. package/dist/providers/claude-tasks.js.map +1 -1
  331. package/dist/providers/from-config.d.ts.map +1 -1
  332. package/dist/providers/from-config.js +35 -17
  333. package/dist/providers/from-config.js.map +1 -1
  334. package/dist/providers/global.d.ts.map +1 -1
  335. package/dist/providers/global.js +5 -0
  336. package/dist/providers/global.js.map +1 -1
  337. package/dist/providers/index.d.ts +6 -4
  338. package/dist/providers/index.d.ts.map +1 -1
  339. package/dist/providers/index.js +4 -3
  340. package/dist/providers/index.js.map +1 -1
  341. package/dist/providers/map.d.ts.map +1 -1
  342. package/dist/providers/map.js +5 -0
  343. package/dist/providers/map.js.map +1 -1
  344. package/dist/providers/materialization.d.ts +23 -5
  345. package/dist/providers/materialization.d.ts.map +1 -1
  346. package/dist/providers/materialization.js +95 -4
  347. package/dist/providers/materialization.js.map +1 -1
  348. package/dist/providers/native.d.ts.map +1 -1
  349. package/dist/providers/native.js +5 -0
  350. package/dist/providers/native.js.map +1 -1
  351. package/dist/providers/sessionlog.d.ts +81 -0
  352. package/dist/providers/sessionlog.d.ts.map +1 -0
  353. package/dist/providers/sessionlog.js +478 -0
  354. package/dist/providers/sessionlog.js.map +1 -0
  355. package/dist/providers/sudocode.d.ts.map +1 -1
  356. package/dist/providers/sudocode.js +17 -1
  357. package/dist/providers/sudocode.js.map +1 -1
  358. package/dist/providers/traits/Reconcilable.d.ts +57 -0
  359. package/dist/providers/traits/Reconcilable.d.ts.map +1 -0
  360. package/dist/providers/traits/Reconcilable.js +18 -0
  361. package/dist/providers/traits/Reconcilable.js.map +1 -0
  362. package/dist/providers/traits/__tests__/RelationshipQueryable.test.d.ts +2 -0
  363. package/dist/providers/traits/__tests__/RelationshipQueryable.test.d.ts.map +1 -0
  364. package/dist/providers/traits/__tests__/RelationshipQueryable.test.js +169 -0
  365. package/dist/providers/traits/__tests__/RelationshipQueryable.test.js.map +1 -0
  366. package/dist/providers/traits/__tests__/TaskManageable.test.d.ts +2 -0
  367. package/dist/providers/traits/__tests__/TaskManageable.test.d.ts.map +1 -0
  368. package/dist/providers/traits/__tests__/TaskManageable.test.js +172 -0
  369. package/dist/providers/traits/__tests__/TaskManageable.test.js.map +1 -0
  370. package/dist/providers/traits/index.d.ts +2 -0
  371. package/dist/providers/traits/index.d.ts.map +1 -1
  372. package/dist/providers/traits/index.js +1 -0
  373. package/dist/providers/traits/index.js.map +1 -1
  374. package/dist/providers/types.d.ts +63 -0
  375. package/dist/providers/types.d.ts.map +1 -1
  376. package/dist/providers/types.js +22 -0
  377. package/dist/providers/types.js.map +1 -1
  378. package/dist/schema/__tests__/validation.test.d.ts +2 -0
  379. package/dist/schema/__tests__/validation.test.d.ts.map +1 -0
  380. package/dist/schema/__tests__/validation.test.js +241 -0
  381. package/dist/schema/__tests__/validation.test.js.map +1 -0
  382. package/dist/schema/edges.d.ts +3 -3
  383. package/dist/schema/edges.d.ts.map +1 -1
  384. package/dist/sessionlog/agent/agents/claude-code.d.ts +76 -0
  385. package/dist/sessionlog/agent/agents/claude-code.d.ts.map +1 -0
  386. package/dist/sessionlog/agent/agents/claude-code.js +759 -0
  387. package/dist/sessionlog/agent/agents/claude-code.js.map +1 -0
  388. package/dist/sessionlog/agent/agents/cursor.d.ts +35 -0
  389. package/dist/sessionlog/agent/agents/cursor.d.ts.map +1 -0
  390. package/dist/sessionlog/agent/agents/cursor.js +294 -0
  391. package/dist/sessionlog/agent/agents/cursor.js.map +1 -0
  392. package/dist/sessionlog/agent/agents/gemini-cli.d.ts +62 -0
  393. package/dist/sessionlog/agent/agents/gemini-cli.d.ts.map +1 -0
  394. package/dist/sessionlog/agent/agents/gemini-cli.js +462 -0
  395. package/dist/sessionlog/agent/agents/gemini-cli.js.map +1 -0
  396. package/dist/sessionlog/agent/agents/opencode.d.ts +100 -0
  397. package/dist/sessionlog/agent/agents/opencode.d.ts.map +1 -0
  398. package/dist/sessionlog/agent/agents/opencode.js +423 -0
  399. package/dist/sessionlog/agent/agents/opencode.js.map +1 -0
  400. package/dist/sessionlog/agent/registry.d.ts +54 -0
  401. package/dist/sessionlog/agent/registry.d.ts.map +1 -0
  402. package/dist/sessionlog/agent/registry.js +123 -0
  403. package/dist/sessionlog/agent/registry.js.map +1 -0
  404. package/dist/sessionlog/agent/session-types.d.ts +45 -0
  405. package/dist/sessionlog/agent/session-types.d.ts.map +1 -0
  406. package/dist/sessionlog/agent/session-types.js +50 -0
  407. package/dist/sessionlog/agent/session-types.js.map +1 -0
  408. package/dist/sessionlog/agent/types.d.ts +126 -0
  409. package/dist/sessionlog/agent/types.d.ts.map +1 -0
  410. package/dist/sessionlog/agent/types.js +39 -0
  411. package/dist/sessionlog/agent/types.js.map +1 -0
  412. package/dist/sessionlog/commands/clean.d.ts +40 -0
  413. package/dist/sessionlog/commands/clean.d.ts.map +1 -0
  414. package/dist/sessionlog/commands/clean.js +105 -0
  415. package/dist/sessionlog/commands/clean.js.map +1 -0
  416. package/dist/sessionlog/commands/disable.d.ts +23 -0
  417. package/dist/sessionlog/commands/disable.d.ts.map +1 -0
  418. package/dist/sessionlog/commands/disable.js +57 -0
  419. package/dist/sessionlog/commands/disable.js.map +1 -0
  420. package/dist/sessionlog/commands/doctor.d.ts +43 -0
  421. package/dist/sessionlog/commands/doctor.d.ts.map +1 -0
  422. package/dist/sessionlog/commands/doctor.js +97 -0
  423. package/dist/sessionlog/commands/doctor.js.map +1 -0
  424. package/dist/sessionlog/commands/enable.d.ts +31 -0
  425. package/dist/sessionlog/commands/enable.d.ts.map +1 -0
  426. package/dist/sessionlog/commands/enable.js +102 -0
  427. package/dist/sessionlog/commands/enable.js.map +1 -0
  428. package/dist/sessionlog/commands/explain.d.ts +69 -0
  429. package/dist/sessionlog/commands/explain.d.ts.map +1 -0
  430. package/dist/sessionlog/commands/explain.js +185 -0
  431. package/dist/sessionlog/commands/explain.js.map +1 -0
  432. package/dist/sessionlog/commands/reset.d.ts +23 -0
  433. package/dist/sessionlog/commands/reset.d.ts.map +1 -0
  434. package/dist/sessionlog/commands/reset.js +68 -0
  435. package/dist/sessionlog/commands/reset.js.map +1 -0
  436. package/dist/sessionlog/commands/resume.d.ts +42 -0
  437. package/dist/sessionlog/commands/resume.d.ts.map +1 -0
  438. package/dist/sessionlog/commands/resume.js +134 -0
  439. package/dist/sessionlog/commands/resume.js.map +1 -0
  440. package/dist/sessionlog/commands/rewind.d.ts +40 -0
  441. package/dist/sessionlog/commands/rewind.d.ts.map +1 -0
  442. package/dist/sessionlog/commands/rewind.js +155 -0
  443. package/dist/sessionlog/commands/rewind.js.map +1 -0
  444. package/dist/sessionlog/commands/status.d.ts +54 -0
  445. package/dist/sessionlog/commands/status.d.ts.map +1 -0
  446. package/dist/sessionlog/commands/status.js +95 -0
  447. package/dist/sessionlog/commands/status.js.map +1 -0
  448. package/dist/sessionlog/config.d.ts +40 -0
  449. package/dist/sessionlog/config.d.ts.map +1 -0
  450. package/dist/sessionlog/config.js +126 -0
  451. package/dist/sessionlog/config.js.map +1 -0
  452. package/dist/sessionlog/git-operations.d.ts +173 -0
  453. package/dist/sessionlog/git-operations.d.ts.map +1 -0
  454. package/dist/sessionlog/git-operations.js +399 -0
  455. package/dist/sessionlog/git-operations.js.map +1 -0
  456. package/dist/sessionlog/hooks/git-hooks.d.ts +22 -0
  457. package/dist/sessionlog/hooks/git-hooks.d.ts.map +1 -0
  458. package/dist/sessionlog/hooks/git-hooks.js +145 -0
  459. package/dist/sessionlog/hooks/git-hooks.js.map +1 -0
  460. package/dist/sessionlog/hooks/lifecycle.d.ts +21 -0
  461. package/dist/sessionlog/hooks/lifecycle.d.ts.map +1 -0
  462. package/dist/sessionlog/hooks/lifecycle.js +179 -0
  463. package/dist/sessionlog/hooks/lifecycle.js.map +1 -0
  464. package/dist/sessionlog/index.d.ts +69 -0
  465. package/dist/sessionlog/index.d.ts.map +1 -0
  466. package/dist/sessionlog/index.js +154 -0
  467. package/dist/sessionlog/index.js.map +1 -0
  468. package/dist/sessionlog/security/redaction.d.ts +35 -0
  469. package/dist/sessionlog/security/redaction.d.ts.map +1 -0
  470. package/dist/sessionlog/security/redaction.js +221 -0
  471. package/dist/sessionlog/security/redaction.js.map +1 -0
  472. package/dist/sessionlog/session/state-machine.d.ts +90 -0
  473. package/dist/sessionlog/session/state-machine.d.ts.map +1 -0
  474. package/dist/sessionlog/session/state-machine.js +347 -0
  475. package/dist/sessionlog/session/state-machine.js.map +1 -0
  476. package/dist/sessionlog/store/checkpoint-store.d.ts +47 -0
  477. package/dist/sessionlog/store/checkpoint-store.d.ts.map +1 -0
  478. package/dist/sessionlog/store/checkpoint-store.js +309 -0
  479. package/dist/sessionlog/store/checkpoint-store.js.map +1 -0
  480. package/dist/sessionlog/store/native-store.d.ts +21 -0
  481. package/dist/sessionlog/store/native-store.d.ts.map +1 -0
  482. package/dist/sessionlog/store/native-store.js +162 -0
  483. package/dist/sessionlog/store/native-store.js.map +1 -0
  484. package/dist/sessionlog/store/provider-types.d.ts +78 -0
  485. package/dist/sessionlog/store/provider-types.d.ts.map +1 -0
  486. package/dist/sessionlog/store/provider-types.js +12 -0
  487. package/dist/sessionlog/store/provider-types.js.map +1 -0
  488. package/dist/sessionlog/store/session-store.d.ts +28 -0
  489. package/dist/sessionlog/store/session-store.d.ts.map +1 -0
  490. package/dist/sessionlog/store/session-store.js +187 -0
  491. package/dist/sessionlog/store/session-store.js.map +1 -0
  492. package/dist/sessionlog/strategy/attribution.d.ts +39 -0
  493. package/dist/sessionlog/strategy/attribution.d.ts.map +1 -0
  494. package/dist/sessionlog/strategy/attribution.js +227 -0
  495. package/dist/sessionlog/strategy/attribution.js.map +1 -0
  496. package/dist/sessionlog/strategy/common.d.ts +60 -0
  497. package/dist/sessionlog/strategy/common.d.ts.map +1 -0
  498. package/dist/sessionlog/strategy/common.js +162 -0
  499. package/dist/sessionlog/strategy/common.js.map +1 -0
  500. package/dist/sessionlog/strategy/content-overlap.d.ts +33 -0
  501. package/dist/sessionlog/strategy/content-overlap.d.ts.map +1 -0
  502. package/dist/sessionlog/strategy/content-overlap.js +168 -0
  503. package/dist/sessionlog/strategy/content-overlap.js.map +1 -0
  504. package/dist/sessionlog/strategy/manual-commit.d.ts +35 -0
  505. package/dist/sessionlog/strategy/manual-commit.d.ts.map +1 -0
  506. package/dist/sessionlog/strategy/manual-commit.js +732 -0
  507. package/dist/sessionlog/strategy/manual-commit.js.map +1 -0
  508. package/dist/sessionlog/strategy/types.d.ts +163 -0
  509. package/dist/sessionlog/strategy/types.d.ts.map +1 -0
  510. package/dist/sessionlog/strategy/types.js +49 -0
  511. package/dist/sessionlog/strategy/types.js.map +1 -0
  512. package/dist/sessionlog/summarize/claude-generator.d.ts +25 -0
  513. package/dist/sessionlog/summarize/claude-generator.d.ts.map +1 -0
  514. package/dist/sessionlog/summarize/claude-generator.js +87 -0
  515. package/dist/sessionlog/summarize/claude-generator.js.map +1 -0
  516. package/dist/sessionlog/summarize/summarize.d.ts +52 -0
  517. package/dist/sessionlog/summarize/summarize.d.ts.map +1 -0
  518. package/dist/sessionlog/summarize/summarize.js +335 -0
  519. package/dist/sessionlog/summarize/summarize.js.map +1 -0
  520. package/dist/sessionlog/types.d.ts +298 -0
  521. package/dist/sessionlog/types.d.ts.map +1 -0
  522. package/dist/sessionlog/types.js +104 -0
  523. package/dist/sessionlog/types.js.map +1 -0
  524. package/dist/sessionlog/utils/chunk-files.d.ts +25 -0
  525. package/dist/sessionlog/utils/chunk-files.d.ts.map +1 -0
  526. package/dist/sessionlog/utils/chunk-files.js +47 -0
  527. package/dist/sessionlog/utils/chunk-files.js.map +1 -0
  528. package/dist/sessionlog/utils/commit-message.d.ts +11 -0
  529. package/dist/sessionlog/utils/commit-message.d.ts.map +1 -0
  530. package/dist/sessionlog/utils/commit-message.js +54 -0
  531. package/dist/sessionlog/utils/commit-message.js.map +1 -0
  532. package/dist/sessionlog/utils/detect-agent.d.ts +19 -0
  533. package/dist/sessionlog/utils/detect-agent.d.ts.map +1 -0
  534. package/dist/sessionlog/utils/detect-agent.js +34 -0
  535. package/dist/sessionlog/utils/detect-agent.js.map +1 -0
  536. package/dist/sessionlog/utils/hook-managers.d.ts +24 -0
  537. package/dist/sessionlog/utils/hook-managers.d.ts.map +1 -0
  538. package/dist/sessionlog/utils/hook-managers.js +87 -0
  539. package/dist/sessionlog/utils/hook-managers.js.map +1 -0
  540. package/dist/sessionlog/utils/ide-tags.d.ts +12 -0
  541. package/dist/sessionlog/utils/ide-tags.d.ts.map +1 -0
  542. package/dist/sessionlog/utils/ide-tags.js +30 -0
  543. package/dist/sessionlog/utils/ide-tags.js.map +1 -0
  544. package/dist/sessionlog/utils/paths.d.ts +32 -0
  545. package/dist/sessionlog/utils/paths.d.ts.map +1 -0
  546. package/dist/sessionlog/utils/paths.js +55 -0
  547. package/dist/sessionlog/utils/paths.js.map +1 -0
  548. package/dist/sessionlog/utils/preview-rewind.d.ts +23 -0
  549. package/dist/sessionlog/utils/preview-rewind.d.ts.map +1 -0
  550. package/dist/sessionlog/utils/preview-rewind.js +63 -0
  551. package/dist/sessionlog/utils/preview-rewind.js.map +1 -0
  552. package/dist/sessionlog/utils/rewind-conflict.d.ts +52 -0
  553. package/dist/sessionlog/utils/rewind-conflict.d.ts.map +1 -0
  554. package/dist/sessionlog/utils/rewind-conflict.js +79 -0
  555. package/dist/sessionlog/utils/rewind-conflict.js.map +1 -0
  556. package/dist/sessionlog/utils/shadow-branch.d.ts +53 -0
  557. package/dist/sessionlog/utils/shadow-branch.d.ts.map +1 -0
  558. package/dist/sessionlog/utils/shadow-branch.js +97 -0
  559. package/dist/sessionlog/utils/shadow-branch.js.map +1 -0
  560. package/dist/sessionlog/utils/string-utils.d.ts +24 -0
  561. package/dist/sessionlog/utils/string-utils.d.ts.map +1 -0
  562. package/dist/sessionlog/utils/string-utils.js +47 -0
  563. package/dist/sessionlog/utils/string-utils.js.map +1 -0
  564. package/dist/sessionlog/utils/todo-extract.d.ts +52 -0
  565. package/dist/sessionlog/utils/todo-extract.d.ts.map +1 -0
  566. package/dist/sessionlog/utils/todo-extract.js +167 -0
  567. package/dist/sessionlog/utils/todo-extract.js.map +1 -0
  568. package/dist/sessionlog/utils/trailers.d.ts +36 -0
  569. package/dist/sessionlog/utils/trailers.d.ts.map +1 -0
  570. package/dist/sessionlog/utils/trailers.js +149 -0
  571. package/dist/sessionlog/utils/trailers.js.map +1 -0
  572. package/dist/sessionlog/utils/transcript-parse.d.ts +57 -0
  573. package/dist/sessionlog/utils/transcript-parse.d.ts.map +1 -0
  574. package/dist/sessionlog/utils/transcript-parse.js +126 -0
  575. package/dist/sessionlog/utils/transcript-parse.js.map +1 -0
  576. package/dist/sessionlog/utils/transcript-timestamp.d.ts +22 -0
  577. package/dist/sessionlog/utils/transcript-timestamp.d.ts.map +1 -0
  578. package/dist/sessionlog/utils/transcript-timestamp.js +56 -0
  579. package/dist/sessionlog/utils/transcript-timestamp.js.map +1 -0
  580. package/dist/sessionlog/utils/tree-ops.d.ts +47 -0
  581. package/dist/sessionlog/utils/tree-ops.d.ts.map +1 -0
  582. package/dist/sessionlog/utils/tree-ops.js +145 -0
  583. package/dist/sessionlog/utils/tree-ops.js.map +1 -0
  584. package/dist/sessionlog/utils/tty.d.ts +25 -0
  585. package/dist/sessionlog/utils/tty.d.ts.map +1 -0
  586. package/dist/sessionlog/utils/tty.js +70 -0
  587. package/dist/sessionlog/utils/tty.js.map +1 -0
  588. package/dist/sessionlog/utils/validation.d.ts +31 -0
  589. package/dist/sessionlog/utils/validation.d.ts.map +1 -0
  590. package/dist/sessionlog/utils/validation.js +59 -0
  591. package/dist/sessionlog/utils/validation.js.map +1 -0
  592. package/dist/sessionlog/utils/worktree.d.ts +16 -0
  593. package/dist/sessionlog/utils/worktree.d.ts.map +1 -0
  594. package/dist/sessionlog/utils/worktree.js +50 -0
  595. package/dist/sessionlog/utils/worktree.js.map +1 -0
  596. package/dist/storage/__tests__/atomic-write.test.d.ts +5 -0
  597. package/dist/storage/__tests__/atomic-write.test.d.ts.map +1 -0
  598. package/dist/storage/__tests__/atomic-write.test.js +170 -0
  599. package/dist/storage/__tests__/atomic-write.test.js.map +1 -0
  600. package/dist/storage/__tests__/file-lock.test.d.ts +2 -0
  601. package/dist/storage/__tests__/file-lock.test.d.ts.map +1 -0
  602. package/dist/storage/__tests__/file-lock.test.js +89 -0
  603. package/dist/storage/__tests__/file-lock.test.js.map +1 -0
  604. package/dist/storage/__tests__/jsonl.test.d.ts +2 -0
  605. package/dist/storage/__tests__/jsonl.test.d.ts.map +1 -0
  606. package/dist/storage/__tests__/jsonl.test.js +228 -0
  607. package/dist/storage/__tests__/jsonl.test.js.map +1 -0
  608. package/dist/storage/__tests__/locked-writer.test.d.ts +2 -0
  609. package/dist/storage/__tests__/locked-writer.test.d.ts.map +1 -0
  610. package/dist/storage/__tests__/locked-writer.test.js +109 -0
  611. package/dist/storage/__tests__/locked-writer.test.js.map +1 -0
  612. package/dist/storage/__tests__/sqlite.test.d.ts +2 -0
  613. package/dist/storage/__tests__/sqlite.test.d.ts.map +1 -0
  614. package/dist/storage/__tests__/sqlite.test.js +470 -0
  615. package/dist/storage/__tests__/sqlite.test.js.map +1 -0
  616. package/dist/storage/sqlite-schema.d.ts.map +1 -1
  617. package/dist/storage/sqlite-schema.js +2 -0
  618. package/dist/storage/sqlite-schema.js.map +1 -1
  619. package/dist/storage/sqlite.d.ts.map +1 -1
  620. package/dist/storage/sqlite.js +18 -4
  621. package/dist/storage/sqlite.js.map +1 -1
  622. package/dist/tools/__tests__/annotate.test.d.ts +5 -0
  623. package/dist/tools/__tests__/annotate.test.d.ts.map +1 -0
  624. package/dist/tools/__tests__/annotate.test.js +314 -0
  625. package/dist/tools/__tests__/annotate.test.js.map +1 -0
  626. package/dist/tools/__tests__/link.test.d.ts +5 -0
  627. package/dist/tools/__tests__/link.test.d.ts.map +1 -0
  628. package/dist/tools/__tests__/link.test.js +245 -0
  629. package/dist/tools/__tests__/link.test.js.map +1 -0
  630. package/dist/tools/__tests__/query.test.d.ts +5 -0
  631. package/dist/tools/__tests__/query.test.d.ts.map +1 -0
  632. package/dist/tools/__tests__/query.test.js +288 -0
  633. package/dist/tools/__tests__/query.test.js.map +1 -0
  634. package/dist/tools/__tests__/task.test.d.ts +5 -0
  635. package/dist/tools/__tests__/task.test.d.ts.map +1 -0
  636. package/dist/tools/__tests__/task.test.js +178 -0
  637. package/dist/tools/__tests__/task.test.js.map +1 -0
  638. package/dist/tools/index.d.ts +1 -1
  639. package/dist/tools/index.d.ts.map +1 -1
  640. package/dist/tools/index.js.map +1 -1
  641. package/dist/tools/query.d.ts +2 -1
  642. package/dist/tools/query.d.ts.map +1 -1
  643. package/dist/tools/query.js +217 -7
  644. package/dist/tools/query.js.map +1 -1
  645. package/dist/tools/types.d.ts +57 -0
  646. package/dist/tools/types.d.ts.map +1 -1
  647. package/dist/tools/types.js.map +1 -1
  648. package/dist/tracking/claude-task-reconstructor.d.ts +41 -0
  649. package/dist/tracking/claude-task-reconstructor.d.ts.map +1 -0
  650. package/dist/tracking/claude-task-reconstructor.js +91 -0
  651. package/dist/tracking/claude-task-reconstructor.js.map +1 -0
  652. package/dist/tracking/plan-mode-tracker.d.ts +20 -0
  653. package/dist/tracking/plan-mode-tracker.d.ts.map +1 -0
  654. package/dist/tracking/plan-mode-tracker.js +35 -0
  655. package/dist/tracking/plan-mode-tracker.js.map +1 -0
  656. package/dist/tracking/transcript-extractor.d.ts +3 -3
  657. package/dist/tracking/transcript-extractor.d.ts.map +1 -1
  658. package/dist/tracking/transcript-extractor.js +1 -1
  659. package/dist/tracking/transcript-extractor.js.map +1 -1
  660. package/package.json +3 -1
@@ -0,0 +1,129 @@
1
+ /**
2
+ * Tests for Sync Manager
3
+ */
4
+ import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
5
+ import { createSyncManager } from '../sync.js';
6
+ describe('SyncManager', () => {
7
+ let storage;
8
+ let syncManager;
9
+ let flushCallback;
10
+ const config = {
11
+ debounceMs: 100,
12
+ maxDelayMs: 500,
13
+ };
14
+ beforeEach(() => {
15
+ vi.useFakeTimers();
16
+ // Create mock storage
17
+ const dirtyNodes = new Set();
18
+ storage = {
19
+ markDirty: vi.fn().mockImplementation(async (nodeId) => {
20
+ dirtyNodes.add(nodeId);
21
+ }),
22
+ getDirtyNodes: vi.fn().mockImplementation(async () => {
23
+ return Array.from(dirtyNodes);
24
+ }),
25
+ clearDirty: vi.fn().mockImplementation(async (nodeIds) => {
26
+ for (const id of nodeIds) {
27
+ dirtyNodes.delete(id);
28
+ }
29
+ }),
30
+ };
31
+ flushCallback = vi.fn().mockResolvedValue(undefined);
32
+ syncManager = createSyncManager(config, storage, flushCallback);
33
+ });
34
+ afterEach(() => {
35
+ vi.useRealTimers();
36
+ syncManager.cancel();
37
+ });
38
+ describe('markDirty', () => {
39
+ it('should track dirty nodes', () => {
40
+ syncManager.markDirty('s-abc1');
41
+ syncManager.markDirty('s-abc2');
42
+ expect(syncManager.hasPendingChanges()).toBe(true);
43
+ });
44
+ });
45
+ describe('scheduleFlush', () => {
46
+ it('should flush after debounce delay', async () => {
47
+ syncManager.markDirty('s-abc1');
48
+ syncManager.scheduleFlush();
49
+ expect(flushCallback).not.toHaveBeenCalled();
50
+ // Advance past debounce
51
+ await vi.advanceTimersByTimeAsync(config.debounceMs + 10);
52
+ expect(flushCallback).toHaveBeenCalled();
53
+ });
54
+ it('should reset debounce timer on multiple calls', async () => {
55
+ syncManager.markDirty('s-abc1');
56
+ syncManager.scheduleFlush();
57
+ // Advance halfway
58
+ await vi.advanceTimersByTimeAsync(config.debounceMs / 2);
59
+ syncManager.markDirty('s-abc2');
60
+ syncManager.scheduleFlush();
61
+ // Advance halfway again (would be past original debounce)
62
+ await vi.advanceTimersByTimeAsync(config.debounceMs / 2);
63
+ // Should not have flushed yet (debounce was reset)
64
+ expect(flushCallback).not.toHaveBeenCalled();
65
+ // Advance remaining time
66
+ await vi.advanceTimersByTimeAsync(config.debounceMs / 2 + 10);
67
+ expect(flushCallback).toHaveBeenCalled();
68
+ });
69
+ it('should force flush at max delay', async () => {
70
+ syncManager.markDirty('s-abc1');
71
+ syncManager.scheduleFlush();
72
+ // Keep resetting debounce but stay under max delay
73
+ // debounceMs = 100, maxDelayMs = 500
74
+ // Advance 80ms at a time, which is less than debounce
75
+ // After 4 iterations: 320ms total (still under 500ms max)
76
+ for (let i = 0; i < 4; i++) {
77
+ await vi.advanceTimersByTimeAsync(80);
78
+ syncManager.scheduleFlush();
79
+ }
80
+ // Should not have flushed yet (320ms < 500ms maxDelay)
81
+ expect(flushCallback).not.toHaveBeenCalled();
82
+ // Advance to max delay (need another 180ms to reach 500ms)
83
+ await vi.advanceTimersByTimeAsync(200);
84
+ expect(flushCallback).toHaveBeenCalled();
85
+ });
86
+ });
87
+ describe('flush', () => {
88
+ it('should flush immediately', async () => {
89
+ syncManager.markDirty('s-abc1');
90
+ await syncManager.flush();
91
+ expect(flushCallback).toHaveBeenCalled();
92
+ });
93
+ it('should clear pending changes after flush', async () => {
94
+ syncManager.markDirty('s-abc1');
95
+ await syncManager.flush();
96
+ expect(syncManager.hasPendingChanges()).toBe(false);
97
+ });
98
+ it('should not flush when no pending changes', async () => {
99
+ await syncManager.flush();
100
+ expect(flushCallback).not.toHaveBeenCalled();
101
+ });
102
+ it('should deduplicate concurrent flush calls', async () => {
103
+ syncManager.markDirty('s-abc1');
104
+ // Call flush multiple times concurrently
105
+ const promise1 = syncManager.flush();
106
+ const promise2 = syncManager.flush();
107
+ const promise3 = syncManager.flush();
108
+ await Promise.all([promise1, promise2, promise3]);
109
+ // Should only call flush callback once
110
+ expect(flushCallback).toHaveBeenCalledTimes(1);
111
+ });
112
+ });
113
+ describe('cancel', () => {
114
+ it('should cancel pending flush', async () => {
115
+ syncManager.markDirty('s-abc1');
116
+ syncManager.scheduleFlush();
117
+ syncManager.cancel();
118
+ // Advance past debounce
119
+ await vi.advanceTimersByTimeAsync(config.debounceMs + 10);
120
+ expect(flushCallback).not.toHaveBeenCalled();
121
+ });
122
+ it('should clear pending changes', () => {
123
+ syncManager.markDirty('s-abc1');
124
+ syncManager.cancel();
125
+ expect(syncManager.hasPendingChanges()).toBe(false);
126
+ });
127
+ });
128
+ });
129
+ //# sourceMappingURL=sync.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync.test.js","sourceRoot":"","sources":["../../../src/graph/__tests__/sync.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=validation.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.test.d.ts","sourceRoot":"","sources":["../../../src/graph/__tests__/validation.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,521 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { createValidationService } from '../validation.js';
3
+ describe('ValidationService', () => {
4
+ const service = createValidationService();
5
+ // =========================================================================
6
+ // Node Creation Validation
7
+ // =========================================================================
8
+ describe('validateCreateNode', () => {
9
+ describe('common fields', () => {
10
+ it('requires type', () => {
11
+ const input = { title: 'Test' };
12
+ const result = service.validateCreateNode(input);
13
+ expect(result.valid).toBe(false);
14
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'type' }));
15
+ });
16
+ it('rejects invalid type', () => {
17
+ const input = { type: 'invalid', title: 'Test' };
18
+ const result = service.validateCreateNode(input);
19
+ expect(result.valid).toBe(false);
20
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'INVALID_TYPE', field: 'type' }));
21
+ });
22
+ it('requires title', () => {
23
+ const input = { type: 'spec' };
24
+ const result = service.validateCreateNode(input);
25
+ expect(result.valid).toBe(false);
26
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'title' }));
27
+ });
28
+ it('rejects empty title', () => {
29
+ const input = { type: 'spec', title: '' };
30
+ const result = service.validateCreateNode(input);
31
+ expect(result.valid).toBe(false);
32
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'title' }));
33
+ });
34
+ it('rejects title exceeding max length', () => {
35
+ const input = {
36
+ type: 'spec',
37
+ title: 'a'.repeat(501),
38
+ };
39
+ const result = service.validateCreateNode(input);
40
+ expect(result.valid).toBe(false);
41
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'MAX_LENGTH', field: 'title' }));
42
+ });
43
+ it('rejects content exceeding max length', () => {
44
+ const input = {
45
+ type: 'spec',
46
+ title: 'Test',
47
+ content: 'a'.repeat(100_001),
48
+ };
49
+ const result = service.validateCreateNode(input);
50
+ expect(result.valid).toBe(false);
51
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'MAX_LENGTH', field: 'content' }));
52
+ });
53
+ it('rejects invalid priority (out of range)', () => {
54
+ const input = {
55
+ type: 'spec',
56
+ title: 'Test',
57
+ priority: 5,
58
+ };
59
+ const result = service.validateCreateNode(input);
60
+ expect(result.valid).toBe(false);
61
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'INVALID_RANGE', field: 'priority' }));
62
+ });
63
+ it('rejects negative priority', () => {
64
+ const input = {
65
+ type: 'spec',
66
+ title: 'Test',
67
+ priority: -1,
68
+ };
69
+ const result = service.validateCreateNode(input);
70
+ expect(result.valid).toBe(false);
71
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'INVALID_RANGE', field: 'priority' }));
72
+ });
73
+ it('accepts valid priority', () => {
74
+ const input = {
75
+ type: 'spec',
76
+ title: 'Test',
77
+ priority: 2,
78
+ };
79
+ const result = service.validateCreateNode(input);
80
+ expect(result.valid).toBe(true);
81
+ });
82
+ });
83
+ describe('spec validation', () => {
84
+ it('accepts valid spec', () => {
85
+ const input = {
86
+ type: 'spec',
87
+ title: 'Test Spec',
88
+ content: 'Some content',
89
+ };
90
+ const result = service.validateCreateNode(input);
91
+ expect(result.valid).toBe(true);
92
+ expect(result.errors).toHaveLength(0);
93
+ });
94
+ it('accepts spec with optional status', () => {
95
+ const input = {
96
+ type: 'spec',
97
+ title: 'Test Spec',
98
+ status: 'draft',
99
+ };
100
+ const result = service.validateCreateNode(input);
101
+ expect(result.valid).toBe(true);
102
+ });
103
+ });
104
+ describe('issue validation', () => {
105
+ it('requires status for issues', () => {
106
+ const input = {
107
+ type: 'issue',
108
+ title: 'Test Issue',
109
+ };
110
+ const result = service.validateCreateNode(input);
111
+ expect(result.valid).toBe(false);
112
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'status' }));
113
+ });
114
+ it('accepts valid issue', () => {
115
+ const input = {
116
+ type: 'issue',
117
+ title: 'Test Issue',
118
+ status: 'open',
119
+ };
120
+ const result = service.validateCreateNode(input);
121
+ expect(result.valid).toBe(true);
122
+ });
123
+ it('warns for non-standard status', () => {
124
+ const input = {
125
+ type: 'issue',
126
+ title: 'Test Issue',
127
+ status: 'custom_status',
128
+ };
129
+ const result = service.validateCreateNode(input);
130
+ expect(result.valid).toBe(true); // Warnings don't fail validation
131
+ expect(result.warnings).toContainEqual(expect.objectContaining({
132
+ code: 'NON_STANDARD_STATUS',
133
+ field: 'status',
134
+ }));
135
+ });
136
+ });
137
+ describe('feedback validation', () => {
138
+ it('requires target_id for feedback', () => {
139
+ const input = {
140
+ type: 'feedback',
141
+ title: 'Test Feedback',
142
+ feedback_type: 'comment',
143
+ };
144
+ const result = service.validateCreateNode(input);
145
+ expect(result.valid).toBe(false);
146
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'target_id' }));
147
+ });
148
+ it('requires feedback_type for feedback', () => {
149
+ const input = {
150
+ type: 'feedback',
151
+ title: 'Test Feedback',
152
+ target_id: 's-abc123',
153
+ };
154
+ const result = service.validateCreateNode(input);
155
+ expect(result.valid).toBe(false);
156
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'feedback_type' }));
157
+ });
158
+ it('accepts valid feedback', () => {
159
+ const input = {
160
+ type: 'feedback',
161
+ title: 'Test Feedback',
162
+ target_id: 's-abc123',
163
+ feedback_type: 'comment',
164
+ };
165
+ const result = service.validateCreateNode(input);
166
+ expect(result.valid).toBe(true);
167
+ });
168
+ });
169
+ describe('external validation', () => {
170
+ it('requires uri for external', () => {
171
+ const input = {
172
+ type: 'external',
173
+ title: 'External Node',
174
+ source: 'jira',
175
+ };
176
+ const result = service.validateCreateNode(input);
177
+ expect(result.valid).toBe(false);
178
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'uri' }));
179
+ });
180
+ it('requires source for external', () => {
181
+ const input = {
182
+ type: 'external',
183
+ title: 'External Node',
184
+ uri: 'jira://PROJ-123',
185
+ };
186
+ const result = service.validateCreateNode(input);
187
+ expect(result.valid).toBe(false);
188
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'source' }));
189
+ });
190
+ it('accepts valid external node', () => {
191
+ const input = {
192
+ type: 'external',
193
+ title: 'External Node',
194
+ uri: 'jira://PROJ-123',
195
+ source: 'jira',
196
+ };
197
+ const result = service.validateCreateNode(input);
198
+ expect(result.valid).toBe(true);
199
+ });
200
+ });
201
+ });
202
+ // =========================================================================
203
+ // Node Update Validation
204
+ // =========================================================================
205
+ describe('validateUpdateNode', () => {
206
+ const existingIssue = {
207
+ id: 'i-test',
208
+ uuid: 'test-uuid',
209
+ type: 'issue',
210
+ title: 'Existing Issue',
211
+ status: 'open',
212
+ created_at: '2025-01-26T10:00:00Z',
213
+ updated_at: '2025-01-26T10:00:00Z',
214
+ };
215
+ it('accepts valid update', () => {
216
+ const result = service.validateUpdateNode(existingIssue, {
217
+ title: 'Updated Title',
218
+ });
219
+ expect(result.valid).toBe(true);
220
+ });
221
+ it('rejects title exceeding max length', () => {
222
+ const result = service.validateUpdateNode(existingIssue, {
223
+ title: 'a'.repeat(501),
224
+ });
225
+ expect(result.valid).toBe(false);
226
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'MAX_LENGTH', field: 'title' }));
227
+ });
228
+ it('warns for non-standard status on issues', () => {
229
+ const result = service.validateUpdateNode(existingIssue, {
230
+ status: 'custom',
231
+ });
232
+ expect(result.valid).toBe(true);
233
+ expect(result.warnings).toContainEqual(expect.objectContaining({ code: 'NON_STANDARD_STATUS' }));
234
+ });
235
+ });
236
+ // =========================================================================
237
+ // Edge Validation
238
+ // =========================================================================
239
+ describe('validateCreateEdge', () => {
240
+ const mockGetNode = async (id) => {
241
+ const nodes = {
242
+ 's-spec1': {
243
+ id: 's-spec1',
244
+ uuid: 'spec-uuid',
245
+ type: 'spec',
246
+ title: 'Test Spec',
247
+ created_at: '2025-01-26T10:00:00Z',
248
+ updated_at: '2025-01-26T10:00:00Z',
249
+ },
250
+ 'i-issue1': {
251
+ id: 'i-issue1',
252
+ uuid: 'issue-uuid',
253
+ type: 'issue',
254
+ title: 'Test Issue',
255
+ status: 'open',
256
+ created_at: '2025-01-26T10:00:00Z',
257
+ updated_at: '2025-01-26T10:00:00Z',
258
+ },
259
+ };
260
+ return nodes[id] || null;
261
+ };
262
+ it('requires from_id', async () => {
263
+ const input = { to_id: 's-spec1', type: 'blocks' };
264
+ const result = await service.validateCreateEdge(input, mockGetNode);
265
+ expect(result.valid).toBe(false);
266
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'from_id' }));
267
+ });
268
+ it('requires to_id', async () => {
269
+ const input = { from_id: 'i-issue1', type: 'blocks' };
270
+ const result = await service.validateCreateEdge(input, mockGetNode);
271
+ expect(result.valid).toBe(false);
272
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'to_id' }));
273
+ });
274
+ it('requires type', async () => {
275
+ const input = { from_id: 'i-issue1', to_id: 's-spec1' };
276
+ const result = await service.validateCreateEdge(input, mockGetNode);
277
+ expect(result.valid).toBe(false);
278
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'REQUIRED', field: 'type' }));
279
+ });
280
+ it('rejects self-reference', async () => {
281
+ const input = {
282
+ from_id: 'i-issue1',
283
+ to_id: 'i-issue1',
284
+ type: 'blocks',
285
+ };
286
+ const result = await service.validateCreateEdge(input, mockGetNode);
287
+ expect(result.valid).toBe(false);
288
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'SELF_REFERENCE' }));
289
+ });
290
+ it('errors when source node not found', async () => {
291
+ const input = {
292
+ from_id: 'i-nonexistent',
293
+ to_id: 's-spec1',
294
+ type: 'implements',
295
+ };
296
+ const result = await service.validateCreateEdge(input, mockGetNode);
297
+ expect(result.valid).toBe(false);
298
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'NOT_FOUND', field: 'from_id' }));
299
+ });
300
+ it('errors when target node not found', async () => {
301
+ const input = {
302
+ from_id: 'i-issue1',
303
+ to_id: 's-nonexistent',
304
+ type: 'implements',
305
+ };
306
+ const result = await service.validateCreateEdge(input, mockGetNode);
307
+ expect(result.valid).toBe(false);
308
+ expect(result.errors).toContainEqual(expect.objectContaining({ code: 'NOT_FOUND', field: 'to_id' }));
309
+ });
310
+ it('accepts valid edge', async () => {
311
+ const input = {
312
+ from_id: 'i-issue1',
313
+ to_id: 's-spec1',
314
+ type: 'implements',
315
+ };
316
+ const result = await service.validateCreateEdge(input, mockGetNode);
317
+ expect(result.valid).toBe(true);
318
+ });
319
+ it('warns when implements edge has non-issue source', async () => {
320
+ const input = {
321
+ from_id: 's-spec1',
322
+ to_id: 'i-issue1',
323
+ type: 'implements',
324
+ };
325
+ const result = await service.validateCreateEdge(input, mockGetNode);
326
+ expect(result.valid).toBe(true); // Warning, not error
327
+ expect(result.warnings).toContainEqual(expect.objectContaining({ code: 'IMPLEMENTS_FROM_NON_ISSUE' }));
328
+ });
329
+ it('warns when implements edge has non-spec target', async () => {
330
+ const input = {
331
+ from_id: 'i-issue1',
332
+ to_id: 'i-issue1',
333
+ type: 'implements',
334
+ };
335
+ // Note: This will hit self-reference first, so use different nodes
336
+ const getNodeWithTwoIssues = async (id) => {
337
+ const nodes = {
338
+ 'i-issue1': {
339
+ id: 'i-issue1',
340
+ uuid: 'issue-uuid-1',
341
+ type: 'issue',
342
+ title: 'Test Issue 1',
343
+ status: 'open',
344
+ created_at: '2025-01-26T10:00:00Z',
345
+ updated_at: '2025-01-26T10:00:00Z',
346
+ },
347
+ 'i-issue2': {
348
+ id: 'i-issue2',
349
+ uuid: 'issue-uuid-2',
350
+ type: 'issue',
351
+ title: 'Test Issue 2',
352
+ status: 'open',
353
+ created_at: '2025-01-26T10:00:00Z',
354
+ updated_at: '2025-01-26T10:00:00Z',
355
+ },
356
+ };
357
+ return nodes[id] || null;
358
+ };
359
+ const input2 = {
360
+ from_id: 'i-issue1',
361
+ to_id: 'i-issue2',
362
+ type: 'implements',
363
+ };
364
+ const result = await service.validateCreateEdge(input2, getNodeWithTwoIssues);
365
+ expect(result.valid).toBe(true);
366
+ expect(result.warnings).toContainEqual(expect.objectContaining({ code: 'IMPLEMENTS_TO_NON_SPEC' }));
367
+ });
368
+ it('allows external URIs without node lookup', async () => {
369
+ const input = {
370
+ from_id: 'i-issue1',
371
+ to_id: 'jira://PROJ-123',
372
+ type: 'references',
373
+ };
374
+ const result = await service.validateCreateEdge(input, mockGetNode);
375
+ expect(result.valid).toBe(true);
376
+ });
377
+ });
378
+ // =========================================================================
379
+ // Cycle Detection
380
+ // =========================================================================
381
+ describe('detectCycle', () => {
382
+ it('detects simple cycle (A→B, adding B→A)', async () => {
383
+ // A blocks B already exists
384
+ const getBlocksEdges = async (nodeId) => {
385
+ if (nodeId === 'i-b') {
386
+ return []; // B doesn't block anything yet
387
+ }
388
+ if (nodeId === 'i-a') {
389
+ return [
390
+ {
391
+ id: 'x-1',
392
+ uuid: 'edge-uuid-1',
393
+ from_id: 'i-a',
394
+ to_id: 'i-b',
395
+ type: 'blocks',
396
+ created_at: '2025-01-26T10:00:00Z',
397
+ },
398
+ ];
399
+ }
400
+ return [];
401
+ };
402
+ // Adding B→A would create cycle
403
+ const result = await service.detectCycle('i-b', 'i-a', getBlocksEdges);
404
+ expect(result.hasCycle).toBe(true);
405
+ expect(result.cycle).toEqual(['i-b', 'i-a']);
406
+ });
407
+ it('detects transitive cycle (A→B→C, adding C→A)', async () => {
408
+ // A→B, B→C exists
409
+ const getBlocksEdges = async (nodeId) => {
410
+ const edges = {
411
+ 'i-a': [
412
+ {
413
+ id: 'x-1',
414
+ uuid: 'edge-1',
415
+ from_id: 'i-a',
416
+ to_id: 'i-b',
417
+ type: 'blocks',
418
+ created_at: '2025-01-26T10:00:00Z',
419
+ },
420
+ ],
421
+ 'i-b': [
422
+ {
423
+ id: 'x-2',
424
+ uuid: 'edge-2',
425
+ from_id: 'i-b',
426
+ to_id: 'i-c',
427
+ type: 'blocks',
428
+ created_at: '2025-01-26T10:00:00Z',
429
+ },
430
+ ],
431
+ 'i-c': [],
432
+ };
433
+ return edges[nodeId] || [];
434
+ };
435
+ // Adding C→A would create cycle
436
+ const result = await service.detectCycle('i-c', 'i-a', getBlocksEdges);
437
+ expect(result.hasCycle).toBe(true);
438
+ expect(result.cycle).toContain('i-a');
439
+ expect(result.cycle).toContain('i-b');
440
+ expect(result.cycle).toContain('i-c');
441
+ });
442
+ it('returns no cycle for valid edge', async () => {
443
+ // A→B exists
444
+ const getBlocksEdges = async (nodeId) => {
445
+ if (nodeId === 'i-a') {
446
+ return [
447
+ {
448
+ id: 'x-1',
449
+ uuid: 'edge-uuid-1',
450
+ from_id: 'i-a',
451
+ to_id: 'i-b',
452
+ type: 'blocks',
453
+ created_at: '2025-01-26T10:00:00Z',
454
+ },
455
+ ];
456
+ }
457
+ return [];
458
+ };
459
+ // Adding A→C is fine (no cycle)
460
+ const result = await service.detectCycle('i-a', 'i-c', getBlocksEdges);
461
+ expect(result.hasCycle).toBe(false);
462
+ expect(result.cycle).toBeUndefined();
463
+ });
464
+ it('handles no existing edges', async () => {
465
+ const getBlocksEdges = async () => [];
466
+ const result = await service.detectCycle('i-a', 'i-b', getBlocksEdges);
467
+ expect(result.hasCycle).toBe(false);
468
+ });
469
+ it('handles complex graph without cycle', async () => {
470
+ // Diamond: A→B, A→C, B→D, C→D
471
+ const getBlocksEdges = async (nodeId) => {
472
+ const edges = {
473
+ 'i-a': [
474
+ {
475
+ id: 'x-1',
476
+ uuid: 'e1',
477
+ from_id: 'i-a',
478
+ to_id: 'i-b',
479
+ type: 'blocks',
480
+ created_at: '2025-01-26T10:00:00Z',
481
+ },
482
+ {
483
+ id: 'x-2',
484
+ uuid: 'e2',
485
+ from_id: 'i-a',
486
+ to_id: 'i-c',
487
+ type: 'blocks',
488
+ created_at: '2025-01-26T10:00:00Z',
489
+ },
490
+ ],
491
+ 'i-b': [
492
+ {
493
+ id: 'x-3',
494
+ uuid: 'e3',
495
+ from_id: 'i-b',
496
+ to_id: 'i-d',
497
+ type: 'blocks',
498
+ created_at: '2025-01-26T10:00:00Z',
499
+ },
500
+ ],
501
+ 'i-c': [
502
+ {
503
+ id: 'x-4',
504
+ uuid: 'e4',
505
+ from_id: 'i-c',
506
+ to_id: 'i-d',
507
+ type: 'blocks',
508
+ created_at: '2025-01-26T10:00:00Z',
509
+ },
510
+ ],
511
+ 'i-d': [],
512
+ };
513
+ return edges[nodeId] || [];
514
+ };
515
+ // Adding D→E is fine
516
+ const result = await service.detectCycle('i-d', 'i-e', getBlocksEdges);
517
+ expect(result.hasCycle).toBe(false);
518
+ });
519
+ });
520
+ });
521
+ //# sourceMappingURL=validation.test.js.map