aimux-cli 0.1.16 → 0.1.19

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 (357) hide show
  1. package/README.md +184 -67
  2. package/bin/aimux-dev +10 -0
  3. package/dist/agent-events.js +0 -1
  4. package/dist/agent-output-parser.js +0 -1
  5. package/dist/agent-prompt-delivery.js +0 -1
  6. package/dist/agent-tracker.js +0 -1
  7. package/dist/agent-watcher.js +0 -1
  8. package/dist/alert-display.d.ts +21 -0
  9. package/dist/alert-display.js +85 -0
  10. package/dist/atomic-write.js +0 -1
  11. package/dist/attachment-store.d.ts +0 -7
  12. package/dist/attachment-store.js +2 -87
  13. package/dist/builtin-metadata-watchers.js +4 -5
  14. package/dist/claude-hooks.d.ts +1 -0
  15. package/dist/claude-hooks.js +25 -1
  16. package/dist/config.d.ts +19 -13
  17. package/dist/config.js +28 -15
  18. package/dist/connection-targets.d.ts +8 -0
  19. package/dist/connection-targets.js +27 -0
  20. package/dist/context/compactor.js +0 -1
  21. package/dist/context/context-bridge.js +0 -1
  22. package/dist/context/context-file.js +0 -1
  23. package/dist/context/history.js +0 -1
  24. package/dist/credentials.d.ts +12 -0
  25. package/dist/credentials.js +48 -0
  26. package/dist/daemon.d.ts +23 -0
  27. package/dist/daemon.js +391 -67
  28. package/dist/dashboard/command-spec.js +0 -1
  29. package/dist/dashboard/feedback.js +0 -1
  30. package/dist/dashboard/index.d.ts +13 -10
  31. package/dist/dashboard/index.js +3 -27
  32. package/dist/dashboard/operation-failures.js +0 -1
  33. package/dist/dashboard/order.d.ts +22 -0
  34. package/dist/dashboard/order.js +54 -0
  35. package/dist/dashboard/pending-actions.d.ts +39 -10
  36. package/dist/dashboard/pending-actions.js +166 -37
  37. package/dist/dashboard/quick-jump.d.ts +2 -1
  38. package/dist/dashboard/quick-jump.js +7 -5
  39. package/dist/dashboard/runtime-evidence.js +0 -1
  40. package/dist/dashboard/session-actions.d.ts +4 -4
  41. package/dist/dashboard/session-actions.js +1 -2
  42. package/dist/dashboard/session-registry.d.ts +4 -3
  43. package/dist/dashboard/session-registry.js +16 -51
  44. package/dist/dashboard/sort.js +0 -1
  45. package/dist/dashboard/state.d.ts +1 -1
  46. package/dist/dashboard/state.js +0 -1
  47. package/dist/dashboard/targets.js +0 -1
  48. package/dist/dashboard/ui-state-store.d.ts +16 -1
  49. package/dist/dashboard/ui-state-store.js +73 -3
  50. package/dist/debug-state.d.ts +97 -0
  51. package/dist/debug-state.js +540 -0
  52. package/dist/debug.d.ts +38 -0
  53. package/dist/debug.js +219 -16
  54. package/dist/default-plugins/gh-pr-context.d.ts +2 -1
  55. package/dist/default-plugins/gh-pr-context.js +17 -12
  56. package/dist/default-plugins/transcript-length.js +15 -3
  57. package/dist/fast-control.js +37 -20
  58. package/dist/hotkeys.js +0 -1
  59. package/dist/http-client.js +31 -3
  60. package/dist/key-parser.js +0 -1
  61. package/dist/last-used.js +0 -1
  62. package/dist/local-ui-server.d.ts +22 -0
  63. package/dist/local-ui-server.js +185 -0
  64. package/dist/login-flow.d.ts +7 -0
  65. package/dist/login-flow.js +119 -0
  66. package/dist/main.js +821 -152
  67. package/dist/managed-launch-env.js +14 -1
  68. package/dist/metadata-server.d.ts +36 -36
  69. package/dist/metadata-server.js +638 -138
  70. package/dist/metadata-store.d.ts +4 -1
  71. package/dist/metadata-store.js +30 -3
  72. package/dist/multiplexer/agent-io-methods.d.ts +2 -10
  73. package/dist/multiplexer/agent-io-methods.js +12 -44
  74. package/dist/multiplexer/archives.js +8 -10
  75. package/dist/multiplexer/dashboard-actions-methods.js +0 -1
  76. package/dist/multiplexer/dashboard-control.js +45 -14
  77. package/dist/multiplexer/dashboard-interaction.d.ts +8 -2
  78. package/dist/multiplexer/dashboard-interaction.js +187 -29
  79. package/dist/multiplexer/dashboard-model.d.ts +10 -3
  80. package/dist/multiplexer/dashboard-model.js +417 -36
  81. package/dist/multiplexer/dashboard-ops.d.ts +9 -7
  82. package/dist/multiplexer/dashboard-ops.js +178 -69
  83. package/dist/multiplexer/dashboard-state-methods.d.ts +2 -1
  84. package/dist/multiplexer/dashboard-state-methods.js +3 -3
  85. package/dist/multiplexer/dashboard-tail-methods.d.ts +22 -10
  86. package/dist/multiplexer/dashboard-tail-methods.js +164 -48
  87. package/dist/multiplexer/dashboard-view-methods.d.ts +1 -1
  88. package/dist/multiplexer/dashboard-view-methods.js +23 -9
  89. package/dist/multiplexer/graveyard-view-model.d.ts +9 -1
  90. package/dist/multiplexer/graveyard-view-model.js +39 -1
  91. package/dist/multiplexer/index.d.ts +15 -12
  92. package/dist/multiplexer/index.js +64 -44
  93. package/dist/multiplexer/navigation.js +0 -1
  94. package/dist/multiplexer/notifications.js +107 -25
  95. package/dist/multiplexer/persistence-methods.d.ts +31 -4
  96. package/dist/multiplexer/persistence-methods.js +304 -309
  97. package/dist/multiplexer/runtime-lifecycle-methods.d.ts +8 -10
  98. package/dist/multiplexer/runtime-lifecycle-methods.js +104 -87
  99. package/dist/multiplexer/runtime-state.d.ts +8 -10
  100. package/dist/multiplexer/runtime-state.js +82 -146
  101. package/dist/multiplexer/runtime-sync.d.ts +2 -10
  102. package/dist/multiplexer/runtime-sync.js +3 -19
  103. package/dist/multiplexer/service-state-snapshot.d.ts +2 -4
  104. package/dist/multiplexer/service-state-snapshot.js +4 -52
  105. package/dist/multiplexer/services.d.ts +1 -0
  106. package/dist/multiplexer/services.js +55 -6
  107. package/dist/multiplexer/session-capture.d.ts +1 -0
  108. package/dist/multiplexer/session-capture.js +23 -0
  109. package/dist/multiplexer/session-launch.d.ts +4 -1
  110. package/dist/multiplexer/session-launch.js +152 -64
  111. package/dist/multiplexer/session-runtime-core.d.ts +8 -20
  112. package/dist/multiplexer/session-runtime-core.js +40 -136
  113. package/dist/multiplexer/subscreens.js +10 -4
  114. package/dist/multiplexer/tool-picker.js +0 -1
  115. package/dist/multiplexer/worktree-graveyard.d.ts +0 -1
  116. package/dist/multiplexer/worktree-graveyard.js +15 -17
  117. package/dist/multiplexer/worktrees.js +96 -41
  118. package/dist/notification-context.js +8 -5
  119. package/dist/notifications.js +163 -102
  120. package/dist/notify.d.ts +4 -0
  121. package/dist/notify.js +14 -1
  122. package/dist/orchestration-actions.js +0 -1
  123. package/dist/orchestration-routing.js +0 -1
  124. package/dist/orchestration.js +0 -1
  125. package/dist/osc-notifications.js +0 -1
  126. package/dist/paths.d.ts +32 -7
  127. package/dist/paths.js +82 -59
  128. package/dist/pending-actions.d.ts +5 -0
  129. package/dist/pending-actions.js +13 -0
  130. package/dist/plugin-runtime.js +9 -3
  131. package/dist/project-events.d.ts +1 -10
  132. package/dist/project-events.js +0 -11
  133. package/dist/project-scanner.d.ts +2 -3
  134. package/dist/project-scanner.js +58 -130
  135. package/dist/project-service-manifest.d.ts +1 -3
  136. package/dist/project-service-manifest.js +1 -4
  137. package/dist/recency.js +0 -1
  138. package/dist/recorder.js +0 -1
  139. package/dist/relay-client.d.ts +30 -0
  140. package/dist/relay-client.js +190 -0
  141. package/dist/remote-access.d.ts +16 -0
  142. package/dist/remote-access.js +90 -0
  143. package/dist/runtime-core/exchange-derived.d.ts +2 -0
  144. package/dist/runtime-core/exchange-derived.js +153 -0
  145. package/dist/runtime-core/exchange-import.d.ts +24 -0
  146. package/dist/runtime-core/exchange-import.js +317 -0
  147. package/dist/runtime-core/exchange-store.d.ts +157 -0
  148. package/dist/runtime-core/exchange-store.js +452 -0
  149. package/dist/runtime-core/topology-services.d.ts +38 -0
  150. package/dist/runtime-core/topology-services.js +170 -0
  151. package/dist/runtime-core/topology-sessions.d.ts +52 -0
  152. package/dist/runtime-core/topology-sessions.js +238 -0
  153. package/dist/runtime-core/topology-store.d.ts +171 -0
  154. package/dist/runtime-core/topology-store.js +419 -0
  155. package/dist/runtime-core/topology-worktrees.d.ts +60 -0
  156. package/dist/runtime-core/topology-worktrees.js +199 -0
  157. package/dist/runtime-migration.d.ts +69 -0
  158. package/dist/runtime-migration.js +398 -0
  159. package/dist/session-bootstrap.d.ts +8 -6
  160. package/dist/session-bootstrap.js +51 -159
  161. package/dist/session-runtime.d.ts +2 -0
  162. package/dist/session-runtime.js +1 -1
  163. package/dist/session-semantics.d.ts +12 -4
  164. package/dist/session-semantics.js +14 -1
  165. package/dist/shell-args.js +0 -1
  166. package/dist/shell-hooks.js +32 -11
  167. package/dist/shell-state.d.ts +2 -0
  168. package/dist/shell-state.js +26 -2
  169. package/dist/status-detector.js +0 -1
  170. package/dist/statusline-model.d.ts +10 -2
  171. package/dist/statusline-model.js +106 -31
  172. package/dist/task-workflow.d.ts +6 -9
  173. package/dist/task-workflow.js +37 -85
  174. package/dist/tasks.d.ts +6 -33
  175. package/dist/tasks.js +46 -89
  176. package/dist/team.d.ts +29 -0
  177. package/dist/team.js +40 -1
  178. package/dist/terminal-host.js +0 -1
  179. package/dist/threads.d.ts +6 -35
  180. package/dist/threads.js +89 -99
  181. package/dist/tmux/doctor.js +0 -1
  182. package/dist/tmux/inbox-popup.js +37 -16
  183. package/dist/tmux/runtime-manager.d.ts +3 -0
  184. package/dist/tmux/runtime-manager.js +21 -5
  185. package/dist/tmux/session-transport.js +0 -1
  186. package/dist/tmux/statusline-cache.js +0 -1
  187. package/dist/tmux/statusline.js +49 -10
  188. package/dist/tmux/switcher.js +0 -1
  189. package/dist/tmux/window-open.js +1 -3
  190. package/dist/tool-output-watchers.d.ts +0 -18
  191. package/dist/tool-output-watchers.js +0 -323
  192. package/dist/tui/render/box.js +0 -1
  193. package/dist/tui/render/text.js +0 -1
  194. package/dist/tui/screens/dashboard-renderers.js +37 -26
  195. package/dist/tui/screens/overlay-renderers.d.ts +2 -0
  196. package/dist/tui/screens/overlay-renderers.js +37 -2
  197. package/dist/tui/screens/subscreen-renderers.js +7 -1
  198. package/dist/workflow.js +0 -1
  199. package/dist/worktree.js +17 -1
  200. package/dist-ui/_expo/static/css/web-30453ede1678c16acb08b97e83e8646d.css +1 -0
  201. package/dist-ui/_expo/static/js/web/entry-477c745b2adc79367a4380ecf07d9ff6.js +14620 -0
  202. package/dist-ui/assets/assets/images/icon.a5413dcd2e811c9f2317d01a28118d8a.png +0 -0
  203. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/back-icon-mask.0a328cd9c1afd0afe8e3b1ec5165b1b4.png +0 -0
  204. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/back-icon.35ba0eaec5a4f5ed12ca16fabeae451d.png +0 -0
  205. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/clear-icon.c94f6478e7ae0cdd9f15de1fcb9e5e55.png +0 -0
  206. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/clear-icon.c94f6478e7ae0cdd9f15de1fcb9e5e55@2x.png +0 -0
  207. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/clear-icon.c94f6478e7ae0cdd9f15de1fcb9e5e55@3x.png +0 -0
  208. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/clear-icon.c94f6478e7ae0cdd9f15de1fcb9e5e55@4x.png +0 -0
  209. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/close-icon.808e1b1b9b53114ec2838071a7e6daa7.png +0 -0
  210. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/close-icon.808e1b1b9b53114ec2838071a7e6daa7@2x.png +0 -0
  211. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/close-icon.808e1b1b9b53114ec2838071a7e6daa7@3x.png +0 -0
  212. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/close-icon.808e1b1b9b53114ec2838071a7e6daa7@4x.png +0 -0
  213. package/dist-ui/assets/node_modules/@react-navigation/elements/lib/module/assets/search-icon.286d67d3f74808a60a78d3ebf1a5fb57.png +0 -0
  214. package/dist-ui/assets/node_modules/expo-router/assets/arrow_down.017bc6ba3fc25503e5eb5e53826d48a8.png +0 -0
  215. package/dist-ui/assets/node_modules/expo-router/assets/error.d1ea1496f9057eb392d5bbf3732a61b7.png +0 -0
  216. package/dist-ui/assets/node_modules/expo-router/assets/file.19eeb73b9593a38f8e9f418337fc7d10.png +0 -0
  217. package/dist-ui/assets/node_modules/expo-router/assets/forward.d8b800c443b8972542883e0b9de2bdc6.png +0 -0
  218. package/dist-ui/assets/node_modules/expo-router/assets/pkg.ab19f4cbc543357183a20571f68380a3.png +0 -0
  219. package/dist-ui/assets/node_modules/expo-router/assets/sitemap.412dd9275b6b48ad28f5e3d81bb1f626.png +0 -0
  220. package/dist-ui/assets/node_modules/expo-router/assets/unmatched.20e71bdf79e3a97bf55fd9e164041578.png +0 -0
  221. package/dist-ui/favicon.ico +0 -0
  222. package/dist-ui/index.html +38 -0
  223. package/dist-ui/metadata.json +1 -0
  224. package/package.json +29 -12
  225. package/dist/agent-events.js.map +0 -1
  226. package/dist/agent-message-parts.d.ts +0 -17
  227. package/dist/agent-message-parts.js +0 -31
  228. package/dist/agent-message-parts.js.map +0 -1
  229. package/dist/agent-output-parser.js.map +0 -1
  230. package/dist/agent-prompt-delivery.js.map +0 -1
  231. package/dist/agent-tracker.js.map +0 -1
  232. package/dist/agent-watcher.js.map +0 -1
  233. package/dist/atomic-write.js.map +0 -1
  234. package/dist/attachment-store.js.map +0 -1
  235. package/dist/builtin-metadata-watchers.js.map +0 -1
  236. package/dist/claude-hooks.js.map +0 -1
  237. package/dist/config.js.map +0 -1
  238. package/dist/context/compactor.js.map +0 -1
  239. package/dist/context/context-bridge.js.map +0 -1
  240. package/dist/context/context-file.js.map +0 -1
  241. package/dist/context/history.js.map +0 -1
  242. package/dist/daemon.js.map +0 -1
  243. package/dist/dashboard/command-spec.js.map +0 -1
  244. package/dist/dashboard/feedback.js.map +0 -1
  245. package/dist/dashboard/index.js.map +0 -1
  246. package/dist/dashboard/operation-failures.js.map +0 -1
  247. package/dist/dashboard/pending-actions.js.map +0 -1
  248. package/dist/dashboard/quick-jump.js.map +0 -1
  249. package/dist/dashboard/runtime-evidence.js.map +0 -1
  250. package/dist/dashboard/session-actions.js.map +0 -1
  251. package/dist/dashboard/session-registry.js.map +0 -1
  252. package/dist/dashboard/sort.js.map +0 -1
  253. package/dist/dashboard/state.js.map +0 -1
  254. package/dist/dashboard/targets.js.map +0 -1
  255. package/dist/dashboard/ui-state-store.js.map +0 -1
  256. package/dist/debug.js.map +0 -1
  257. package/dist/default-plugins/gh-pr-context.js.map +0 -1
  258. package/dist/default-plugins/transcript-length.js.map +0 -1
  259. package/dist/fast-control.js.map +0 -1
  260. package/dist/hotkeys.js.map +0 -1
  261. package/dist/http-client.js.map +0 -1
  262. package/dist/instance-directory.d.ts +0 -32
  263. package/dist/instance-directory.js +0 -82
  264. package/dist/instance-directory.js.map +0 -1
  265. package/dist/instance-registry.d.ts +0 -39
  266. package/dist/instance-registry.js +0 -208
  267. package/dist/instance-registry.js.map +0 -1
  268. package/dist/key-parser.js.map +0 -1
  269. package/dist/last-used.js.map +0 -1
  270. package/dist/main.js.map +0 -1
  271. package/dist/managed-launch-env.js.map +0 -1
  272. package/dist/metadata-server.js.map +0 -1
  273. package/dist/metadata-store.js.map +0 -1
  274. package/dist/multiplexer/agent-io-methods.js.map +0 -1
  275. package/dist/multiplexer/archives.js.map +0 -1
  276. package/dist/multiplexer/dashboard-actions-methods.js.map +0 -1
  277. package/dist/multiplexer/dashboard-control.js.map +0 -1
  278. package/dist/multiplexer/dashboard-interaction.js.map +0 -1
  279. package/dist/multiplexer/dashboard-model.js.map +0 -1
  280. package/dist/multiplexer/dashboard-ops.js.map +0 -1
  281. package/dist/multiplexer/dashboard-state-methods.js.map +0 -1
  282. package/dist/multiplexer/dashboard-tail-methods.js.map +0 -1
  283. package/dist/multiplexer/dashboard-view-methods.js.map +0 -1
  284. package/dist/multiplexer/graveyard-view-model.js.map +0 -1
  285. package/dist/multiplexer/index.js.map +0 -1
  286. package/dist/multiplexer/navigation.js.map +0 -1
  287. package/dist/multiplexer/notifications.js.map +0 -1
  288. package/dist/multiplexer/persistence-methods.js.map +0 -1
  289. package/dist/multiplexer/runtime-lifecycle-methods.js.map +0 -1
  290. package/dist/multiplexer/runtime-state.js.map +0 -1
  291. package/dist/multiplexer/runtime-sync.js.map +0 -1
  292. package/dist/multiplexer/service-state-snapshot.js.map +0 -1
  293. package/dist/multiplexer/services.js.map +0 -1
  294. package/dist/multiplexer/session-actions.d.ts +0 -40
  295. package/dist/multiplexer/session-actions.js +0 -110
  296. package/dist/multiplexer/session-actions.js.map +0 -1
  297. package/dist/multiplexer/session-launch.js.map +0 -1
  298. package/dist/multiplexer/session-runtime-core.js.map +0 -1
  299. package/dist/multiplexer/subscreens.js.map +0 -1
  300. package/dist/multiplexer/tool-picker.js.map +0 -1
  301. package/dist/multiplexer/worktree-graveyard.js.map +0 -1
  302. package/dist/multiplexer/worktrees.js.map +0 -1
  303. package/dist/notification-context.js.map +0 -1
  304. package/dist/notifications.js.map +0 -1
  305. package/dist/notify.js.map +0 -1
  306. package/dist/orchestration-actions.js.map +0 -1
  307. package/dist/orchestration-dispatcher.d.ts +0 -25
  308. package/dist/orchestration-dispatcher.js +0 -59
  309. package/dist/orchestration-dispatcher.js.map +0 -1
  310. package/dist/orchestration-routing.js.map +0 -1
  311. package/dist/orchestration.js.map +0 -1
  312. package/dist/osc-notifications.js.map +0 -1
  313. package/dist/paths.js.map +0 -1
  314. package/dist/plugin-runtime.js.map +0 -1
  315. package/dist/project-events.js.map +0 -1
  316. package/dist/project-scanner.js.map +0 -1
  317. package/dist/project-service-manifest.js.map +0 -1
  318. package/dist/recency.js.map +0 -1
  319. package/dist/recorder.js.map +0 -1
  320. package/dist/session-bootstrap.js.map +0 -1
  321. package/dist/session-input-operations.d.ts +0 -19
  322. package/dist/session-input-operations.js +0 -46
  323. package/dist/session-input-operations.js.map +0 -1
  324. package/dist/session-message-history.d.ts +0 -27
  325. package/dist/session-message-history.js +0 -105
  326. package/dist/session-message-history.js.map +0 -1
  327. package/dist/session-runtime.js.map +0 -1
  328. package/dist/session-semantics.js.map +0 -1
  329. package/dist/shell-args.js.map +0 -1
  330. package/dist/shell-hooks.js.map +0 -1
  331. package/dist/shell-state.js.map +0 -1
  332. package/dist/status-detector.js.map +0 -1
  333. package/dist/statusline-model.js.map +0 -1
  334. package/dist/task-dispatcher.d.ts +0 -64
  335. package/dist/task-dispatcher.js +0 -213
  336. package/dist/task-dispatcher.js.map +0 -1
  337. package/dist/task-workflow.js.map +0 -1
  338. package/dist/tasks.js.map +0 -1
  339. package/dist/team.js.map +0 -1
  340. package/dist/terminal-host.js.map +0 -1
  341. package/dist/threads.js.map +0 -1
  342. package/dist/tmux/doctor.js.map +0 -1
  343. package/dist/tmux/inbox-popup.js.map +0 -1
  344. package/dist/tmux/runtime-manager.js.map +0 -1
  345. package/dist/tmux/session-transport.js.map +0 -1
  346. package/dist/tmux/statusline-cache.js.map +0 -1
  347. package/dist/tmux/statusline.js.map +0 -1
  348. package/dist/tmux/switcher.js.map +0 -1
  349. package/dist/tmux/window-open.js.map +0 -1
  350. package/dist/tool-output-watchers.js.map +0 -1
  351. package/dist/tui/render/box.js.map +0 -1
  352. package/dist/tui/render/text.js.map +0 -1
  353. package/dist/tui/screens/dashboard-renderers.js.map +0 -1
  354. package/dist/tui/screens/overlay-renderers.js.map +0 -1
  355. package/dist/tui/screens/subscreen-renderers.js.map +0 -1
  356. package/dist/workflow.js.map +0 -1
  357. package/dist/worktree.js.map +0 -1
@@ -0,0 +1,419 @@
1
+ import { existsSync, mkdirSync, readFileSync, renameSync, rmSync, writeFileSync } from "node:fs";
2
+ import { dirname, join } from "node:path";
3
+ import { parse, stringify } from "yaml";
4
+ import { getRuntimeTopologyPath } from "../paths.js";
5
+ export const RUNTIME_TOPOLOGY_VERSION = 1;
6
+ const UPDATE_LOCK_TIMEOUT_MS = 5_000;
7
+ const UPDATE_LOCK_RETRY_MS = 25;
8
+ export function emptyRuntimeTopology(now = new Date().toISOString()) {
9
+ return {
10
+ version: RUNTIME_TOPOLOGY_VERSION,
11
+ generatedAt: now,
12
+ rigs: [],
13
+ nodes: [],
14
+ edges: [],
15
+ bindings: [],
16
+ sessions: [],
17
+ services: [],
18
+ worktrees: [],
19
+ worktreeGraveyard: [],
20
+ teamRoles: [],
21
+ remoteClients: [],
22
+ lifecycleOperations: [],
23
+ exchangeRefs: [],
24
+ };
25
+ }
26
+ function asRecord(value, context) {
27
+ if (!value || typeof value !== "object" || Array.isArray(value)) {
28
+ throw new Error(`invalid runtime topology: ${context} must be an object`);
29
+ }
30
+ return value;
31
+ }
32
+ function asString(value, context) {
33
+ if (typeof value !== "string" || value.trim().length === 0) {
34
+ throw new Error(`invalid runtime topology: ${context} must be a non-empty string`);
35
+ }
36
+ return value;
37
+ }
38
+ function asOptionalString(value) {
39
+ return typeof value === "string" && value.trim().length > 0 ? value : undefined;
40
+ }
41
+ function asStringArray(value) {
42
+ if (!Array.isArray(value))
43
+ return undefined;
44
+ return value.map((entry) => String(entry));
45
+ }
46
+ function asOptionalNumber(value) {
47
+ return typeof value === "number" && Number.isFinite(value) ? value : undefined;
48
+ }
49
+ function asArray(value) {
50
+ return Array.isArray(value) ? value : [];
51
+ }
52
+ function normalizeRuntimeTopology(topology) {
53
+ const rigIds = new Set(topology.rigs.map((rig) => rig.id));
54
+ const nodes = topology.nodes.filter((node) => rigIds.has(node.rigId));
55
+ const nodeIds = new Set(nodes.map((node) => node.id));
56
+ const sessions = topology.sessions.filter((session) => nodeIds.has(session.nodeId));
57
+ const sessionIds = new Set(sessions.map((session) => session.id));
58
+ const services = topology.services.filter((service) => rigIds.has(service.rigId) && (!service.nodeId || nodeIds.has(service.nodeId)));
59
+ const serviceIds = new Set(services.map((service) => service.id));
60
+ const worktrees = topology.worktrees.filter((worktree) => rigIds.has(worktree.rigId));
61
+ const worktreeIds = new Set(worktrees.map((worktree) => worktree.id));
62
+ const hasLifecycleTarget = (operation) => {
63
+ if (operation.targetKind === "rig")
64
+ return rigIds.has(operation.targetId);
65
+ if (operation.targetKind === "node")
66
+ return nodeIds.has(operation.targetId);
67
+ if (operation.targetKind === "session")
68
+ return sessionIds.has(operation.targetId);
69
+ if (operation.targetKind === "service")
70
+ return serviceIds.has(operation.targetId);
71
+ return worktreeIds.has(operation.targetId);
72
+ };
73
+ return {
74
+ ...topology,
75
+ nodes,
76
+ edges: topology.edges.filter((edge) => rigIds.has(edge.rigId) && nodeIds.has(edge.sourceNodeId) && nodeIds.has(edge.targetNodeId)),
77
+ bindings: topology.bindings.filter((binding) => nodeIds.has(binding.nodeId)),
78
+ sessions,
79
+ services,
80
+ worktrees,
81
+ worktreeGraveyard: topology.worktreeGraveyard.filter((entry) => rigIds.has(entry.rigId)),
82
+ teamRoles: topology.teamRoles.filter((role) => rigIds.has(role.rigId) &&
83
+ (!role.nodeId || nodeIds.has(role.nodeId)) &&
84
+ (!role.parentNodeId || nodeIds.has(role.parentNodeId))),
85
+ remoteClients: topology.remoteClients
86
+ .filter((client) => rigIds.has(client.rigId))
87
+ .map((client) => ({
88
+ ...client,
89
+ ownsSessionIds: client.ownsSessionIds?.filter((sessionId) => sessionIds.has(sessionId)),
90
+ })),
91
+ lifecycleOperations: topology.lifecycleOperations.filter((operation) => rigIds.has(operation.rigId) && hasLifecycleTarget(operation)),
92
+ exchangeRefs: topology.exchangeRefs.filter((ref) => rigIds.has(ref.rigId) &&
93
+ (!ref.nodeId || nodeIds.has(ref.nodeId)) &&
94
+ (!ref.sessionId || sessionIds.has(ref.sessionId))),
95
+ };
96
+ }
97
+ function coerceRuntimeTopology(raw) {
98
+ const record = asRecord(raw, "root");
99
+ if (record.version !== RUNTIME_TOPOLOGY_VERSION) {
100
+ throw new Error(`unsupported runtime topology version: ${String(record.version)}`);
101
+ }
102
+ return normalizeRuntimeTopology({
103
+ version: RUNTIME_TOPOLOGY_VERSION,
104
+ generatedAt: asString(record.generatedAt, "generatedAt"),
105
+ rigs: asArray(record.rigs).map((entry, index) => {
106
+ const row = asRecord(entry, `rigs[${index}]`);
107
+ return {
108
+ id: asString(row.id, `rigs[${index}].id`),
109
+ name: asString(row.name, `rigs[${index}].name`),
110
+ projectRoot: asString(row.projectRoot, `rigs[${index}].projectRoot`),
111
+ createdAt: asString(row.createdAt, `rigs[${index}].createdAt`),
112
+ updatedAt: asString(row.updatedAt, `rigs[${index}].updatedAt`),
113
+ };
114
+ }),
115
+ nodes: asArray(record.nodes).map((entry, index) => {
116
+ const row = asRecord(entry, `nodes[${index}]`);
117
+ return {
118
+ id: asString(row.id, `nodes[${index}].id`),
119
+ rigId: asString(row.rigId, `nodes[${index}].rigId`),
120
+ logicalId: asString(row.logicalId, `nodes[${index}].logicalId`),
121
+ role: asOptionalString(row.role),
122
+ runtime: asOptionalString(row.runtime),
123
+ toolConfigKey: asOptionalString(row.toolConfigKey),
124
+ model: asOptionalString(row.model),
125
+ cwd: asOptionalString(row.cwd),
126
+ label: asOptionalString(row.label),
127
+ createdAt: asString(row.createdAt, `nodes[${index}].createdAt`),
128
+ };
129
+ }),
130
+ edges: asArray(record.edges).map((entry, index) => {
131
+ const row = asRecord(entry, `edges[${index}]`);
132
+ return {
133
+ id: asString(row.id, `edges[${index}].id`),
134
+ rigId: asString(row.rigId, `edges[${index}].rigId`),
135
+ sourceNodeId: asString(row.sourceNodeId, `edges[${index}].sourceNodeId`),
136
+ targetNodeId: asString(row.targetNodeId, `edges[${index}].targetNodeId`),
137
+ kind: asString(row.kind, `edges[${index}].kind`),
138
+ createdAt: asString(row.createdAt, `edges[${index}].createdAt`),
139
+ };
140
+ }),
141
+ bindings: asArray(record.bindings).map((entry, index) => {
142
+ const row = asRecord(entry, `bindings[${index}]`);
143
+ return {
144
+ id: asString(row.id, `bindings[${index}].id`),
145
+ nodeId: asString(row.nodeId, `bindings[${index}].nodeId`),
146
+ tmuxSession: asOptionalString(row.tmuxSession),
147
+ tmuxWindowId: asOptionalString(row.tmuxWindowId),
148
+ tmuxWindowIndex: asOptionalNumber(row.tmuxWindowIndex),
149
+ tmuxWindowName: asOptionalString(row.tmuxWindowName),
150
+ tmuxPane: asOptionalString(row.tmuxPane),
151
+ updatedAt: asString(row.updatedAt, `bindings[${index}].updatedAt`),
152
+ };
153
+ }),
154
+ sessions: asArray(record.sessions).map((entry, index) => {
155
+ const row = asRecord(entry, `sessions[${index}]`);
156
+ return {
157
+ id: asString(row.id, `sessions[${index}].id`),
158
+ nodeId: asString(row.nodeId, `sessions[${index}].nodeId`),
159
+ status: asRuntimeSessionStatus(row.status),
160
+ tool: asOptionalString(row.tool),
161
+ command: asOptionalString(row.command),
162
+ args: asStringArray(row.args),
163
+ backendSessionId: asOptionalString(row.backendSessionId),
164
+ worktreePath: asOptionalString(row.worktreePath),
165
+ label: asOptionalString(row.label),
166
+ headline: asOptionalString(row.headline),
167
+ team: row.team,
168
+ createdAt: asString(row.createdAt, `sessions[${index}].createdAt`),
169
+ updatedAt: asString(row.updatedAt, `sessions[${index}].updatedAt`),
170
+ lastSeenAt: asOptionalString(row.lastSeenAt),
171
+ };
172
+ }),
173
+ services: asArray(record.services).map((entry, index) => {
174
+ const row = asRecord(entry, `services[${index}]`);
175
+ return {
176
+ id: asString(row.id, `services[${index}].id`),
177
+ rigId: asString(row.rigId, `services[${index}].rigId`),
178
+ nodeId: asOptionalString(row.nodeId),
179
+ status: asServiceStatus(row.status),
180
+ command: asOptionalString(row.command),
181
+ args: asStringArray(row.args),
182
+ launchCommandLine: asOptionalString(row.launchCommandLine),
183
+ worktreePath: asOptionalString(row.worktreePath),
184
+ cwd: asOptionalString(row.cwd),
185
+ label: asOptionalString(row.label),
186
+ createdAt: asString(row.createdAt, `services[${index}].createdAt`),
187
+ updatedAt: asString(row.updatedAt, `services[${index}].updatedAt`),
188
+ lastSeenAt: asOptionalString(row.lastSeenAt),
189
+ };
190
+ }),
191
+ worktrees: asArray(record.worktrees).map((entry, index) => {
192
+ const row = asRecord(entry, `worktrees[${index}]`);
193
+ return {
194
+ id: asString(row.id, `worktrees[${index}].id`),
195
+ rigId: asString(row.rigId, `worktrees[${index}].rigId`),
196
+ path: asString(row.path, `worktrees[${index}].path`),
197
+ name: asString(row.name, `worktrees[${index}].name`),
198
+ status: asWorktreeStatus(row.status),
199
+ branch: asOptionalString(row.branch),
200
+ head: asOptionalString(row.head),
201
+ basePath: asOptionalString(row.basePath),
202
+ createdAt: asString(row.createdAt, `worktrees[${index}].createdAt`),
203
+ updatedAt: asString(row.updatedAt, `worktrees[${index}].updatedAt`),
204
+ removedAt: asOptionalString(row.removedAt),
205
+ operationFailure: asOptionalString(row.operationFailure),
206
+ };
207
+ }),
208
+ worktreeGraveyard: asArray(record.worktreeGraveyard).map((entry, index) => {
209
+ const row = asRecord(entry, `worktreeGraveyard[${index}]`);
210
+ return {
211
+ id: asString(row.id, `worktreeGraveyard[${index}].id`),
212
+ rigId: asString(row.rigId, `worktreeGraveyard[${index}].rigId`),
213
+ worktreeId: asOptionalString(row.worktreeId),
214
+ path: asString(row.path, `worktreeGraveyard[${index}].path`),
215
+ name: asOptionalString(row.name),
216
+ branch: asOptionalString(row.branch),
217
+ graveyardedAt: asString(row.graveyardedAt, `worktreeGraveyard[${index}].graveyardedAt`),
218
+ reason: asOptionalString(row.reason),
219
+ deletedAt: asOptionalString(row.deletedAt),
220
+ };
221
+ }),
222
+ teamRoles: asArray(record.teamRoles).map((entry, index) => {
223
+ const row = asRecord(entry, `teamRoles[${index}]`);
224
+ return {
225
+ id: asString(row.id, `teamRoles[${index}].id`),
226
+ rigId: asString(row.rigId, `teamRoles[${index}].rigId`),
227
+ nodeId: asOptionalString(row.nodeId),
228
+ parentNodeId: asOptionalString(row.parentNodeId),
229
+ role: asString(row.role, `teamRoles[${index}].role`),
230
+ label: asOptionalString(row.label),
231
+ order: asOptionalNumber(row.order),
232
+ createdAt: asString(row.createdAt, `teamRoles[${index}].createdAt`),
233
+ updatedAt: asString(row.updatedAt, `teamRoles[${index}].updatedAt`),
234
+ };
235
+ }),
236
+ remoteClients: asArray(record.remoteClients).map((entry, index) => {
237
+ const row = asRecord(entry, `remoteClients[${index}]`);
238
+ return {
239
+ id: asString(row.id, `remoteClients[${index}].id`),
240
+ rigId: asString(row.rigId, `remoteClients[${index}].rigId`),
241
+ userId: asOptionalString(row.userId),
242
+ displayName: asOptionalString(row.displayName),
243
+ shareId: asOptionalString(row.shareId),
244
+ ownerUserId: asOptionalString(row.ownerUserId),
245
+ status: asRemoteClientStatus(row.status),
246
+ connectedAt: asOptionalString(row.connectedAt),
247
+ lastSeenAt: asString(row.lastSeenAt, `remoteClients[${index}].lastSeenAt`),
248
+ ownsSessionIds: asStringArray(row.ownsSessionIds),
249
+ };
250
+ }),
251
+ lifecycleOperations: asArray(record.lifecycleOperations).map((entry, index) => {
252
+ const row = asRecord(entry, `lifecycleOperations[${index}]`);
253
+ return {
254
+ id: asString(row.id, `lifecycleOperations[${index}].id`),
255
+ rigId: asString(row.rigId, `lifecycleOperations[${index}].rigId`),
256
+ kind: asString(row.kind, `lifecycleOperations[${index}].kind`),
257
+ status: asLifecycleOperationStatus(row.status),
258
+ targetKind: asLifecycleOperationTargetKind(row.targetKind, `lifecycleOperations[${index}].targetKind`),
259
+ targetId: asString(row.targetId, `lifecycleOperations[${index}].targetId`),
260
+ requestedBy: asOptionalString(row.requestedBy),
261
+ startedAt: asString(row.startedAt, `lifecycleOperations[${index}].startedAt`),
262
+ updatedAt: asString(row.updatedAt, `lifecycleOperations[${index}].updatedAt`),
263
+ completedAt: asOptionalString(row.completedAt),
264
+ error: asOptionalString(row.error),
265
+ };
266
+ }),
267
+ exchangeRefs: asArray(record.exchangeRefs).map((entry, index) => {
268
+ const row = asRecord(entry, `exchangeRefs[${index}]`);
269
+ return {
270
+ id: asString(row.id, `exchangeRefs[${index}].id`),
271
+ rigId: asString(row.rigId, `exchangeRefs[${index}].rigId`),
272
+ kind: asExchangeRefKind(row.kind, `exchangeRefs[${index}].kind`),
273
+ exchangeId: asString(row.exchangeId, `exchangeRefs[${index}].exchangeId`),
274
+ nodeId: asOptionalString(row.nodeId),
275
+ sessionId: asOptionalString(row.sessionId),
276
+ createdAt: asString(row.createdAt, `exchangeRefs[${index}].createdAt`),
277
+ updatedAt: asString(row.updatedAt, `exchangeRefs[${index}].updatedAt`),
278
+ };
279
+ }),
280
+ });
281
+ }
282
+ function asRuntimeSessionStatus(value) {
283
+ const status = String(value);
284
+ if (status === "planned" ||
285
+ status === "starting" ||
286
+ status === "running" ||
287
+ status === "idle" ||
288
+ status === "offline" ||
289
+ status === "graveyard" ||
290
+ status === "error") {
291
+ return status;
292
+ }
293
+ return "error";
294
+ }
295
+ function asServiceStatus(value) {
296
+ const status = String(value);
297
+ if (status === "planned" ||
298
+ status === "starting" ||
299
+ status === "running" ||
300
+ status === "stopped" ||
301
+ status === "offline" ||
302
+ status === "error") {
303
+ return status;
304
+ }
305
+ return "error";
306
+ }
307
+ function asWorktreeStatus(value) {
308
+ const status = String(value);
309
+ if (status === "planned" ||
310
+ status === "creating" ||
311
+ status === "active" ||
312
+ status === "removing" ||
313
+ status === "graveyard" ||
314
+ status === "missing" ||
315
+ status === "error") {
316
+ return status;
317
+ }
318
+ return "error";
319
+ }
320
+ function asRemoteClientStatus(value) {
321
+ const status = String(value);
322
+ if (status === "online" || status === "stale" || status === "offline")
323
+ return status;
324
+ return "offline";
325
+ }
326
+ function asLifecycleOperationStatus(value) {
327
+ const status = String(value);
328
+ if (status === "pending" ||
329
+ status === "running" ||
330
+ status === "succeeded" ||
331
+ status === "failed" ||
332
+ status === "cancelled") {
333
+ return status;
334
+ }
335
+ return "failed";
336
+ }
337
+ function asLifecycleOperationTargetKind(value, context) {
338
+ const kind = String(value);
339
+ if (kind === "session" || kind === "service" || kind === "worktree" || kind === "node" || kind === "rig") {
340
+ return kind;
341
+ }
342
+ throw new Error(`invalid runtime topology: ${context} must be a supported target kind`);
343
+ }
344
+ function asExchangeRefKind(value, context) {
345
+ const kind = String(value);
346
+ if (kind === "message" ||
347
+ kind === "handoff" ||
348
+ kind === "task" ||
349
+ kind === "review" ||
350
+ kind === "plan" ||
351
+ kind === "wait" ||
352
+ kind === "continuity" ||
353
+ kind === "attachment") {
354
+ return kind;
355
+ }
356
+ throw new Error(`invalid runtime topology: ${context} must be a supported exchange ref kind`);
357
+ }
358
+ function sleepSync(ms) {
359
+ Atomics.wait(new Int32Array(new SharedArrayBuffer(4)), 0, 0, ms);
360
+ }
361
+ export class RuntimeTopologyStore {
362
+ path;
363
+ constructor(path = getRuntimeTopologyPath()) {
364
+ this.path = path;
365
+ }
366
+ read() {
367
+ if (!existsSync(this.path))
368
+ return emptyRuntimeTopology();
369
+ const parsed = parse(readFileSync(this.path, "utf-8"));
370
+ return coerceRuntimeTopology(parsed);
371
+ }
372
+ write(topology) {
373
+ mkdirSync(dirname(this.path), { recursive: true });
374
+ const normalized = coerceRuntimeTopology(topology);
375
+ const tmpPath = `${this.path}.${process.pid}.${Date.now()}.${Math.random().toString(16).slice(2)}.tmp`;
376
+ writeFileSync(tmpPath, stringify(normalized, {
377
+ lineWidth: 120,
378
+ sortMapEntries: false,
379
+ }));
380
+ renameSync(tmpPath, this.path);
381
+ return normalized;
382
+ }
383
+ acquireUpdateLock() {
384
+ mkdirSync(dirname(this.path), { recursive: true });
385
+ const lockPath = `${this.path}.lock`;
386
+ const deadline = Date.now() + UPDATE_LOCK_TIMEOUT_MS;
387
+ while (true) {
388
+ try {
389
+ mkdirSync(lockPath);
390
+ try {
391
+ writeFileSync(join(lockPath, "owner"), `${process.pid}\n`);
392
+ }
393
+ catch (ownerError) {
394
+ rmSync(lockPath, { recursive: true, force: true });
395
+ throw ownerError;
396
+ }
397
+ return () => rmSync(lockPath, { recursive: true, force: true });
398
+ }
399
+ catch (error) {
400
+ if (Date.now() >= deadline) {
401
+ throw new Error(`Timed out acquiring runtime topology update lock at ${lockPath}`, { cause: error });
402
+ }
403
+ sleepSync(UPDATE_LOCK_RETRY_MS);
404
+ }
405
+ }
406
+ }
407
+ update(mutator) {
408
+ const release = this.acquireUpdateLock();
409
+ try {
410
+ return this.write(mutator(this.read()));
411
+ }
412
+ finally {
413
+ release();
414
+ }
415
+ }
416
+ }
417
+ export function createRuntimeTopologyStore(path) {
418
+ return new RuntimeTopologyStore(path);
419
+ }
@@ -0,0 +1,60 @@
1
+ import { type RuntimeTopology, type RuntimeTopologyStore, type RuntimeTopologyWorktree, type RuntimeTopologyWorktreeGraveyardEntry, type RuntimeTopologyWorktreeStatus } from "./topology-store.js";
2
+ export type RuntimeTopologyWorktreeState = {
3
+ id?: string;
4
+ path: string;
5
+ name?: string;
6
+ status?: RuntimeTopologyWorktreeStatus;
7
+ branch?: string;
8
+ head?: string;
9
+ basePath?: string;
10
+ createdAt?: string;
11
+ removedAt?: string;
12
+ operationFailure?: string;
13
+ };
14
+ export type RuntimeTopologyWorktreeGraveyardState = {
15
+ id: string;
16
+ worktreeId?: string;
17
+ path: string;
18
+ name?: string;
19
+ branch?: string;
20
+ graveyardedAt: string;
21
+ reason?: string;
22
+ deletedAt?: string;
23
+ };
24
+ export declare function topologyWorktreeToWorktreeState(worktree: RuntimeTopologyWorktree): RuntimeTopologyWorktreeState;
25
+ export declare function topologyWorktreeGraveyardToState(entry: RuntimeTopologyWorktreeGraveyardEntry): RuntimeTopologyWorktreeGraveyardState;
26
+ export declare function listTopologyWorktreeStates(input?: {
27
+ statuses?: RuntimeTopologyWorktreeStatus[];
28
+ store?: RuntimeTopologyStore;
29
+ }): RuntimeTopologyWorktreeState[];
30
+ export declare function listTopologyWorktreeGraveyard(input?: {
31
+ includeDeleted?: boolean;
32
+ store?: RuntimeTopologyStore;
33
+ }): RuntimeTopologyWorktreeGraveyardState[];
34
+ export declare function listTopologyWorktreeGraveyardPaths(input?: {
35
+ store?: RuntimeTopologyStore;
36
+ }): Set<string>;
37
+ export declare function upsertTopologyWorktree(worktree: RuntimeTopologyWorktreeState, status: RuntimeTopologyWorktreeStatus, input?: {
38
+ store?: RuntimeTopologyStore;
39
+ now?: string;
40
+ projectRoot?: string;
41
+ }): RuntimeTopology;
42
+ export declare function moveTopologyWorktreeToGraveyard(path: string, input?: {
43
+ store?: RuntimeTopologyStore;
44
+ now?: string;
45
+ reason?: string;
46
+ projectRoot?: string;
47
+ }): RuntimeTopologyWorktreeGraveyardState | undefined;
48
+ export declare function deleteTopologyWorktreeGraveyardEntry(path: string, input?: {
49
+ store?: RuntimeTopologyStore;
50
+ now?: string;
51
+ }): RuntimeTopologyWorktreeGraveyardState | undefined;
52
+ export declare function resurrectTopologyWorktreeFromGraveyard(path: string, input?: {
53
+ store?: RuntimeTopologyStore;
54
+ now?: string;
55
+ projectRoot?: string;
56
+ }): RuntimeTopologyWorktreeState | undefined;
57
+ export declare function removeTopologyWorktree(path: string, input?: {
58
+ store?: RuntimeTopologyStore;
59
+ now?: string;
60
+ }): RuntimeTopologyWorktreeState | undefined;
@@ -0,0 +1,199 @@
1
+ import { createHash } from "node:crypto";
2
+ import { basename } from "node:path";
3
+ import { getProjectId, getRepoRoot } from "../paths.js";
4
+ import { createRuntimeTopologyStore, emptyRuntimeTopology, } from "./topology-store.js";
5
+ function worktreeIdForPath(path) {
6
+ return `worktree:${createHash("sha256").update(path).digest("base64url").slice(0, 24)}`;
7
+ }
8
+ function graveyardIdForPath(path) {
9
+ return `worktree-graveyard:${createHash("sha256").update(path).digest("base64url").slice(0, 24)}`;
10
+ }
11
+ function ensureRig(topology, projectRoot, now) {
12
+ const id = getProjectId();
13
+ const existing = topology.rigs.find((rig) => rig.id === id);
14
+ if (existing) {
15
+ existing.projectRoot = projectRoot;
16
+ existing.name = basename(projectRoot);
17
+ existing.updatedAt = now;
18
+ return existing.id;
19
+ }
20
+ topology.rigs.push({
21
+ id,
22
+ name: basename(projectRoot),
23
+ projectRoot,
24
+ createdAt: now,
25
+ updatedAt: now,
26
+ });
27
+ return id;
28
+ }
29
+ function worktreeToTopologyWorktree(worktree, rigId, status, now) {
30
+ return {
31
+ id: worktree.id ?? worktreeIdForPath(worktree.path),
32
+ rigId,
33
+ path: worktree.path,
34
+ name: worktree.name ?? basename(worktree.path),
35
+ status,
36
+ branch: worktree.branch,
37
+ head: worktree.head,
38
+ basePath: worktree.basePath,
39
+ createdAt: worktree.createdAt ?? now,
40
+ updatedAt: now,
41
+ removedAt: worktree.removedAt,
42
+ operationFailure: worktree.operationFailure,
43
+ };
44
+ }
45
+ export function topologyWorktreeToWorktreeState(worktree) {
46
+ return {
47
+ id: worktree.id,
48
+ path: worktree.path,
49
+ name: worktree.name,
50
+ status: worktree.status,
51
+ branch: worktree.branch,
52
+ head: worktree.head,
53
+ basePath: worktree.basePath,
54
+ createdAt: worktree.createdAt,
55
+ removedAt: worktree.removedAt,
56
+ operationFailure: worktree.operationFailure,
57
+ };
58
+ }
59
+ export function topologyWorktreeGraveyardToState(entry) {
60
+ return {
61
+ id: entry.id,
62
+ worktreeId: entry.worktreeId,
63
+ path: entry.path,
64
+ name: entry.name,
65
+ branch: entry.branch,
66
+ graveyardedAt: entry.graveyardedAt,
67
+ reason: entry.reason,
68
+ deletedAt: entry.deletedAt,
69
+ };
70
+ }
71
+ export function listTopologyWorktreeStates(input) {
72
+ const topology = (input?.store ?? createRuntimeTopologyStore()).read();
73
+ const statuses = input?.statuses ? new Set(input.statuses) : undefined;
74
+ return topology.worktrees
75
+ .filter((worktree) => !statuses || statuses.has(worktree.status))
76
+ .map(topologyWorktreeToWorktreeState);
77
+ }
78
+ export function listTopologyWorktreeGraveyard(input) {
79
+ const topology = (input?.store ?? createRuntimeTopologyStore()).read();
80
+ return topology.worktreeGraveyard
81
+ .filter((entry) => input?.includeDeleted || !entry.deletedAt)
82
+ .map(topologyWorktreeGraveyardToState);
83
+ }
84
+ export function listTopologyWorktreeGraveyardPaths(input) {
85
+ return new Set(listTopologyWorktreeGraveyard(input).map((entry) => entry.path));
86
+ }
87
+ export function upsertTopologyWorktree(worktree, status, input) {
88
+ const store = input?.store ?? createRuntimeTopologyStore();
89
+ const now = input?.now ?? new Date().toISOString();
90
+ const projectRoot = input?.projectRoot ?? getRepoRoot();
91
+ return store.update((current) => {
92
+ const topology = current.version ? current : emptyRuntimeTopology(now);
93
+ topology.generatedAt = now;
94
+ const rigId = ensureRig(topology, projectRoot, now);
95
+ const next = worktreeToTopologyWorktree(worktree, rigId, status, now);
96
+ topology.worktrees = [...topology.worktrees.filter((entry) => entry.id !== next.id), next];
97
+ return topology;
98
+ });
99
+ }
100
+ export function moveTopologyWorktreeToGraveyard(path, input) {
101
+ const store = input?.store ?? createRuntimeTopologyStore();
102
+ const now = input?.now ?? new Date().toISOString();
103
+ const projectRoot = input?.projectRoot ?? getRepoRoot();
104
+ let moved;
105
+ store.update((current) => {
106
+ const topology = current.version ? current : emptyRuntimeTopology(now);
107
+ topology.generatedAt = now;
108
+ const rigId = ensureRig(topology, projectRoot, now);
109
+ const existing = topology.worktrees.find((worktree) => worktree.path === path);
110
+ if (!existing)
111
+ return topology;
112
+ const graveyardEntry = {
113
+ id: graveyardIdForPath(path),
114
+ rigId,
115
+ worktreeId: existing.id,
116
+ path,
117
+ name: existing.name,
118
+ branch: existing.branch,
119
+ graveyardedAt: now,
120
+ reason: input?.reason,
121
+ };
122
+ existing.status = "graveyard";
123
+ existing.removedAt = now;
124
+ existing.updatedAt = now;
125
+ topology.worktreeGraveyard = [...topology.worktreeGraveyard.filter((entry) => entry.path !== path), graveyardEntry];
126
+ moved = topologyWorktreeGraveyardToState(graveyardEntry);
127
+ return topology;
128
+ });
129
+ return moved;
130
+ }
131
+ export function deleteTopologyWorktreeGraveyardEntry(path, input) {
132
+ const store = input?.store ?? createRuntimeTopologyStore();
133
+ const now = input?.now ?? new Date().toISOString();
134
+ let deleted;
135
+ store.update((topology) => {
136
+ const existing = topology.worktreeGraveyard.find((entry) => entry.path === path && !entry.deletedAt);
137
+ if (!existing)
138
+ return topology;
139
+ existing.deletedAt = now;
140
+ topology.generatedAt = now;
141
+ deleted = topologyWorktreeGraveyardToState(existing);
142
+ return topology;
143
+ });
144
+ return deleted;
145
+ }
146
+ export function resurrectTopologyWorktreeFromGraveyard(path, input) {
147
+ const store = input?.store ?? createRuntimeTopologyStore();
148
+ const now = input?.now ?? new Date().toISOString();
149
+ const projectRoot = input?.projectRoot ?? getRepoRoot();
150
+ let resurrected;
151
+ store.update((current) => {
152
+ const topology = current.version ? current : emptyRuntimeTopology(now);
153
+ topology.generatedAt = now;
154
+ const rigId = ensureRig(topology, projectRoot, now);
155
+ const graveyardEntry = topology.worktreeGraveyard.find((entry) => entry.path === path && !entry.deletedAt);
156
+ if (!graveyardEntry)
157
+ return topology;
158
+ const existing = topology.worktrees.find((worktree) => worktree.id === graveyardEntry.worktreeId || worktree.path === path);
159
+ if (existing) {
160
+ existing.rigId = rigId;
161
+ existing.path = path;
162
+ existing.name = existing.name ?? graveyardEntry.name;
163
+ existing.branch = existing.branch ?? graveyardEntry.branch;
164
+ existing.status = "active";
165
+ existing.updatedAt = now;
166
+ delete existing.removedAt;
167
+ resurrected = topologyWorktreeToWorktreeState(existing);
168
+ }
169
+ else {
170
+ const next = worktreeToTopologyWorktree({
171
+ path,
172
+ name: graveyardEntry.name,
173
+ branch: graveyardEntry.branch,
174
+ createdAt: graveyardEntry.graveyardedAt,
175
+ }, rigId, "active", now);
176
+ topology.worktrees.push(next);
177
+ resurrected = topologyWorktreeToWorktreeState(next);
178
+ }
179
+ topology.worktreeGraveyard = topology.worktreeGraveyard.filter((entry) => entry.path !== path);
180
+ return topology;
181
+ });
182
+ return resurrected;
183
+ }
184
+ export function removeTopologyWorktree(path, input) {
185
+ const store = input?.store ?? createRuntimeTopologyStore();
186
+ const now = input?.now ?? new Date().toISOString();
187
+ let removed;
188
+ store.update((topology) => {
189
+ const existing = topology.worktrees.find((worktree) => worktree.path === path);
190
+ if (!existing)
191
+ return topology;
192
+ removed = topologyWorktreeToWorktreeState(existing);
193
+ topology.generatedAt = now;
194
+ topology.worktrees = topology.worktrees.filter((worktree) => worktree.path !== path);
195
+ topology.lifecycleOperations = topology.lifecycleOperations.filter((operation) => !(operation.targetKind === "worktree" && operation.targetId === existing.id));
196
+ return topology;
197
+ });
198
+ return removed;
199
+ }