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,791 @@
1
+ /**
2
+ * Tests for Provider-Aware Graph Store
3
+ */
4
+ import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
5
+ import { createProviderAwareStore, } from '../provider-store.js';
6
+ import { createProviderRegistry } from '../../providers/registry.js';
7
+ describe('ProviderAwareStore', () => {
8
+ let baseStore;
9
+ let providerStore;
10
+ let mockBeadsProvider;
11
+ // Mock nodes
12
+ const mockSpec = {
13
+ id: 's-abc1',
14
+ uuid: 'uuid-1',
15
+ type: 'spec',
16
+ title: 'Test Spec',
17
+ content: 'Spec content',
18
+ created_at: '2024-01-01T00:00:00.000Z',
19
+ updated_at: '2024-01-01T00:00:00.000Z',
20
+ };
21
+ const mockExternalNode = {
22
+ id: 'x-ext1',
23
+ uuid: 'uuid-ext-1',
24
+ type: 'external',
25
+ title: 'External Issue',
26
+ uri: 'beads://./bd-123',
27
+ source: 'beads',
28
+ materialized: true,
29
+ cached_at: new Date().toISOString(),
30
+ created_at: '2024-01-01T00:00:00.000Z',
31
+ updated_at: '2024-01-01T00:00:00.000Z',
32
+ };
33
+ const mockProviderNode = {
34
+ id: 'bd-123',
35
+ uri: 'beads://./bd-123',
36
+ type: 'issue',
37
+ title: 'Beads Issue',
38
+ content: 'Issue content',
39
+ status: 'open',
40
+ priority: 2,
41
+ rawData: { custom: 'data' },
42
+ fetchedAt: '2024-01-01T00:00:00.000Z',
43
+ };
44
+ beforeEach(() => {
45
+ vi.useFakeTimers();
46
+ // Create mock base store
47
+ baseStore = {
48
+ getNode: vi.fn(),
49
+ createNode: vi.fn(),
50
+ updateNode: vi.fn(),
51
+ deleteNode: vi.fn(),
52
+ query: {
53
+ nodes: vi.fn().mockResolvedValue([]),
54
+ edges: vi.fn(),
55
+ blockers: vi.fn(),
56
+ blocking: vi.fn(),
57
+ ready: vi.fn(),
58
+ children: vi.fn(),
59
+ descendants: vi.fn(),
60
+ ancestors: vi.fn(),
61
+ feedback: vi.fn(),
62
+ },
63
+ initialize: vi.fn(),
64
+ close: vi.fn(),
65
+ flush: vi.fn(),
66
+ createEdge: vi.fn(),
67
+ getEdge: vi.fn(),
68
+ deleteEdge: vi.fn(),
69
+ restoreNode: vi.fn(),
70
+ addTags: vi.fn(),
71
+ removeTags: vi.fn(),
72
+ setTags: vi.fn(),
73
+ transaction: vi.fn(),
74
+ };
75
+ // Create mock beads provider
76
+ mockBeadsProvider = {
77
+ name: 'beads',
78
+ schemes: ['beads', 'bd'],
79
+ capabilities: { read: true, write: true, search: true, watch: false, mount: true, feedback: false },
80
+ parseUri: vi.fn((uri) => {
81
+ if (uri.startsWith('beads://') || uri.startsWith('bd://')) {
82
+ const id = uri.split('/').pop() || '';
83
+ return { scheme: 'beads', id, isRelative: uri.includes('/.') };
84
+ }
85
+ return null;
86
+ }),
87
+ buildUri: vi.fn((id) => `beads://./${id}`),
88
+ isValidUri: vi.fn((uri) => uri.startsWith('beads://') || uri.startsWith('bd://')),
89
+ get: vi.fn().mockResolvedValue(mockProviderNode),
90
+ list: vi.fn(),
91
+ create: vi.fn(),
92
+ update: vi.fn(),
93
+ delete: vi.fn(),
94
+ };
95
+ // Create provider store
96
+ providerStore = createProviderAwareStore(baseStore);
97
+ });
98
+ afterEach(() => {
99
+ providerStore.stopBackgroundSync();
100
+ vi.useRealTimers();
101
+ });
102
+ describe('initialization', () => {
103
+ it('should have providers registry', () => {
104
+ expect(providerStore.providers).toBeDefined();
105
+ expect(providerStore.providers.list).toBeDefined();
106
+ });
107
+ it('should have materialization manager', () => {
108
+ expect(providerStore.materialization).toBeDefined();
109
+ expect(providerStore.materialization.config).toBeDefined();
110
+ });
111
+ it('should auto-register native provider', () => {
112
+ const nativeProvider = providerStore.providers.get('native');
113
+ expect(nativeProvider).toBeDefined();
114
+ expect(nativeProvider?.name).toBe('native');
115
+ });
116
+ it('should not auto-register native provider when disabled', () => {
117
+ const storeWithoutNative = createProviderAwareStore(baseStore, {
118
+ autoRegisterNative: false,
119
+ });
120
+ expect(storeWithoutNative.providers.get('native')).toBeUndefined();
121
+ });
122
+ it('should use provided registry', () => {
123
+ const customRegistry = createProviderRegistry();
124
+ customRegistry.register(mockBeadsProvider);
125
+ const storeWithCustomRegistry = createProviderAwareStore(baseStore, {
126
+ registry: customRegistry,
127
+ autoRegisterNative: false,
128
+ });
129
+ expect(storeWithCustomRegistry.providers.get('beads')).toBe(mockBeadsProvider);
130
+ });
131
+ });
132
+ describe('resolveNode', () => {
133
+ describe('local IDs', () => {
134
+ it('should resolve local IDs via getNode', async () => {
135
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
136
+ const result = await providerStore.resolveNode('s-abc1');
137
+ expect(baseStore.getNode).toHaveBeenCalledWith('s-abc1');
138
+ expect(result).toBe(mockSpec);
139
+ });
140
+ it('should return null for non-existent local node', async () => {
141
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
142
+ const result = await providerStore.resolveNode('s-notfound');
143
+ expect(result).toBeNull();
144
+ });
145
+ it('should handle all local ID prefixes', async () => {
146
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
147
+ for (const prefix of ['s-', 'i-', 'f-', 'e-', 'x-']) {
148
+ await providerStore.resolveNode(`${prefix}abc1`);
149
+ expect(baseStore.getNode).toHaveBeenCalledWith(`${prefix}abc1`);
150
+ }
151
+ });
152
+ });
153
+ describe('external URIs', () => {
154
+ beforeEach(() => {
155
+ providerStore.providers.register(mockBeadsProvider);
156
+ });
157
+ it('should resolve URIs via provider', async () => {
158
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
159
+ const result = await providerStore.resolveNode('beads://./bd-123');
160
+ expect(mockBeadsProvider.get).toHaveBeenCalledWith('bd-123', undefined);
161
+ expect(result).toBeDefined();
162
+ });
163
+ it('should return null for unknown schemes', async () => {
164
+ const result = await providerStore.resolveNode('unknown://./something');
165
+ expect(result).toBeNull();
166
+ });
167
+ it('should use cached node when not stale', async () => {
168
+ const freshNode = {
169
+ ...mockExternalNode,
170
+ cached_at: new Date().toISOString(),
171
+ };
172
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([freshNode]);
173
+ const result = await providerStore.resolveNode('beads://./bd-123');
174
+ expect(mockBeadsProvider.get).not.toHaveBeenCalled();
175
+ expect(result).toEqual(freshNode);
176
+ });
177
+ it('should bypass cache when refresh=true', async () => {
178
+ const freshNode = {
179
+ ...mockExternalNode,
180
+ cached_at: new Date().toISOString(),
181
+ };
182
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([freshNode]);
183
+ await providerStore.resolveNode('beads://./bd-123', { refresh: true });
184
+ expect(mockBeadsProvider.get).toHaveBeenCalled();
185
+ });
186
+ it('should materialize when explicitly requested', async () => {
187
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
188
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
189
+ const result = await providerStore.resolveNode('beads://./bd-123', {
190
+ materialize: true,
191
+ });
192
+ expect(baseStore.createNode).toHaveBeenCalledWith(expect.objectContaining({
193
+ type: 'external',
194
+ uri: 'beads://./bd-123',
195
+ }));
196
+ });
197
+ it('should return provider node directly when not materializing', async () => {
198
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
199
+ const result = await providerStore.resolveNode('beads://./bd-123');
200
+ // With on-demand strategy (default), should not materialize without explicit flag
201
+ expect(baseStore.createNode).not.toHaveBeenCalled();
202
+ });
203
+ });
204
+ });
205
+ describe('materializeNode', () => {
206
+ beforeEach(() => {
207
+ providerStore.providers.register(mockBeadsProvider);
208
+ });
209
+ it('should create external node from provider data', async () => {
210
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
211
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
212
+ const result = await providerStore.materializeNode('beads://./bd-123');
213
+ expect(mockBeadsProvider.get).toHaveBeenCalledWith('bd-123');
214
+ expect(baseStore.createNode).toHaveBeenCalledWith(expect.objectContaining({
215
+ type: 'external',
216
+ uri: 'beads://./bd-123',
217
+ source: 'beads',
218
+ title: 'Beads Issue',
219
+ }));
220
+ });
221
+ it('should throw for unknown URI scheme', async () => {
222
+ await expect(providerStore.materializeNode('unknown://./something')).rejects.toThrow('No provider found');
223
+ });
224
+ it('should throw if node not found in provider', async () => {
225
+ vi.mocked(mockBeadsProvider.get).mockResolvedValue(null);
226
+ await expect(providerStore.materializeNode('beads://./bd-notfound')).rejects.toThrow('Node not found');
227
+ });
228
+ it('should update existing node if already materialized', async () => {
229
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockExternalNode]);
230
+ vi.mocked(baseStore.updateNode).mockResolvedValue(mockExternalNode);
231
+ await providerStore.materializeNode('beads://./bd-123');
232
+ expect(baseStore.updateNode).toHaveBeenCalled();
233
+ expect(baseStore.createNode).not.toHaveBeenCalled();
234
+ });
235
+ });
236
+ describe('refreshNode', () => {
237
+ beforeEach(() => {
238
+ providerStore.providers.register(mockBeadsProvider);
239
+ });
240
+ it('should refresh an external node', async () => {
241
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockExternalNode);
242
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockExternalNode]);
243
+ vi.mocked(baseStore.updateNode).mockResolvedValue(mockExternalNode);
244
+ const result = await providerStore.refreshNode('x-ext1');
245
+ expect(mockBeadsProvider.get).toHaveBeenCalledWith('beads://./bd-123', undefined);
246
+ expect(baseStore.updateNode).toHaveBeenCalled();
247
+ });
248
+ it('should throw for non-external node', async () => {
249
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
250
+ await expect(providerStore.refreshNode('s-abc1')).rejects.toThrow('not external');
251
+ });
252
+ it('should throw for non-existent node', async () => {
253
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
254
+ await expect(providerStore.refreshNode('x-notfound')).rejects.toThrow('Node not found');
255
+ });
256
+ });
257
+ describe('background sync', () => {
258
+ it('should start and stop background sync', () => {
259
+ providerStore = createProviderAwareStore(baseStore, {
260
+ materialization: {
261
+ backgroundSyncInterval: 1000,
262
+ },
263
+ });
264
+ expect(providerStore.isBackgroundSyncRunning()).toBe(false);
265
+ providerStore.startBackgroundSync();
266
+ expect(providerStore.isBackgroundSyncRunning()).toBe(true);
267
+ providerStore.stopBackgroundSync();
268
+ expect(providerStore.isBackgroundSyncRunning()).toBe(false);
269
+ });
270
+ it('should not start if interval is 0', () => {
271
+ providerStore = createProviderAwareStore(baseStore, {
272
+ materialization: {
273
+ backgroundSyncInterval: 0,
274
+ },
275
+ });
276
+ providerStore.startBackgroundSync();
277
+ expect(providerStore.isBackgroundSyncRunning()).toBe(false);
278
+ });
279
+ });
280
+ describe('provider registration', () => {
281
+ it('should allow registering additional providers', () => {
282
+ providerStore.providers.register(mockBeadsProvider);
283
+ expect(providerStore.providers.get('beads')).toBe(mockBeadsProvider);
284
+ expect(providerStore.providers.canResolve('beads://./bd-123')).toBe(true);
285
+ });
286
+ it('should allow unregistering providers', () => {
287
+ providerStore.providers.register(mockBeadsProvider);
288
+ providerStore.providers.unregister('beads');
289
+ expect(providerStore.providers.get('beads')).toBeUndefined();
290
+ });
291
+ });
292
+ describe('base store methods', () => {
293
+ it('should pass through getNode to base store', async () => {
294
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
295
+ const result = await providerStore.getNode('s-abc1');
296
+ expect(baseStore.getNode).toHaveBeenCalledWith('s-abc1');
297
+ expect(result).toBe(mockSpec);
298
+ });
299
+ it('should pass through createNode to base store', async () => {
300
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockSpec);
301
+ const result = await providerStore.createNode({
302
+ type: 'spec',
303
+ title: 'New Spec',
304
+ });
305
+ expect(baseStore.createNode).toHaveBeenCalled();
306
+ expect(result).toBe(mockSpec);
307
+ });
308
+ it('should pass through query to base store', async () => {
309
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockSpec]);
310
+ const result = await providerStore.query.nodes({ type: 'spec' });
311
+ expect(baseStore.query.nodes).toHaveBeenCalledWith({ type: 'spec' });
312
+ expect(result).toEqual([mockSpec]);
313
+ });
314
+ });
315
+ describe('materialization strategies', () => {
316
+ beforeEach(() => {
317
+ providerStore.providers.register(mockBeadsProvider);
318
+ });
319
+ it('should materialize with lazy strategy on resolve', async () => {
320
+ const lazyStore = createProviderAwareStore(baseStore, {
321
+ materialization: { default: 'lazy' },
322
+ });
323
+ lazyStore.providers.register(mockBeadsProvider);
324
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
325
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
326
+ await lazyStore.resolveNode('beads://./bd-123');
327
+ expect(baseStore.createNode).toHaveBeenCalled();
328
+ });
329
+ it('should materialize with eager strategy always', async () => {
330
+ const eagerStore = createProviderAwareStore(baseStore, {
331
+ materialization: { default: 'eager' },
332
+ });
333
+ eagerStore.providers.register(mockBeadsProvider);
334
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
335
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
336
+ await eagerStore.resolveNode('beads://./bd-123');
337
+ expect(baseStore.createNode).toHaveBeenCalled();
338
+ });
339
+ it('should not materialize with none strategy', async () => {
340
+ const noneStore = createProviderAwareStore(baseStore, {
341
+ materialization: { default: 'none' },
342
+ });
343
+ noneStore.providers.register(mockBeadsProvider);
344
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
345
+ await noneStore.resolveNode('beads://./bd-123', { materialize: true });
346
+ // Even with explicit flag, 'none' should not materialize
347
+ expect(baseStore.createNode).not.toHaveBeenCalled();
348
+ });
349
+ });
350
+ // ===========================================================================
351
+ // Provider-Routed CRUD (Unified Interface)
352
+ // ===========================================================================
353
+ describe('providerCreate', () => {
354
+ it('should create locally when defaultProvider is native', async () => {
355
+ const store = createProviderAwareStore(baseStore, {
356
+ defaultProvider: 'native',
357
+ });
358
+ const mockNode = { ...mockSpec };
359
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockNode);
360
+ const result = await store.providerCreate({
361
+ type: 'spec',
362
+ title: 'Test Spec',
363
+ });
364
+ expect(result.provider).toBe('native');
365
+ expect(result.uri).toBeUndefined();
366
+ expect(baseStore.createNode).toHaveBeenCalledWith({
367
+ type: 'spec',
368
+ title: 'Test Spec',
369
+ });
370
+ });
371
+ it('should route to external provider via scheme param', async () => {
372
+ const store = createProviderAwareStore(baseStore, {
373
+ defaultProvider: 'native',
374
+ });
375
+ store.providers.register(mockBeadsProvider);
376
+ const createdProviderNode = {
377
+ id: 'bd-new1',
378
+ uri: 'beads://./bd-new1',
379
+ type: 'issue',
380
+ title: 'New Issue',
381
+ fetchedAt: new Date().toISOString(),
382
+ };
383
+ vi.mocked(mockBeadsProvider.create).mockResolvedValue(createdProviderNode);
384
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
385
+ vi.mocked(baseStore.createNode).mockResolvedValue({
386
+ ...mockExternalNode,
387
+ id: 'x-new1',
388
+ title: 'New Issue',
389
+ uri: 'beads://./bd-new1',
390
+ });
391
+ const result = await store.providerCreate({ type: 'issue', title: 'New Issue' }, { scheme: 'beads' });
392
+ expect(result.provider).toBe('beads');
393
+ expect(result.uri).toBe('beads://./bd-new1');
394
+ expect(mockBeadsProvider.create).toHaveBeenCalledWith({
395
+ type: 'issue',
396
+ title: 'New Issue',
397
+ content: undefined,
398
+ status: undefined,
399
+ priority: undefined,
400
+ metadata: undefined,
401
+ }, undefined);
402
+ // Should materialize the node locally
403
+ expect(baseStore.createNode).toHaveBeenCalled();
404
+ });
405
+ it('should use defaultProvider when no scheme specified', async () => {
406
+ const store = createProviderAwareStore(baseStore, {
407
+ defaultProvider: 'beads',
408
+ });
409
+ store.providers.register(mockBeadsProvider);
410
+ const createdProviderNode = {
411
+ id: 'bd-new2',
412
+ uri: 'beads://./bd-new2',
413
+ type: 'issue',
414
+ title: 'Default Provider Issue',
415
+ fetchedAt: new Date().toISOString(),
416
+ };
417
+ vi.mocked(mockBeadsProvider.create).mockResolvedValue(createdProviderNode);
418
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
419
+ vi.mocked(baseStore.createNode).mockResolvedValue({
420
+ ...mockExternalNode,
421
+ id: 'x-new2',
422
+ });
423
+ const result = await store.providerCreate({
424
+ type: 'issue',
425
+ title: 'Default Provider Issue',
426
+ });
427
+ // Should route to beads since it's the defaultProvider
428
+ expect(result.provider).toBe('beads');
429
+ expect(mockBeadsProvider.create).toHaveBeenCalled();
430
+ });
431
+ it('should throw for unknown provider', async () => {
432
+ const store = createProviderAwareStore(baseStore);
433
+ await expect(store.providerCreate({ type: 'issue', title: 'Test' }, { scheme: 'nonexistent' })).rejects.toThrow('Unknown provider: nonexistent');
434
+ });
435
+ it('should throw for read-only provider', async () => {
436
+ const readOnlyProvider = {
437
+ ...mockBeadsProvider,
438
+ name: 'readonly',
439
+ schemes: ['readonly'],
440
+ capabilities: { read: true, write: false, search: false, watch: false, mount: false, feedback: false },
441
+ };
442
+ const store = createProviderAwareStore(baseStore);
443
+ store.providers.register(readOnlyProvider);
444
+ await expect(store.providerCreate({ type: 'issue', title: 'Test' }, { scheme: 'readonly' })).rejects.toThrow('read-only');
445
+ });
446
+ it('should throw for non-mountable provider', async () => {
447
+ const nonMountable = {
448
+ ...mockBeadsProvider,
449
+ name: 'nomount',
450
+ schemes: ['nomount'],
451
+ capabilities: { read: true, write: true, search: false, watch: false, mount: false, feedback: false },
452
+ };
453
+ const store = createProviderAwareStore(baseStore);
454
+ store.providers.register(nonMountable);
455
+ await expect(store.providerCreate({ type: 'issue', title: 'Test' }, { scheme: 'nomount' })).rejects.toThrow('does not support mounting');
456
+ });
457
+ });
458
+ describe('providerGet', () => {
459
+ it('should get local node directly', async () => {
460
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
461
+ const result = await providerStore.providerGet('s-abc1');
462
+ expect(baseStore.getNode).toHaveBeenCalledWith('s-abc1');
463
+ expect(result).toEqual(mockSpec);
464
+ });
465
+ it('should refresh stale external node', async () => {
466
+ const staleNode = {
467
+ ...mockExternalNode,
468
+ cached_at: '2020-01-01T00:00:00.000Z', // very old
469
+ };
470
+ vi.mocked(baseStore.getNode).mockResolvedValue(staleNode);
471
+ providerStore.providers.register(mockBeadsProvider);
472
+ // Mock the refresh path: query for existing, then update
473
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([staleNode]);
474
+ vi.mocked(baseStore.updateNode).mockResolvedValue({
475
+ ...staleNode,
476
+ title: 'Beads Issue',
477
+ cached_at: new Date().toISOString(),
478
+ });
479
+ const result = await providerStore.providerGet('x-ext1');
480
+ expect(mockBeadsProvider.get).toHaveBeenCalled();
481
+ });
482
+ it('should resolve provider URI and materialize', async () => {
483
+ providerStore.providers.register(mockBeadsProvider);
484
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
485
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
486
+ const result = await providerStore.providerGet('beads://./bd-123');
487
+ expect(mockBeadsProvider.get).toHaveBeenCalled();
488
+ });
489
+ it('should return null for nonexistent local node', async () => {
490
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
491
+ const result = await providerStore.providerGet('s-nonexist');
492
+ expect(result).toBeNull();
493
+ });
494
+ });
495
+ describe('providerUpdate', () => {
496
+ it('should update local node directly', async () => {
497
+ const updatedNode = { ...mockSpec, title: 'Updated' };
498
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
499
+ vi.mocked(baseStore.updateNode).mockResolvedValue(updatedNode);
500
+ const result = await providerStore.providerUpdate('s-abc1', { title: 'Updated' });
501
+ expect(baseStore.updateNode).toHaveBeenCalledWith('s-abc1', { title: 'Updated' });
502
+ expect(result.title).toBe('Updated');
503
+ });
504
+ it('should route update to external provider for materialized node', async () => {
505
+ providerStore.providers.register(mockBeadsProvider);
506
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockExternalNode);
507
+ const updatedProviderNode = {
508
+ ...mockProviderNode,
509
+ title: 'Updated in Beads',
510
+ };
511
+ vi.mocked(mockBeadsProvider.update).mockResolvedValue(updatedProviderNode);
512
+ // Mock materialization (find existing, then update)
513
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockExternalNode]);
514
+ vi.mocked(baseStore.updateNode).mockResolvedValue({
515
+ ...mockExternalNode,
516
+ title: 'Updated in Beads',
517
+ });
518
+ const result = await providerStore.providerUpdate('x-ext1', { title: 'Updated in Beads' });
519
+ expect(mockBeadsProvider.update).toHaveBeenCalledWith('bd-123', {
520
+ title: 'Updated in Beads',
521
+ content: undefined,
522
+ status: undefined,
523
+ priority: undefined,
524
+ metadata: undefined,
525
+ }, undefined);
526
+ });
527
+ it('should route update via provider URI', async () => {
528
+ providerStore.providers.register(mockBeadsProvider);
529
+ // No existing materialized copy
530
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
531
+ // Mock fetching the node for materialization
532
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockExternalNode);
533
+ const updatedProviderNode = {
534
+ ...mockProviderNode,
535
+ title: 'Updated via URI',
536
+ };
537
+ vi.mocked(mockBeadsProvider.update).mockResolvedValue(updatedProviderNode);
538
+ // After materialization, the update path will re-materialize
539
+ vi.mocked(baseStore.updateNode).mockResolvedValue({
540
+ ...mockExternalNode,
541
+ title: 'Updated via URI',
542
+ });
543
+ const result = await providerStore.providerUpdate('beads://./bd-123', { title: 'Updated via URI' });
544
+ expect(mockBeadsProvider.update).toHaveBeenCalled();
545
+ });
546
+ it('should throw for nonexistent node', async () => {
547
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
548
+ await expect(providerStore.providerUpdate('s-nonexist', { title: 'Test' })).rejects.toThrow('not found');
549
+ });
550
+ it('should throw when external node provider is not registered', async () => {
551
+ // External node exists but its provider (beads) is NOT registered
552
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockExternalNode);
553
+ await expect(providerStore.providerUpdate('x-ext1', { title: 'Update' })).rejects.toThrow('Provider not registered');
554
+ });
555
+ });
556
+ describe('providerDelete', () => {
557
+ it('should delete local node directly', async () => {
558
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockSpec);
559
+ vi.mocked(baseStore.deleteNode).mockResolvedValue(undefined);
560
+ await providerStore.providerDelete('s-abc1');
561
+ expect(baseStore.deleteNode).toHaveBeenCalledWith('s-abc1', undefined);
562
+ });
563
+ it('should route delete to external provider and remove local copy', async () => {
564
+ providerStore.providers.register(mockBeadsProvider);
565
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockExternalNode);
566
+ vi.mocked(mockBeadsProvider.delete).mockResolvedValue(undefined);
567
+ vi.mocked(baseStore.deleteNode).mockResolvedValue(undefined);
568
+ await providerStore.providerDelete('x-ext1');
569
+ // Should delete in provider
570
+ expect(mockBeadsProvider.delete).toHaveBeenCalledWith('bd-123', undefined);
571
+ // Should hard-delete local materialized copy
572
+ expect(baseStore.deleteNode).toHaveBeenCalledWith('x-ext1', { hard: true });
573
+ });
574
+ it('should throw for nonexistent node', async () => {
575
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
576
+ await expect(providerStore.providerDelete('s-nonexist')).rejects.toThrow('not found');
577
+ });
578
+ it('should throw when external node provider is not registered', async () => {
579
+ // External node exists but its provider (beads) is NOT registered
580
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockExternalNode);
581
+ await expect(providerStore.providerDelete('x-ext1')).rejects.toThrow('Provider not registered');
582
+ });
583
+ });
584
+ describe('defaultProvider', () => {
585
+ it('should default to native', () => {
586
+ const store = createProviderAwareStore(baseStore);
587
+ expect(store.defaultProvider).toBe('native');
588
+ });
589
+ it('should respect config', () => {
590
+ const store = createProviderAwareStore(baseStore, {
591
+ defaultProvider: 'sudocode',
592
+ });
593
+ expect(store.defaultProvider).toBe('sudocode');
594
+ });
595
+ });
596
+ // =========================================================================
597
+ // Task Lifecycle (TaskManageable)
598
+ // =========================================================================
599
+ describe('task lifecycle', () => {
600
+ let mockTaskProvider;
601
+ const mockTaskProviderNode = {
602
+ id: 'task-1',
603
+ uri: 'taskprov://task-1',
604
+ type: 'issue',
605
+ title: 'Task One',
606
+ status: 'in_progress',
607
+ fetchedAt: new Date().toISOString(),
608
+ };
609
+ const mockTaskExternalNode = {
610
+ id: 'x-task1',
611
+ uuid: 'uuid-task-1',
612
+ type: 'external',
613
+ title: 'Task One',
614
+ uri: 'taskprov://task-1',
615
+ source: 'taskprov',
616
+ materialized: true,
617
+ cached_at: new Date().toISOString(),
618
+ created_at: '2024-01-01T00:00:00.000Z',
619
+ updated_at: '2024-01-01T00:00:00.000Z',
620
+ };
621
+ beforeEach(() => {
622
+ mockTaskProvider = {
623
+ name: 'taskprov',
624
+ schemes: ['taskprov'],
625
+ capabilities: { read: true, write: true, search: false, watch: false, mount: true, feedback: false },
626
+ parseUri: vi.fn((uri) => {
627
+ if (uri.startsWith('taskprov://')) {
628
+ const id = uri.slice('taskprov://'.length);
629
+ return { scheme: 'taskprov', id, isRelative: false };
630
+ }
631
+ return null;
632
+ }),
633
+ buildUri: vi.fn((id) => `taskprov://${id}`),
634
+ isValidUri: vi.fn((uri) => uri.startsWith('taskprov://')),
635
+ get: vi.fn().mockResolvedValue(mockTaskProviderNode),
636
+ list: vi.fn().mockResolvedValue([]),
637
+ create: vi.fn(),
638
+ update: vi.fn(),
639
+ delete: vi.fn(),
640
+ taskCapabilities: {
641
+ actions: ['start', 'complete', 'block', 'reopen'],
642
+ supportsAssignment: true,
643
+ supportsReadyQuery: true,
644
+ statusModel: ['open', 'in_progress', 'blocked', 'done'],
645
+ },
646
+ transitionTask: vi.fn().mockResolvedValue(mockTaskProviderNode),
647
+ readyTasks: vi.fn().mockResolvedValue([mockTaskProviderNode]),
648
+ assignTask: vi.fn().mockResolvedValue(mockTaskProviderNode),
649
+ validActions: vi.fn().mockResolvedValue(['start', 'close']),
650
+ };
651
+ });
652
+ describe('taskTransition', () => {
653
+ it('should transition a task via provider URI', async () => {
654
+ providerStore.providers.register(mockTaskProvider);
655
+ // Mock materialization: no existing node, so create one
656
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
657
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
658
+ const result = await providerStore.taskTransition('taskprov://task-1', 'start');
659
+ expect(mockTaskProvider.transitionTask).toHaveBeenCalledWith('task-1', 'start', undefined);
660
+ expect(result.action).toBe('start');
661
+ expect(result.provider).toBe('taskprov');
662
+ expect(result.node).toBeDefined();
663
+ });
664
+ it('should transition a task via local external node ID', async () => {
665
+ providerStore.providers.register(mockTaskProvider);
666
+ vi.mocked(baseStore.getNode).mockResolvedValue(mockTaskExternalNode);
667
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockTaskExternalNode]);
668
+ vi.mocked(baseStore.updateNode).mockResolvedValue(mockTaskExternalNode);
669
+ const result = await providerStore.taskTransition('x-task1', 'complete');
670
+ expect(mockTaskProvider.transitionTask).toHaveBeenCalledWith('task-1', 'complete', undefined);
671
+ expect(result.action).toBe('complete');
672
+ });
673
+ it('should throw for provider that does not support tasks', async () => {
674
+ providerStore.providers.register(mockBeadsProvider);
675
+ await expect(providerStore.taskTransition('beads://./bd-123', 'start')).rejects.toThrow('does not support task operations');
676
+ });
677
+ it('should throw for nonexistent node', async () => {
678
+ vi.mocked(baseStore.getNode).mockResolvedValue(null);
679
+ await expect(providerStore.taskTransition('s-nonexist', 'start')).rejects.toThrow('not found');
680
+ });
681
+ it('should forward context to provider', async () => {
682
+ providerStore.providers.register(mockTaskProvider);
683
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
684
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
685
+ const ctx = { cwd: '/other/dir', timeout: 5000 };
686
+ await providerStore.taskTransition('taskprov://task-1', 'start', { context: ctx });
687
+ expect(mockTaskProvider.transitionTask).toHaveBeenCalledWith('task-1', 'start', ctx);
688
+ });
689
+ });
690
+ describe('taskReady', () => {
691
+ it('should aggregate ready tasks across task-capable providers', async () => {
692
+ providerStore.providers.register(mockTaskProvider);
693
+ // Mock materialization for each ready task
694
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
695
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
696
+ const ready = await providerStore.taskReady();
697
+ expect(mockTaskProvider.readyTasks).toHaveBeenCalled();
698
+ expect(ready.length).toBeGreaterThanOrEqual(1);
699
+ });
700
+ it('should skip non-task-capable providers', async () => {
701
+ providerStore.providers.register(mockBeadsProvider);
702
+ providerStore.providers.register(mockTaskProvider);
703
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
704
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
705
+ await providerStore.taskReady();
706
+ // Only taskprov should be queried
707
+ expect(mockTaskProvider.readyTasks).toHaveBeenCalled();
708
+ });
709
+ it('should filter by provider name', async () => {
710
+ providerStore.providers.register(mockTaskProvider);
711
+ await providerStore.taskReady({ providers: ['other-provider'] });
712
+ // taskprov should NOT be queried because it wasn't in the filter
713
+ expect(mockTaskProvider.readyTasks).not.toHaveBeenCalled();
714
+ });
715
+ it('should return existing materialized nodes when available', async () => {
716
+ providerStore.providers.register(mockTaskProvider);
717
+ // Mock: existing materialized node found
718
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([mockTaskExternalNode]);
719
+ const ready = await providerStore.taskReady();
720
+ expect(ready[0]).toEqual(mockTaskExternalNode);
721
+ });
722
+ it('should forward options to provider', async () => {
723
+ providerStore.providers.register(mockTaskProvider);
724
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
725
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
726
+ await providerStore.taskReady({ limit: 5, tags: ['urgent'], assignee: 'alice' });
727
+ expect(mockTaskProvider.readyTasks).toHaveBeenCalledWith({ limit: 5, tags: ['urgent'], priority: undefined, assignee: 'alice' }, undefined);
728
+ });
729
+ });
730
+ describe('taskAssign', () => {
731
+ it('should assign a task via provider', async () => {
732
+ providerStore.providers.register(mockTaskProvider);
733
+ vi.mocked(baseStore.query.nodes).mockResolvedValue([]);
734
+ vi.mocked(baseStore.createNode).mockResolvedValue(mockTaskExternalNode);
735
+ const result = await providerStore.taskAssign('taskprov://task-1', 'alice');
736
+ expect(mockTaskProvider.assignTask).toHaveBeenCalledWith('task-1', 'alice', undefined);
737
+ expect(result).toBeDefined();
738
+ });
739
+ it('should throw for provider that does not support assignment', async () => {
740
+ const noAssignProvider = {
741
+ ...mockTaskProvider,
742
+ name: 'noassign',
743
+ schemes: ['noassign'],
744
+ parseUri: vi.fn((uri) => {
745
+ if (uri.startsWith('noassign://')) {
746
+ return { scheme: 'noassign', id: uri.slice('noassign://'.length), isRelative: false };
747
+ }
748
+ return null;
749
+ }),
750
+ buildUri: vi.fn((id) => `noassign://${id}`),
751
+ isValidUri: vi.fn((uri) => uri.startsWith('noassign://')),
752
+ taskCapabilities: {
753
+ ...mockTaskProvider.taskCapabilities,
754
+ supportsAssignment: false,
755
+ },
756
+ assignTask: undefined,
757
+ };
758
+ providerStore.providers.register(noAssignProvider);
759
+ await expect(providerStore.taskAssign('noassign://task-1', 'alice')).rejects.toThrow('does not support task assignment');
760
+ });
761
+ });
762
+ describe('taskValidActions', () => {
763
+ it('should return valid actions from provider', async () => {
764
+ providerStore.providers.register(mockTaskProvider);
765
+ const actions = await providerStore.taskValidActions('taskprov://task-1');
766
+ expect(mockTaskProvider.validActions).toHaveBeenCalledWith('task-1');
767
+ expect(actions).toEqual(['start', 'close']);
768
+ });
769
+ it('should fall back to taskCapabilities.actions when validActions not implemented', async () => {
770
+ const noValidActionsProvider = {
771
+ ...mockTaskProvider,
772
+ name: 'novalid',
773
+ schemes: ['novalid'],
774
+ parseUri: vi.fn((uri) => {
775
+ if (uri.startsWith('novalid://')) {
776
+ return { scheme: 'novalid', id: uri.slice('novalid://'.length), isRelative: false };
777
+ }
778
+ return null;
779
+ }),
780
+ buildUri: vi.fn((id) => `novalid://${id}`),
781
+ isValidUri: vi.fn((uri) => uri.startsWith('novalid://')),
782
+ validActions: undefined,
783
+ };
784
+ providerStore.providers.register(noValidActionsProvider);
785
+ const actions = await providerStore.taskValidActions('novalid://task-1');
786
+ expect(actions).toEqual(['start', 'complete', 'block', 'reopen']);
787
+ });
788
+ });
789
+ });
790
+ });
791
+ //# sourceMappingURL=provider-store.test.js.map