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
package/README.md CHANGED
@@ -158,6 +158,61 @@ graph LR
158
158
 
159
159
  External nodes start as bare URIs. When you query them, OpenTasks fetches the data from the provider and caches it locally.
160
160
 
161
+ ## Context Sources
162
+
163
+ Context nodes support multiple content source types. **Inline** contexts store content directly. **File-backed** contexts are lightweight pointers to codebase files — content is resolved on access with git-based drift detection.
164
+
165
+ ```typescript
166
+ // Inline context — content stored in the node
167
+ const spec = await client.createNode({
168
+ type: 'context',
169
+ title: 'OAuth2 for API',
170
+ content: '## Requirements\n- Google OAuth2 with PKCE\n...',
171
+ })
172
+
173
+ // File-backed context — pointer only, content resolved on demand
174
+ const fileCtx = await client.createContextFile({
175
+ filePath: 'docs/auth-architecture.md',
176
+ tags: ['auth'],
177
+ })
178
+ // → { id: 'c-x7k9', metadata: { context_file: true, context_file_path: '...', context_file_commit: 'abc123', ... } }
179
+
180
+ // Resolve content from the working tree (includes drift detection)
181
+ const resolved = await client.resolveContextFile('c-x7k9')
182
+ // → { content: '# Auth Architecture\n...', drifted: true, commit: 'def456', ... }
183
+
184
+ // Re-pin to current HEAD after changes
185
+ await client.syncContextFile('c-x7k9')
186
+ ```
187
+
188
+ Via MCP (`--scope context`):
189
+
190
+ ```json
191
+ // Create file-backed context
192
+ { "tool": "create_context", "source": { "type": "file", "path": "docs/auth-architecture.md" } }
193
+
194
+ // Get with resolved file content
195
+ { "tool": "get_context", "id": "c-x7k9", "resolve": true }
196
+
197
+ // Sync to current HEAD
198
+ { "tool": "update_context", "id": "c-x7k9", "sync": true }
199
+
200
+ // Batch drift check
201
+ { "tool": "list_contexts", "filesOnly": true, "checkDrift": true }
202
+ ```
203
+
204
+ File-backed contexts never duplicate file content into the graph store. They record a content hash and git commit SHA at capture time, then detect drift by comparing the current file against that snapshot.
205
+
206
+ ## MCP Server
207
+
208
+ Expose the full tool interface via [Model Context Protocol](https://modelcontextprotocol.io):
209
+
210
+ ```bash
211
+ opentasks mcp --scope tasks,graph,annotate,context
212
+ ```
213
+
214
+ 15 tools across 4 scopes: `tasks` (CRUD + lifecycle), `graph` (edges, queries, context summary), `annotate` (feedback), `context` (context CRUD with file/snippet/inline sources).
215
+
161
216
  ## Programmatic API
162
217
 
163
218
  For direct graph manipulation without the daemon:
@@ -262,6 +317,19 @@ graph LR
262
317
 
263
318
  No upfront API calls. References stay cheap until you need the data.
264
319
 
320
+ ### Provider Caching & Reconciliation
321
+
322
+ Provider-backed nodes cache data locally but treat the provider as the source of truth. When `graph.jsonl` is git-synced across environments, cached data can diverge from the provider's current state.
323
+
324
+ OpenTasks reconciles automatically:
325
+ - On file watcher reload (git pull, branch switch) — re-fetches provider-backed nodes
326
+ - Positive-writes-only — never deletes or archives nodes when a provider is unavailable
327
+ - Edge reconciliation — provider relationships extracted from node data, no extra API calls
328
+
329
+ Providers can opt into **pointer-only mode** (`materializeMode: 'pointer'`) where only the URI reference is stored and data is resolved transparently on every access.
330
+
331
+ See [docs/PROVIDER-RECONCILIATION.md](./docs/PROVIDER-RECONCILIATION.md) for the full design.
332
+
265
333
  ## Locations
266
334
 
267
335
  Multiple `.opentasks/` directories at different filesystem levels. Each is isolated by default.
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Tests for CLI Tool Commands
3
+ *
4
+ * Tests link, query, annotate, create, get, update, delete commands
5
+ * by calling the exported functions with a mocked OpenTasksClient.
6
+ */
7
+ export {};
8
+ //# sourceMappingURL=cli-tools.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli-tools.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/cli-tools.test.ts"],"names":[],"mappings":"AAAA;;;;;GAKG"}
@@ -0,0 +1,546 @@
1
+ /**
2
+ * Tests for CLI Tool Commands
3
+ *
4
+ * Tests link, query, annotate, create, get, update, delete commands
5
+ * by calling the exported functions with a mocked OpenTasksClient.
6
+ */
7
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
8
+ // Mock the client module before importing CLI functions
9
+ const mockClient = {
10
+ link: vi.fn(),
11
+ query: vi.fn(),
12
+ annotate: vi.fn(),
13
+ createNode: vi.fn(),
14
+ getNode: vi.fn(),
15
+ updateNode: vi.fn(),
16
+ deleteNode: vi.fn(),
17
+ disconnect: vi.fn(),
18
+ };
19
+ vi.mock('../client/client.js', () => ({
20
+ OpenTasksClient: vi.fn().mockImplementation(function () {
21
+ return mockClient;
22
+ }),
23
+ }));
24
+ // Mock process.exit to throw instead of exiting
25
+ vi.spyOn(process, 'exit').mockImplementation(((code) => {
26
+ throw new Error(`process.exit(${code})`);
27
+ }));
28
+ import { cmdLink, cmdQuery, cmdAnnotate, cmdCreate, cmdGet, cmdUpdate, cmdDelete, getFlag, hasFlag, } from '../cli.js';
29
+ // ============================================================================
30
+ // Helpers
31
+ // ============================================================================
32
+ let logOutput;
33
+ let errorOutput;
34
+ let logSpy;
35
+ let errorSpy;
36
+ function getLoggedJson() {
37
+ const json = logOutput.join('');
38
+ return JSON.parse(json);
39
+ }
40
+ // ============================================================================
41
+ // Tests
42
+ // ============================================================================
43
+ describe('CLI Tool Commands', () => {
44
+ beforeEach(() => {
45
+ vi.clearAllMocks();
46
+ logOutput = [];
47
+ errorOutput = [];
48
+ logSpy = vi.spyOn(console, 'log').mockImplementation((...args) => {
49
+ logOutput.push(args.map(String).join(' '));
50
+ });
51
+ errorSpy = vi.spyOn(console, 'error').mockImplementation((...args) => {
52
+ errorOutput.push(args.map(String).join(' '));
53
+ });
54
+ });
55
+ afterEach(() => {
56
+ logSpy.mockRestore();
57
+ errorSpy.mockRestore();
58
+ });
59
+ // ==========================================================================
60
+ // Helper functions
61
+ // ==========================================================================
62
+ describe('getFlag', () => {
63
+ it('should return flag value', () => {
64
+ expect(getFlag(['--from', 'i-test1', '--to', 's-test1'], '--from')).toBe('i-test1');
65
+ expect(getFlag(['--from', 'i-test1', '--to', 's-test1'], '--to')).toBe('s-test1');
66
+ });
67
+ it('should return undefined for missing flag', () => {
68
+ expect(getFlag(['--from', 'i-test1'], '--to')).toBeUndefined();
69
+ });
70
+ it('should return undefined for empty args', () => {
71
+ expect(getFlag([], '--from')).toBeUndefined();
72
+ });
73
+ });
74
+ describe('hasFlag', () => {
75
+ it('should return true when flag present', () => {
76
+ expect(hasFlag(['--remove', '--from', 'x'], '--remove')).toBe(true);
77
+ });
78
+ it('should return false when flag absent', () => {
79
+ expect(hasFlag(['--from', 'x'], '--remove')).toBe(false);
80
+ });
81
+ });
82
+ // ==========================================================================
83
+ // link
84
+ // ==========================================================================
85
+ describe('cmdLink', () => {
86
+ it('should create an edge between nodes', async () => {
87
+ mockClient.link.mockResolvedValueOnce({ success: true, edgeId: 'x-new1' });
88
+ await cmdLink(['--from', 'i-test1', '--to', 's-test1', '--type', 'implements']);
89
+ expect(mockClient.link).toHaveBeenCalledWith({
90
+ fromId: 'i-test1',
91
+ toId: 's-test1',
92
+ type: 'implements',
93
+ });
94
+ const result = getLoggedJson();
95
+ expect(result.success).toBe(true);
96
+ expect(result.edgeId).toBe('x-new1');
97
+ });
98
+ it('should remove an edge with --remove', async () => {
99
+ mockClient.link.mockResolvedValueOnce({ success: true });
100
+ await cmdLink(['--from', 'i-test1', '--to', 's-test1', '--type', 'implements', '--remove']);
101
+ expect(mockClient.link).toHaveBeenCalledWith({
102
+ fromId: 'i-test1',
103
+ toId: 's-test1',
104
+ type: 'implements',
105
+ remove: true,
106
+ });
107
+ });
108
+ it('should pass metadata as parsed JSON', async () => {
109
+ mockClient.link.mockResolvedValueOnce({ success: true, edgeId: 'x-new2' });
110
+ await cmdLink([
111
+ '--from', 'i-test1', '--to', 's-test1', '--type', 'references',
112
+ '--metadata', '{"context":"bug report"}',
113
+ ]);
114
+ expect(mockClient.link).toHaveBeenCalledWith({
115
+ fromId: 'i-test1',
116
+ toId: 's-test1',
117
+ type: 'references',
118
+ metadata: { context: 'bug report' },
119
+ });
120
+ });
121
+ it('should disconnect after success', async () => {
122
+ mockClient.link.mockResolvedValueOnce({ success: true });
123
+ await cmdLink(['--from', 'i-test1', '--to', 's-test1', '--type', 'implements']);
124
+ expect(mockClient.disconnect).toHaveBeenCalled();
125
+ });
126
+ it('should error on missing --from', async () => {
127
+ await expect(cmdLink(['--to', 's-test1', '--type', 'implements'])).rejects.toThrow('process.exit(1)');
128
+ expect(errorOutput.join('')).toContain('Usage:');
129
+ expect(mockClient.link).not.toHaveBeenCalled();
130
+ });
131
+ it('should error on missing --to', async () => {
132
+ await expect(cmdLink(['--from', 'i-test1', '--type', 'implements'])).rejects.toThrow('process.exit(1)');
133
+ expect(errorOutput.join('')).toContain('Usage:');
134
+ });
135
+ it('should error on missing --type', async () => {
136
+ await expect(cmdLink(['--from', 'i-test1', '--to', 's-test1'])).rejects.toThrow('process.exit(1)');
137
+ expect(errorOutput.join('')).toContain('Usage:');
138
+ });
139
+ it('should handle RPC errors', async () => {
140
+ mockClient.link.mockRejectedValueOnce(new Error('Connection refused'));
141
+ await expect(cmdLink(['--from', 'i-test1', '--to', 's-test1', '--type', 'blocks'])).rejects.toThrow('process.exit(1)');
142
+ const errorJson = JSON.parse(errorOutput.join(''));
143
+ expect(errorJson.error).toContain('Connection refused');
144
+ });
145
+ });
146
+ // ==========================================================================
147
+ // query
148
+ // ==========================================================================
149
+ describe('cmdQuery', () => {
150
+ it('should query ready work', async () => {
151
+ mockClient.query.mockResolvedValueOnce({
152
+ items: [{ id: 'i-1', title: 'Ready Issue' }],
153
+ hasMore: false,
154
+ });
155
+ await cmdQuery(['{"ready":{}}']);
156
+ expect(mockClient.query).toHaveBeenCalledWith({ ready: {} });
157
+ const result = getLoggedJson();
158
+ expect(result.items).toHaveLength(1);
159
+ expect(result.items[0].id).toBe('i-1');
160
+ });
161
+ it('should query nodes with filter', async () => {
162
+ mockClient.query.mockResolvedValueOnce({
163
+ items: [{ id: 's-1' }],
164
+ hasMore: false,
165
+ });
166
+ await cmdQuery(['{"nodes":{"type":"spec"}}']);
167
+ expect(mockClient.query).toHaveBeenCalledWith({ nodes: { type: 'spec' } });
168
+ });
169
+ it('should query edges', async () => {
170
+ mockClient.query.mockResolvedValueOnce({
171
+ items: [{ id: 'x-1', from_id: 'i-1', to_id: 's-1' }],
172
+ hasMore: false,
173
+ });
174
+ await cmdQuery(['{"edges":{"from_id":"i-1"}}']);
175
+ expect(mockClient.query).toHaveBeenCalledWith({ edges: { from_id: 'i-1' } });
176
+ });
177
+ it('should query blockers with options', async () => {
178
+ mockClient.query.mockResolvedValueOnce({
179
+ items: [{ id: 's-1', title: 'Blocker' }],
180
+ hasMore: false,
181
+ });
182
+ await cmdQuery(['{"blockers":{"nodeId":"i-1","activeOnly":true,"transitive":true}}']);
183
+ expect(mockClient.query).toHaveBeenCalledWith({
184
+ blockers: { nodeId: 'i-1', activeOnly: true, transitive: true },
185
+ });
186
+ });
187
+ it('should query implementers', async () => {
188
+ mockClient.query.mockResolvedValueOnce({
189
+ items: [{ id: 'i-1' }],
190
+ hasMore: false,
191
+ });
192
+ await cmdQuery(['{"implementers":{"specId":"s-1"}}']);
193
+ expect(mockClient.query).toHaveBeenCalledWith({ implementers: { specId: 's-1' } });
194
+ });
195
+ it('should query feedback', async () => {
196
+ mockClient.query.mockResolvedValueOnce({
197
+ items: [{ id: 'f-1' }],
198
+ hasMore: false,
199
+ });
200
+ await cmdQuery(['{"feedback":{"nodeId":"s-1","resolved":false}}']);
201
+ expect(mockClient.query).toHaveBeenCalledWith({
202
+ feedback: { nodeId: 's-1', resolved: false },
203
+ });
204
+ });
205
+ it('should query unresolved feedback', async () => {
206
+ mockClient.query.mockResolvedValueOnce({ items: [], hasMore: false });
207
+ await cmdQuery(['{"unresolvedFeedback":{}}']);
208
+ expect(mockClient.query).toHaveBeenCalledWith({ unresolvedFeedback: {} });
209
+ });
210
+ it('should disconnect after success', async () => {
211
+ mockClient.query.mockResolvedValueOnce({ items: [], hasMore: false });
212
+ await cmdQuery(['{"ready":{}}']);
213
+ expect(mockClient.disconnect).toHaveBeenCalled();
214
+ });
215
+ it('should error on missing JSON argument', async () => {
216
+ await expect(cmdQuery([])).rejects.toThrow('process.exit(1)');
217
+ expect(errorOutput.join('')).toContain('Usage:');
218
+ expect(mockClient.query).not.toHaveBeenCalled();
219
+ });
220
+ it('should error on invalid JSON', async () => {
221
+ await expect(cmdQuery(['not-valid-json'])).rejects.toThrow('process.exit(1)');
222
+ const errorJson = JSON.parse(errorOutput.join(''));
223
+ expect(errorJson.error).toBeDefined();
224
+ });
225
+ });
226
+ // ==========================================================================
227
+ // annotate
228
+ // ==========================================================================
229
+ describe('cmdAnnotate', () => {
230
+ it('should create feedback', async () => {
231
+ mockClient.annotate.mockResolvedValueOnce({ success: true, feedbackId: 'f-new1' });
232
+ await cmdAnnotate(['{"targetId":"s-1","create":{"content":"Nice work","type":"comment"}}']);
233
+ expect(mockClient.annotate).toHaveBeenCalledWith({
234
+ targetId: 's-1',
235
+ create: { content: 'Nice work', type: 'comment' },
236
+ });
237
+ const result = getLoggedJson();
238
+ expect(result.success).toBe(true);
239
+ expect(result.feedbackId).toBe('f-new1');
240
+ });
241
+ it('should create feedback with fromId', async () => {
242
+ mockClient.annotate.mockResolvedValueOnce({ success: true, feedbackId: 'f-new2' });
243
+ await cmdAnnotate(['{"targetId":"s-1","fromId":"i-1","create":{"content":"Done","type":"comment"}}']);
244
+ expect(mockClient.annotate).toHaveBeenCalledWith({
245
+ targetId: 's-1',
246
+ fromId: 'i-1',
247
+ create: { content: 'Done', type: 'comment' },
248
+ });
249
+ });
250
+ it('should create anchored suggestion', async () => {
251
+ mockClient.annotate.mockResolvedValueOnce({ success: true, feedbackId: 'f-new3' });
252
+ await cmdAnnotate(['{"targetId":"s-1","create":{"content":"Add rate limiting","type":"suggestion","anchor":{"line":15}}}']);
253
+ expect(mockClient.annotate).toHaveBeenCalledWith({
254
+ targetId: 's-1',
255
+ create: { content: 'Add rate limiting', type: 'suggestion', anchor: { line: 15 } },
256
+ });
257
+ });
258
+ it('should resolve feedback', async () => {
259
+ mockClient.annotate.mockResolvedValueOnce({ success: true });
260
+ await cmdAnnotate(['{"targetId":"s-1","resolve":"f-1"}']);
261
+ expect(mockClient.annotate).toHaveBeenCalledWith({
262
+ targetId: 's-1',
263
+ resolve: 'f-1',
264
+ });
265
+ });
266
+ it('should dismiss feedback', async () => {
267
+ mockClient.annotate.mockResolvedValueOnce({ success: true });
268
+ await cmdAnnotate(['{"targetId":"s-1","dismiss":"f-1"}']);
269
+ expect(mockClient.annotate).toHaveBeenCalledWith({
270
+ targetId: 's-1',
271
+ dismiss: 'f-1',
272
+ });
273
+ });
274
+ it('should reopen feedback', async () => {
275
+ mockClient.annotate.mockResolvedValueOnce({ success: true });
276
+ await cmdAnnotate(['{"targetId":"s-1","reopen":"f-1"}']);
277
+ expect(mockClient.annotate).toHaveBeenCalledWith({
278
+ targetId: 's-1',
279
+ reopen: 'f-1',
280
+ });
281
+ });
282
+ it('should error on missing JSON argument', async () => {
283
+ await expect(cmdAnnotate([])).rejects.toThrow('process.exit(1)');
284
+ expect(errorOutput.join('')).toContain('Usage:');
285
+ });
286
+ it('should error on invalid JSON', async () => {
287
+ await expect(cmdAnnotate(['{bad json}'])).rejects.toThrow('process.exit(1)');
288
+ });
289
+ it('should handle RPC errors', async () => {
290
+ mockClient.annotate.mockRejectedValueOnce(new Error('Target not found'));
291
+ await expect(cmdAnnotate(['{"targetId":"s-nonexistent","create":{"content":"x","type":"comment"}}'])).rejects.toThrow('process.exit(1)');
292
+ const errorJson = JSON.parse(errorOutput.join(''));
293
+ expect(errorJson.error).toContain('Target not found');
294
+ });
295
+ });
296
+ // ==========================================================================
297
+ // create
298
+ // ==========================================================================
299
+ describe('cmdCreate', () => {
300
+ it('should create an issue node', async () => {
301
+ mockClient.createNode.mockResolvedValueOnce({
302
+ id: 'i-new1', type: 'issue', title: 'New Issue', status: 'open',
303
+ });
304
+ await cmdCreate(['--type', 'issue', '--title', 'New Issue', '--status', 'open']);
305
+ expect(mockClient.createNode).toHaveBeenCalledWith({
306
+ type: 'issue',
307
+ title: 'New Issue',
308
+ status: 'open',
309
+ });
310
+ const result = getLoggedJson();
311
+ expect(result.id).toBe('i-new1');
312
+ expect(result.type).toBe('issue');
313
+ });
314
+ it('should create a spec node', async () => {
315
+ mockClient.createNode.mockResolvedValueOnce({
316
+ id: 's-new1', type: 'spec', title: 'Auth Spec', content: '## Reqs',
317
+ });
318
+ await cmdCreate(['--type', 'spec', '--title', 'Auth Spec', '--content', '## Reqs']);
319
+ expect(mockClient.createNode).toHaveBeenCalledWith({
320
+ type: 'spec',
321
+ title: 'Auth Spec',
322
+ content: '## Reqs',
323
+ });
324
+ });
325
+ it('should create an external node with uri and source', async () => {
326
+ mockClient.createNode.mockResolvedValueOnce({
327
+ id: 'e-new1', type: 'external', title: 'Slack Msg',
328
+ uri: 'slack://C04ABCD/p123', source: 'slack',
329
+ });
330
+ await cmdCreate([
331
+ '--type', 'external', '--title', 'Slack Msg',
332
+ '--uri', 'slack://C04ABCD/p123', '--source', 'slack',
333
+ ]);
334
+ expect(mockClient.createNode).toHaveBeenCalledWith({
335
+ type: 'external',
336
+ title: 'Slack Msg',
337
+ uri: 'slack://C04ABCD/p123',
338
+ source: 'slack',
339
+ });
340
+ });
341
+ it('should parse comma-separated tags', async () => {
342
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
343
+ await cmdCreate(['--type', 'issue', '--title', 'Tagged', '--tags', 'auth,bug,urgent']);
344
+ expect(mockClient.createNode).toHaveBeenCalledWith(expect.objectContaining({
345
+ tags: ['auth', 'bug', 'urgent'],
346
+ }));
347
+ });
348
+ it('should trim whitespace from tags', async () => {
349
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
350
+ await cmdCreate(['--type', 'issue', '--title', 'Tagged', '--tags', 'auth, bug, urgent']);
351
+ expect(mockClient.createNode).toHaveBeenCalledWith(expect.objectContaining({
352
+ tags: ['auth', 'bug', 'urgent'],
353
+ }));
354
+ });
355
+ it('should parse priority as number', async () => {
356
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
357
+ await cmdCreate(['--type', 'issue', '--title', 'Prio', '--priority', '2']);
358
+ expect(mockClient.createNode).toHaveBeenCalledWith(expect.objectContaining({ priority: 2 }));
359
+ });
360
+ it('should pass metadata as parsed JSON', async () => {
361
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
362
+ await cmdCreate([
363
+ '--type', 'issue', '--title', 'Meta',
364
+ '--metadata', '{"source_url":"https://example.com"}',
365
+ ]);
366
+ expect(mockClient.createNode).toHaveBeenCalledWith(expect.objectContaining({
367
+ metadata: { source_url: 'https://example.com' },
368
+ }));
369
+ });
370
+ it('should pass parent ID', async () => {
371
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
372
+ await cmdCreate(['--type', 'issue', '--title', 'Child', '--parent', 's-test1']);
373
+ expect(mockClient.createNode).toHaveBeenCalledWith(expect.objectContaining({ parent_id: 's-test1' }));
374
+ });
375
+ it('should only include specified options', async () => {
376
+ mockClient.createNode.mockResolvedValueOnce({ id: 'i-new1' });
377
+ await cmdCreate(['--type', 'issue', '--title', 'Minimal']);
378
+ expect(mockClient.createNode).toHaveBeenCalledWith({
379
+ type: 'issue',
380
+ title: 'Minimal',
381
+ });
382
+ });
383
+ it('should error on missing --type', async () => {
384
+ await expect(cmdCreate(['--title', 'No Type'])).rejects.toThrow('process.exit(1)');
385
+ expect(errorOutput.join('')).toContain('Usage:');
386
+ expect(mockClient.createNode).not.toHaveBeenCalled();
387
+ });
388
+ it('should error on missing --title', async () => {
389
+ await expect(cmdCreate(['--type', 'issue'])).rejects.toThrow('process.exit(1)');
390
+ expect(errorOutput.join('')).toContain('Usage:');
391
+ });
392
+ });
393
+ // ==========================================================================
394
+ // get
395
+ // ==========================================================================
396
+ describe('cmdGet', () => {
397
+ it('should get a node by ID', async () => {
398
+ mockClient.getNode.mockResolvedValueOnce({
399
+ id: 's-test1', type: 'spec', title: 'Test Spec',
400
+ });
401
+ await cmdGet(['s-test1']);
402
+ expect(mockClient.getNode).toHaveBeenCalledWith('s-test1');
403
+ const result = getLoggedJson();
404
+ expect(result.id).toBe('s-test1');
405
+ expect(result.title).toBe('Test Spec');
406
+ });
407
+ it('should return null for non-existent node', async () => {
408
+ mockClient.getNode.mockResolvedValueOnce(null);
409
+ await cmdGet(['i-nonexistent']);
410
+ expect(mockClient.getNode).toHaveBeenCalledWith('i-nonexistent');
411
+ const result = getLoggedJson();
412
+ expect(result).toBeNull();
413
+ });
414
+ it('should disconnect after success', async () => {
415
+ mockClient.getNode.mockResolvedValueOnce({ id: 's-1' });
416
+ await cmdGet(['s-1']);
417
+ expect(mockClient.disconnect).toHaveBeenCalled();
418
+ });
419
+ it('should error on missing ID', async () => {
420
+ await expect(cmdGet([])).rejects.toThrow('process.exit(1)');
421
+ expect(errorOutput.join('')).toContain('Usage:');
422
+ expect(mockClient.getNode).not.toHaveBeenCalled();
423
+ });
424
+ });
425
+ // ==========================================================================
426
+ // update
427
+ // ==========================================================================
428
+ describe('cmdUpdate', () => {
429
+ it('should update status', async () => {
430
+ mockClient.updateNode.mockResolvedValueOnce({
431
+ id: 'i-test1', status: 'closed',
432
+ });
433
+ await cmdUpdate(['i-test1', '--status', 'closed']);
434
+ expect(mockClient.updateNode).toHaveBeenCalledWith('i-test1', {
435
+ status: 'closed',
436
+ });
437
+ const result = getLoggedJson();
438
+ expect(result.status).toBe('closed');
439
+ });
440
+ it('should update title', async () => {
441
+ mockClient.updateNode.mockResolvedValueOnce({
442
+ id: 's-test1', title: 'Updated Title',
443
+ });
444
+ await cmdUpdate(['s-test1', '--title', 'Updated Title']);
445
+ expect(mockClient.updateNode).toHaveBeenCalledWith('s-test1', {
446
+ title: 'Updated Title',
447
+ });
448
+ });
449
+ it('should set archived flag', async () => {
450
+ mockClient.updateNode.mockResolvedValueOnce({
451
+ id: 'i-test1', archived: true,
452
+ });
453
+ await cmdUpdate(['i-test1', '--archived']);
454
+ expect(mockClient.updateNode).toHaveBeenCalledWith('i-test1', {
455
+ archived: true,
456
+ });
457
+ });
458
+ it('should update metadata as parsed JSON', async () => {
459
+ mockClient.updateNode.mockResolvedValueOnce({
460
+ id: 'i-test1', metadata: { reviewed: true },
461
+ });
462
+ await cmdUpdate(['i-test1', '--metadata', '{"reviewed":true}']);
463
+ expect(mockClient.updateNode).toHaveBeenCalledWith('i-test1', {
464
+ metadata: { reviewed: true },
465
+ });
466
+ });
467
+ it('should update multiple fields at once', async () => {
468
+ mockClient.updateNode.mockResolvedValueOnce({
469
+ id: 'i-test1', status: 'in_progress', title: 'Renamed',
470
+ });
471
+ await cmdUpdate(['i-test1', '--status', 'in_progress', '--title', 'Renamed']);
472
+ expect(mockClient.updateNode).toHaveBeenCalledWith('i-test1', {
473
+ status: 'in_progress',
474
+ title: 'Renamed',
475
+ });
476
+ });
477
+ it('should error on missing ID', async () => {
478
+ await expect(cmdUpdate([])).rejects.toThrow('process.exit(1)');
479
+ expect(errorOutput.join('')).toContain('Usage:');
480
+ expect(mockClient.updateNode).not.toHaveBeenCalled();
481
+ });
482
+ it('should error when no updates specified', async () => {
483
+ await expect(cmdUpdate(['i-test1'])).rejects.toThrow('process.exit(1)');
484
+ expect(errorOutput.join('')).toContain('No updates specified');
485
+ expect(mockClient.updateNode).not.toHaveBeenCalled();
486
+ });
487
+ });
488
+ // ==========================================================================
489
+ // delete
490
+ // ==========================================================================
491
+ describe('cmdDelete', () => {
492
+ it('should soft delete a node', async () => {
493
+ mockClient.deleteNode.mockResolvedValueOnce(undefined);
494
+ await cmdDelete(['i-test1']);
495
+ expect(mockClient.deleteNode).toHaveBeenCalledWith('i-test1', undefined);
496
+ const result = getLoggedJson();
497
+ expect(result.success).toBe(true);
498
+ expect(result.id).toBe('i-test1');
499
+ expect(result.hard).toBe(false);
500
+ });
501
+ it('should hard delete with --hard', async () => {
502
+ mockClient.deleteNode.mockResolvedValueOnce(undefined);
503
+ await cmdDelete(['i-test1', '--hard']);
504
+ expect(mockClient.deleteNode).toHaveBeenCalledWith('i-test1', { hard: true });
505
+ const result = getLoggedJson();
506
+ expect(result.success).toBe(true);
507
+ expect(result.hard).toBe(true);
508
+ });
509
+ it('should disconnect after success', async () => {
510
+ mockClient.deleteNode.mockResolvedValueOnce(undefined);
511
+ await cmdDelete(['i-test1']);
512
+ expect(mockClient.disconnect).toHaveBeenCalled();
513
+ });
514
+ it('should error on missing ID', async () => {
515
+ await expect(cmdDelete([])).rejects.toThrow('process.exit(1)');
516
+ expect(errorOutput.join('')).toContain('Usage:');
517
+ expect(mockClient.deleteNode).not.toHaveBeenCalled();
518
+ });
519
+ it('should handle RPC errors', async () => {
520
+ mockClient.deleteNode.mockRejectedValueOnce(new Error('Node not found'));
521
+ await expect(cmdDelete(['i-nonexistent'])).rejects.toThrow('process.exit(1)');
522
+ const errorJson = JSON.parse(errorOutput.join(''));
523
+ expect(errorJson.error).toContain('Node not found');
524
+ });
525
+ });
526
+ // ==========================================================================
527
+ // JSON output formatting
528
+ // ==========================================================================
529
+ describe('output formatting', () => {
530
+ it('should output pretty-printed JSON to stdout', async () => {
531
+ mockClient.getNode.mockResolvedValueOnce({ id: 's-1', title: 'Test' });
532
+ await cmdGet(['s-1']);
533
+ // runToolCommand uses JSON.stringify with 2-space indent
534
+ const raw = logOutput.join('');
535
+ expect(raw).toContain('{\n');
536
+ expect(raw).toContain('"id": "s-1"');
537
+ });
538
+ it('should output error JSON to stderr on RPC failure', async () => {
539
+ mockClient.getNode.mockRejectedValueOnce(new Error('Timeout'));
540
+ await expect(cmdGet(['s-1'])).rejects.toThrow('process.exit(1)');
541
+ const errorJson = JSON.parse(errorOutput.join(''));
542
+ expect(errorJson).toEqual({ error: 'Timeout' });
543
+ });
544
+ });
545
+ });
546
+ //# sourceMappingURL=cli-tools.test.js.map