macro-agent 0.1.8 → 0.1.11

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 (306) hide show
  1. package/CLAUDE.md +263 -33
  2. package/README.md +781 -131
  3. package/dist/acp/claude-code-replay.d.ts +11 -0
  4. package/dist/acp/claude-code-replay.d.ts.map +1 -0
  5. package/dist/acp/claude-code-replay.js +190 -0
  6. package/dist/acp/claude-code-replay.js.map +1 -0
  7. package/dist/acp/macro-agent.d.ts.map +1 -1
  8. package/dist/acp/macro-agent.js +192 -7
  9. package/dist/acp/macro-agent.js.map +1 -1
  10. package/dist/acp/types.d.ts +9 -0
  11. package/dist/acp/types.d.ts.map +1 -1
  12. package/dist/acp/types.js.map +1 -1
  13. package/dist/adapters/tasks-adapter.d.ts.map +1 -1
  14. package/dist/adapters/tasks-adapter.js +3 -0
  15. package/dist/adapters/tasks-adapter.js.map +1 -1
  16. package/dist/adapters/types.d.ts +1 -0
  17. package/dist/adapters/types.d.ts.map +1 -1
  18. package/dist/agent/agent-manager-v2.d.ts +21 -0
  19. package/dist/agent/agent-manager-v2.d.ts.map +1 -1
  20. package/dist/agent/agent-manager-v2.js +308 -54
  21. package/dist/agent/agent-manager-v2.js.map +1 -1
  22. package/dist/agent/agent-manager.d.ts +12 -0
  23. package/dist/agent/agent-manager.d.ts.map +1 -1
  24. package/dist/agent/agent-manager.js.map +1 -1
  25. package/dist/agent/agent-store.d.ts +10 -0
  26. package/dist/agent/agent-store.d.ts.map +1 -1
  27. package/dist/agent/agent-store.js +22 -0
  28. package/dist/agent/agent-store.js.map +1 -1
  29. package/dist/agent/types.d.ts +15 -2
  30. package/dist/agent/types.d.ts.map +1 -1
  31. package/dist/agent/types.js.map +1 -1
  32. package/dist/boot-v2.d.ts +129 -1
  33. package/dist/boot-v2.d.ts.map +1 -1
  34. package/dist/boot-v2.js +359 -8
  35. package/dist/boot-v2.js.map +1 -1
  36. package/dist/cli/acp.js +4 -0
  37. package/dist/cli/acp.js.map +1 -1
  38. package/dist/cli/index.js +56 -0
  39. package/dist/cli/index.js.map +1 -1
  40. package/dist/cognitive/macro-agent-backend.d.ts.map +1 -1
  41. package/dist/cognitive/macro-agent-backend.js +40 -22
  42. package/dist/cognitive/macro-agent-backend.js.map +1 -1
  43. package/dist/integrations/skilltree.d.ts.map +1 -1
  44. package/dist/integrations/skilltree.js +1 -0
  45. package/dist/integrations/skilltree.js.map +1 -1
  46. package/dist/lifecycle/cascade.d.ts +25 -2
  47. package/dist/lifecycle/cascade.d.ts.map +1 -1
  48. package/dist/lifecycle/cascade.js +70 -2
  49. package/dist/lifecycle/cascade.js.map +1 -1
  50. package/dist/lifecycle/cleanup.d.ts +33 -2
  51. package/dist/lifecycle/cleanup.d.ts.map +1 -1
  52. package/dist/lifecycle/cleanup.js +28 -6
  53. package/dist/lifecycle/cleanup.js.map +1 -1
  54. package/dist/lifecycle/handlers-v2.d.ts +7 -0
  55. package/dist/lifecycle/handlers-v2.d.ts.map +1 -1
  56. package/dist/lifecycle/handlers-v2.js +28 -2
  57. package/dist/lifecycle/handlers-v2.js.map +1 -1
  58. package/dist/lifecycle/types.d.ts +11 -0
  59. package/dist/lifecycle/types.d.ts.map +1 -1
  60. package/dist/lifecycle/types.js.map +1 -1
  61. package/dist/map/acp-bridge.d.ts +9 -0
  62. package/dist/map/acp-bridge.d.ts.map +1 -1
  63. package/dist/map/acp-bridge.js +15 -2
  64. package/dist/map/acp-bridge.js.map +1 -1
  65. package/dist/map/cascade-action-handler.d.ts +24 -0
  66. package/dist/map/cascade-action-handler.d.ts.map +1 -0
  67. package/dist/map/cascade-action-handler.js +170 -0
  68. package/dist/map/cascade-action-handler.js.map +1 -0
  69. package/dist/map/cascade-bridge.d.ts +44 -0
  70. package/dist/map/cascade-bridge.d.ts.map +1 -0
  71. package/dist/map/cascade-bridge.js +294 -0
  72. package/dist/map/cascade-bridge.js.map +1 -0
  73. package/dist/map/coordination-handler.d.ts.map +1 -1
  74. package/dist/map/coordination-handler.js +12 -1
  75. package/dist/map/coordination-handler.js.map +1 -1
  76. package/dist/map/lifecycle-bridge.d.ts +1 -1
  77. package/dist/map/lifecycle-bridge.d.ts.map +1 -1
  78. package/dist/map/lifecycle-bridge.js +58 -23
  79. package/dist/map/lifecycle-bridge.js.map +1 -1
  80. package/dist/map/server.d.ts.map +1 -1
  81. package/dist/map/server.js +219 -7
  82. package/dist/map/server.js.map +1 -1
  83. package/dist/map/sidecar.d.ts.map +1 -1
  84. package/dist/map/sidecar.js +49 -2
  85. package/dist/map/sidecar.js.map +1 -1
  86. package/dist/map/types.d.ts +22 -0
  87. package/dist/map/types.d.ts.map +1 -1
  88. package/dist/mcp/tools/done-v2.d.ts.map +1 -1
  89. package/dist/mcp/tools/done-v2.js +8 -0
  90. package/dist/mcp/tools/done-v2.js.map +1 -1
  91. package/dist/teams/team-manager-v2.d.ts.map +1 -1
  92. package/dist/teams/team-manager-v2.js +26 -0
  93. package/dist/teams/team-manager-v2.js.map +1 -1
  94. package/dist/teams/team-runtime-v2.d.ts.map +1 -1
  95. package/dist/teams/team-runtime-v2.js +16 -3
  96. package/dist/teams/team-runtime-v2.js.map +1 -1
  97. package/dist/workspace/config.d.ts +10 -10
  98. package/dist/workspace/config.d.ts.map +1 -1
  99. package/dist/workspace/config.js +4 -4
  100. package/dist/workspace/config.js.map +1 -1
  101. package/dist/workspace/git-cascade-adapter.d.ts +510 -0
  102. package/dist/workspace/git-cascade-adapter.d.ts.map +1 -0
  103. package/dist/workspace/git-cascade-adapter.js +934 -0
  104. package/dist/workspace/git-cascade-adapter.js.map +1 -0
  105. package/dist/workspace/index.d.ts +3 -3
  106. package/dist/workspace/index.d.ts.map +1 -1
  107. package/dist/workspace/index.js +4 -4
  108. package/dist/workspace/index.js.map +1 -1
  109. package/dist/workspace/landing/direct-push.d.ts +20 -0
  110. package/dist/workspace/landing/direct-push.d.ts.map +1 -0
  111. package/dist/workspace/landing/direct-push.js +74 -0
  112. package/dist/workspace/landing/direct-push.js.map +1 -0
  113. package/dist/workspace/landing/index.d.ts +29 -0
  114. package/dist/workspace/landing/index.d.ts.map +1 -0
  115. package/dist/workspace/landing/index.js +37 -0
  116. package/dist/workspace/landing/index.js.map +1 -0
  117. package/dist/workspace/landing/merge-to-parent.d.ts +41 -0
  118. package/dist/workspace/landing/merge-to-parent.d.ts.map +1 -0
  119. package/dist/workspace/landing/merge-to-parent.js +186 -0
  120. package/dist/workspace/landing/merge-to-parent.js.map +1 -0
  121. package/dist/workspace/landing/optimistic-push.d.ts +16 -0
  122. package/dist/workspace/landing/optimistic-push.d.ts.map +1 -0
  123. package/dist/workspace/landing/optimistic-push.js +27 -0
  124. package/dist/workspace/landing/optimistic-push.js.map +1 -0
  125. package/dist/workspace/landing/queue-to-branch.d.ts +24 -0
  126. package/dist/workspace/landing/queue-to-branch.d.ts.map +1 -0
  127. package/dist/workspace/landing/queue-to-branch.js +79 -0
  128. package/dist/workspace/landing/queue-to-branch.js.map +1 -0
  129. package/dist/workspace/merge-queue/merge-queue.d.ts +10 -0
  130. package/dist/workspace/merge-queue/merge-queue.d.ts.map +1 -1
  131. package/dist/workspace/merge-queue/merge-queue.js +10 -0
  132. package/dist/workspace/merge-queue/merge-queue.js.map +1 -1
  133. package/dist/workspace/merge-queue/types.d.ts +16 -2
  134. package/dist/workspace/merge-queue/types.d.ts.map +1 -1
  135. package/dist/workspace/merge-queue/types.js +9 -0
  136. package/dist/workspace/merge-queue/types.js.map +1 -1
  137. package/dist/workspace/pool/types.d.ts +1 -0
  138. package/dist/workspace/pool/types.d.ts.map +1 -1
  139. package/dist/workspace/pool/worktree-pool.d.ts.map +1 -1
  140. package/dist/workspace/pool/worktree-pool.js +1 -0
  141. package/dist/workspace/pool/worktree-pool.js.map +1 -1
  142. package/dist/workspace/recovery/abandon.d.ts +15 -0
  143. package/dist/workspace/recovery/abandon.d.ts.map +1 -0
  144. package/dist/workspace/recovery/abandon.js +45 -0
  145. package/dist/workspace/recovery/abandon.js.map +1 -0
  146. package/dist/workspace/recovery/auto-resolve.d.ts +27 -0
  147. package/dist/workspace/recovery/auto-resolve.d.ts.map +1 -0
  148. package/dist/workspace/recovery/auto-resolve.js +99 -0
  149. package/dist/workspace/recovery/auto-resolve.js.map +1 -0
  150. package/dist/workspace/recovery/defer.d.ts +15 -0
  151. package/dist/workspace/recovery/defer.d.ts.map +1 -0
  152. package/dist/workspace/recovery/defer.js +16 -0
  153. package/dist/workspace/recovery/defer.js.map +1 -0
  154. package/dist/workspace/recovery/escalate.d.ts +16 -0
  155. package/dist/workspace/recovery/escalate.d.ts.map +1 -0
  156. package/dist/workspace/recovery/escalate.js +24 -0
  157. package/dist/workspace/recovery/escalate.js.map +1 -0
  158. package/dist/workspace/recovery/index.d.ts +32 -0
  159. package/dist/workspace/recovery/index.d.ts.map +1 -0
  160. package/dist/workspace/recovery/index.js +45 -0
  161. package/dist/workspace/recovery/index.js.map +1 -0
  162. package/dist/workspace/recovery/spawn-resolver.d.ts +45 -0
  163. package/dist/workspace/recovery/spawn-resolver.d.ts.map +1 -0
  164. package/dist/workspace/recovery/spawn-resolver.js +118 -0
  165. package/dist/workspace/recovery/spawn-resolver.js.map +1 -0
  166. package/dist/workspace/recovery/types.d.ts +63 -0
  167. package/dist/workspace/recovery/types.d.ts.map +1 -0
  168. package/dist/workspace/recovery/types.js +12 -0
  169. package/dist/workspace/recovery/types.js.map +1 -0
  170. package/dist/workspace/topology/index.d.ts +9 -0
  171. package/dist/workspace/topology/index.d.ts.map +1 -0
  172. package/dist/workspace/topology/index.js +8 -0
  173. package/dist/workspace/topology/index.js.map +1 -0
  174. package/dist/workspace/topology/no-workspace.d.ts +18 -0
  175. package/dist/workspace/topology/no-workspace.d.ts.map +1 -0
  176. package/dist/workspace/topology/no-workspace.js +25 -0
  177. package/dist/workspace/topology/no-workspace.js.map +1 -0
  178. package/dist/workspace/topology/types.d.ts +97 -0
  179. package/dist/workspace/topology/types.d.ts.map +1 -0
  180. package/dist/workspace/topology/types.js +20 -0
  181. package/dist/workspace/topology/types.js.map +1 -0
  182. package/dist/workspace/topology/yaml-driven.d.ts +69 -0
  183. package/dist/workspace/topology/yaml-driven.d.ts.map +1 -0
  184. package/dist/workspace/topology/yaml-driven.js +273 -0
  185. package/dist/workspace/topology/yaml-driven.js.map +1 -0
  186. package/dist/workspace/types-v3.d.ts +117 -0
  187. package/dist/workspace/types-v3.d.ts.map +1 -0
  188. package/dist/workspace/types-v3.js +20 -0
  189. package/dist/workspace/types-v3.js.map +1 -0
  190. package/dist/workspace/types.d.ts +162 -17
  191. package/dist/workspace/types.d.ts.map +1 -1
  192. package/dist/workspace/workspace-manager.d.ts +101 -13
  193. package/dist/workspace/workspace-manager.d.ts.map +1 -1
  194. package/dist/workspace/workspace-manager.js +416 -13
  195. package/dist/workspace/workspace-manager.js.map +1 -1
  196. package/dist/workspace/yaml-schema.d.ts +254 -0
  197. package/dist/workspace/yaml-schema.d.ts.map +1 -0
  198. package/dist/workspace/yaml-schema.js +170 -0
  199. package/dist/workspace/yaml-schema.js.map +1 -0
  200. package/docs/conflict-recovery.md +472 -0
  201. package/docs/design/task-dispatcher.md +880 -0
  202. package/docs/git-cascade-integration-gaps.md +678 -0
  203. package/docs/workspace-interfaces.md +731 -0
  204. package/docs/workspace-redesign-plan.md +302 -0
  205. package/package.json +6 -5
  206. package/src/__tests__/boot-v2.test.ts +435 -0
  207. package/src/__tests__/e2e/acp-over-map.e2e.test.ts +92 -0
  208. package/src/__tests__/e2e/auto-sync.e2e.test.ts +257 -0
  209. package/src/__tests__/e2e/bootstrap.e2e.test.ts +319 -0
  210. package/src/__tests__/e2e/cascade-rebase.e2e.test.ts +254 -0
  211. package/src/__tests__/e2e/cli-run.e2e.test.ts +167 -0
  212. package/src/__tests__/e2e/dispatch-coordination.e2e.test.ts +495 -0
  213. package/src/__tests__/e2e/dispatch-live.e2e.test.ts +564 -0
  214. package/src/__tests__/e2e/dispatch-opentasks.e2e.test.ts +496 -0
  215. package/src/__tests__/e2e/dispatch-phase2-live.e2e.test.ts +456 -0
  216. package/src/__tests__/e2e/dispatch-phase2.e2e.test.ts +386 -0
  217. package/src/__tests__/e2e/dispatch.e2e.test.ts +376 -0
  218. package/src/__tests__/e2e/self-driving-v3.e2e.test.ts +197 -0
  219. package/src/__tests__/e2e/spawn-resolver.e2e.test.ts +200 -0
  220. package/src/__tests__/e2e/workspace-lifecycle.e2e.test.ts +30 -22
  221. package/src/__tests__/e2e/workspace-v3.e2e.test.ts +413 -0
  222. package/src/acp/__tests__/claude-code-replay.test.ts +225 -0
  223. package/src/acp/__tests__/macro-agent.test.ts +39 -1
  224. package/src/acp/claude-code-replay.ts +208 -0
  225. package/src/acp/macro-agent.ts +203 -10
  226. package/src/acp/types.ts +10 -0
  227. package/src/adapters/__tests__/tasks-adapter.test.ts +1 -0
  228. package/src/adapters/tasks-adapter.ts +3 -0
  229. package/src/adapters/types.ts +1 -0
  230. package/src/agent/__tests__/agent-manager-topology.test.ts +73 -0
  231. package/src/agent/__tests__/agent-manager-v2.test.ts +66 -0
  232. package/src/agent/__tests__/agent-store.test.ts +52 -0
  233. package/src/agent/__tests__/task-ref-resolution.test.ts +231 -0
  234. package/src/agent/agent-manager-v2.ts +372 -59
  235. package/src/agent/agent-manager.ts +14 -0
  236. package/src/agent/agent-store.ts +24 -0
  237. package/src/agent/types.ts +16 -2
  238. package/src/boot-v2.ts +589 -35
  239. package/src/cli/acp.ts +4 -0
  240. package/src/cli/index.ts +61 -0
  241. package/src/cognitive/macro-agent-backend.ts +45 -29
  242. package/src/integrations/skilltree.ts +1 -0
  243. package/src/lifecycle/__tests__/cascade-consolidation.test.ts +240 -0
  244. package/src/lifecycle/cascade.ts +77 -2
  245. package/src/lifecycle/cleanup.ts +52 -3
  246. package/src/lifecycle/handlers-v2.ts +40 -3
  247. package/src/lifecycle/types.ts +12 -0
  248. package/src/map/__tests__/cascade-bridge.test.ts +229 -0
  249. package/src/map/__tests__/emit-event.test.ts +71 -0
  250. package/src/map/__tests__/lifecycle-bridge.test.ts +86 -10
  251. package/src/map/acp-bridge.ts +26 -3
  252. package/src/map/cascade-action-handler.ts +205 -0
  253. package/src/map/cascade-bridge.ts +339 -0
  254. package/src/map/coordination-handler.ts +13 -1
  255. package/src/map/lifecycle-bridge.ts +52 -17
  256. package/src/map/server.ts +225 -7
  257. package/src/map/sidecar.ts +48 -1
  258. package/src/map/types.ts +23 -0
  259. package/src/mcp/tools/done-v2.ts +9 -0
  260. package/src/teams/team-manager-v2.ts +37 -0
  261. package/src/teams/team-runtime-v2.ts +23 -3
  262. package/src/workspace/__tests__/{dataplane-adapter.test.ts → git-cascade-adapter.test.ts} +209 -14
  263. package/src/workspace/__tests__/land-dispatch.test.ts +214 -0
  264. package/src/workspace/__tests__/self-driving-yaml.test.ts +114 -0
  265. package/src/workspace/__tests__/shared-worktree-refcount.test.ts +154 -0
  266. package/src/workspace/__tests__/standalone-mode.test.ts +118 -0
  267. package/src/workspace/__tests__/workspace-manager-v3.test.ts +245 -0
  268. package/src/workspace/__tests__/yaml-schema.test.ts +210 -0
  269. package/src/workspace/config.ts +11 -11
  270. package/src/workspace/git-cascade-adapter.ts +1213 -0
  271. package/src/workspace/index.ts +11 -11
  272. package/src/workspace/landing/__tests__/strategies.test.ts +184 -0
  273. package/src/workspace/landing/direct-push.ts +91 -0
  274. package/src/workspace/landing/index.ts +40 -0
  275. package/src/workspace/landing/merge-to-parent.ts +229 -0
  276. package/src/workspace/landing/optimistic-push.ts +36 -0
  277. package/src/workspace/landing/queue-to-branch.ts +108 -0
  278. package/src/workspace/merge-queue/merge-queue.ts +10 -0
  279. package/src/workspace/merge-queue/types.ts +16 -2
  280. package/src/workspace/pool/__tests__/worktree-pool.integration.test.ts +5 -5
  281. package/src/workspace/pool/types.ts +1 -0
  282. package/src/workspace/pool/worktree-pool.ts +1 -0
  283. package/src/workspace/recovery/__tests__/auto-resolve-integration.test.ts +127 -0
  284. package/src/workspace/recovery/__tests__/spawn-resolver.test.ts +139 -0
  285. package/src/workspace/recovery/__tests__/strategies.test.ts +145 -0
  286. package/src/workspace/recovery/abandon.ts +51 -0
  287. package/src/workspace/recovery/auto-resolve.ts +119 -0
  288. package/src/workspace/recovery/defer.ts +23 -0
  289. package/src/workspace/recovery/escalate.ts +30 -0
  290. package/src/workspace/recovery/index.ts +58 -0
  291. package/src/workspace/recovery/spawn-resolver.ts +152 -0
  292. package/src/workspace/recovery/types.ts +54 -0
  293. package/src/workspace/topology/__tests__/yaml-driven.test.ts +345 -0
  294. package/src/workspace/topology/index.ts +18 -0
  295. package/src/workspace/topology/no-workspace.ts +39 -0
  296. package/src/workspace/topology/types.ts +116 -0
  297. package/src/workspace/topology/yaml-driven.ts +316 -0
  298. package/src/workspace/types-v3.ts +162 -0
  299. package/src/workspace/types.ts +211 -20
  300. package/src/workspace/workspace-manager.ts +533 -19
  301. package/src/workspace/yaml-schema.ts +216 -0
  302. package/dist/workspace/dataplane-adapter.d.ts +0 -260
  303. package/dist/workspace/dataplane-adapter.d.ts.map +0 -1
  304. package/dist/workspace/dataplane-adapter.js +0 -416
  305. package/dist/workspace/dataplane-adapter.js.map +0 -1
  306. package/src/workspace/dataplane-adapter.ts +0 -546
@@ -0,0 +1,1213 @@
1
+ /**
2
+ * git-cascade Adapter
3
+ *
4
+ * Wraps git-cascade's MultiAgentRepoTracker to integrate with macro-agent's
5
+ * event system and provide a simplified interface for workspace management.
6
+ *
7
+ * @module workspace/git-cascade-adapter
8
+ */
9
+
10
+ import Database from 'better-sqlite3';
11
+ import {
12
+ MultiAgentRepoTracker,
13
+ type TrackerOptions,
14
+ type Stream,
15
+ type StreamStatus,
16
+ type StreamNode,
17
+ type CreateStreamOptions,
18
+ type ForkStreamOptions,
19
+ type MergeStreamOptions,
20
+ type MergeResult,
21
+ type RebaseOntoStreamOptions,
22
+ type RebaseResult,
23
+ type ConflictStrategy,
24
+ type ConflictRecord,
25
+ type CreateConflictOptions,
26
+ type AgentWorktree,
27
+ type CreateWorktreeOptions,
28
+ type WorkerTask,
29
+ type CreateTaskOptions,
30
+ type StartTaskOptions,
31
+ type CompleteTaskOptions,
32
+ type StartTaskResult,
33
+ type CompleteTaskResult,
34
+ type ListTasksOptions,
35
+ type CleanupWorkerBranchesOptions,
36
+ type CleanupResult,
37
+ type Checkpoint,
38
+ type Change,
39
+ type ChangeStatus,
40
+ workerTasks,
41
+ diffStacks,
42
+ mergeQueue as mergeQueueModule,
43
+ reconcile as reconcileModule,
44
+ cascade as cascadeModule,
45
+ matchCascadeSuffix,
46
+ type StreamOpenedParams,
47
+ type StreamCommittedParams,
48
+ type StreamMergedParams,
49
+ type StreamConflictedParams,
50
+ type StreamAbandonedParams,
51
+ type CascadeRebasedParams,
52
+ type CascadeCompletedParams,
53
+ type QueueAddedParams,
54
+ type QueueReadyParams,
55
+ type QueueCancelledParams,
56
+ type QueueRemovedParams,
57
+ } from 'git-cascade';
58
+ // git-cascade 0.0.3+ exposes both events (via `emit` callback) and the
59
+ // `cascade` namespace for cascadeRebase. All v3 primitives are now reachable.
60
+ import type { GitCascadeConfig } from './config.js';
61
+ import { DEFAULT_GIT_CASCADE_CONFIG } from './config.js';
62
+
63
+ /**
64
+ * Event types emitted by GitCascadeAdapter.
65
+ *
66
+ * Grouped by source:
67
+ * - `stream:*` — from git-cascade `x-cascade/stream.*` events + local emits
68
+ * - `worktree:*`, `task:*` — local emits (macro-agent-only concepts)
69
+ * - `change:*` — change-id lifecycle from git-cascade
70
+ * - `cascade:*` — cascadeRebase completion
71
+ * - `conflict:*` — conflict lifecycle
72
+ * - `mergeQueue:*` — built-in merge queue lifecycle
73
+ */
74
+ export type GitCascadeEventType =
75
+ | 'stream:created' // mapped from git-cascade stream.opened
76
+ | 'stream:updated' // local (updateStream)
77
+ | 'stream:forked' // local (forkStream)
78
+ | 'stream:committed' // mapped from git-cascade stream.committed
79
+ | 'stream:merged' // mapped from git-cascade stream.merged
80
+ | 'stream:conflicted' // mapped from git-cascade stream.conflicted
81
+ | 'stream:abandoned' // mapped from git-cascade stream.abandoned
82
+ | 'stream:paused' // local (pauseStream) + cascade emit
83
+ | 'stream:resumed' // local (resumeStream) + cascade emit
84
+ | 'stream:rolled_back' // cascade emit (rollbackN, rollbackToOperation, rollbackToForkPoint)
85
+ | 'worktree:created'
86
+ | 'worktree:deallocated'
87
+ | 'task:created'
88
+ | 'task:started'
89
+ | 'task:completed'
90
+ | 'task:abandoned'
91
+ | 'change:merged'
92
+ | 'change:dropped'
93
+ | 'stream:pushed' // local — emitted by direct-push/optimistic-push landing strategies
94
+ | 'cascade:rebased' // mapped from git-cascade cascade.rebased
95
+ | 'cascade:completed' // mapped from git-cascade cascade.completed
96
+ | 'conflict:created'
97
+ | 'conflict:resolved'
98
+ | 'mergeQueue:added'
99
+ | 'mergeQueue:ready'
100
+ | 'mergeQueue:cancelled'
101
+ | 'mergeQueue:removed';
102
+
103
+ /**
104
+ * Event payload for git-cascade events
105
+ */
106
+ export interface GitCascadeEvent {
107
+ type: GitCascadeEventType;
108
+ timestamp: number;
109
+ data: Record<string, unknown>;
110
+ }
111
+
112
+ /**
113
+ * Callback for git-cascade events
114
+ */
115
+ export type GitCascadeEventCallback = (event: GitCascadeEvent) => void;
116
+
117
+ /**
118
+ * GitCascadeAdapter wraps MultiAgentRepoTracker for macro-agent integration.
119
+ *
120
+ * Key responsibilities:
121
+ * - Initialize tracker with shared or dedicated database
122
+ * - Emit events on tracker operations
123
+ * - Provide simplified API for workspace management
124
+ */
125
+ export class GitCascadeAdapter {
126
+ private readonly tracker: MultiAgentRepoTracker;
127
+ private readonly config: Required<
128
+ Pick<GitCascadeConfig, 'enabled' | 'tablePrefix' | 'verbose' | 'skipRecovery'>
129
+ > & { repoPath: string };
130
+ private readonly eventListeners: Set<GitCascadeEventCallback> = new Set();
131
+ private readonly ownsDb: boolean;
132
+
133
+ /**
134
+ * Create a new GitCascadeAdapter.
135
+ *
136
+ * @param config - git-cascade configuration
137
+ */
138
+ constructor(config: GitCascadeConfig) {
139
+ const mergedConfig = {
140
+ ...DEFAULT_GIT_CASCADE_CONFIG,
141
+ ...config,
142
+ repoPath: config.repoPath ?? process.cwd(),
143
+ };
144
+
145
+ this.config = {
146
+ enabled: mergedConfig.enabled ?? true,
147
+ repoPath: mergedConfig.repoPath,
148
+ tablePrefix: mergedConfig.tablePrefix ?? 'git_cascade_',
149
+ verbose: mergedConfig.verbose ?? false,
150
+ skipRecovery: mergedConfig.skipRecovery ?? false,
151
+ };
152
+
153
+ // Determine if we own the database connection
154
+ this.ownsDb = !config.db;
155
+
156
+ const trackerOptions: TrackerOptions = {
157
+ repoPath: this.config.repoPath,
158
+ tablePrefix: this.config.tablePrefix,
159
+ verbose: this.config.verbose,
160
+ skipRecovery: this.config.skipRecovery,
161
+ // Wire git-cascade's native event emitter (0.0.2+) so stream lifecycle
162
+ // events are re-published through our own onEvent channel.
163
+ emit: (method: string, params: unknown) => this.forwardCascadeEvent(method, params),
164
+ };
165
+
166
+ if (config.db) {
167
+ trackerOptions.db = config.db;
168
+ } else if (config.dbPath) {
169
+ trackerOptions.dbPath = config.dbPath;
170
+ }
171
+ // If neither db nor dbPath provided, tracker uses default path
172
+
173
+ this.tracker = new MultiAgentRepoTracker(trackerOptions);
174
+ }
175
+
176
+
177
+ /**
178
+ * Get whether the adapter is enabled.
179
+ */
180
+ get enabled(): boolean {
181
+ return this.config.enabled;
182
+ }
183
+
184
+ /**
185
+ * Get the repository path.
186
+ */
187
+ get repoPath(): string {
188
+ return this.config.repoPath;
189
+ }
190
+
191
+ /**
192
+ * Get the underlying database connection.
193
+ * Use with caution - prefer adapter methods for operations.
194
+ */
195
+ get db(): Database.Database {
196
+ return this.tracker.db;
197
+ }
198
+
199
+ /**
200
+ * Get the underlying tracker.
201
+ * Use with caution - prefer adapter methods for operations.
202
+ */
203
+ get rawTracker(): MultiAgentRepoTracker {
204
+ return this.tracker;
205
+ }
206
+
207
+ // ─────────────────────────────────────────────────────────────────────────────
208
+ // Event System
209
+ // ─────────────────────────────────────────────────────────────────────────────
210
+
211
+ /**
212
+ * Subscribe to git-cascade events.
213
+ *
214
+ * @param callback - Function called when events occur
215
+ * @returns Unsubscribe function
216
+ */
217
+ onEvent(callback: GitCascadeEventCallback): () => void {
218
+ this.eventListeners.add(callback);
219
+ return () => this.eventListeners.delete(callback);
220
+ }
221
+
222
+ /**
223
+ * Emit an event to all listeners.
224
+ */
225
+ private emit(type: GitCascadeEventType, data: Record<string, unknown>): void {
226
+ const event: GitCascadeEvent = {
227
+ type,
228
+ timestamp: Date.now(),
229
+ data,
230
+ };
231
+ for (const listener of this.eventListeners) {
232
+ try {
233
+ listener(event);
234
+ } catch (error) {
235
+ console.error('[GitCascadeAdapter] Event listener error:', error);
236
+ }
237
+ }
238
+ }
239
+
240
+ /**
241
+ * Forward events emitted by git-cascade (`x-cascade/stream.*`) into our
242
+ * structured `GitCascadeEvent` stream. Called via the `emit` callback wired
243
+ * into the tracker constructor.
244
+ *
245
+ * Note: `stream.opened` is mapped to `stream:created`. The local
246
+ * wrapper methods still emit their own events so consumers don't miss out
247
+ * on operations that don't round-trip through git-cascade's emit (e.g.,
248
+ * updateStream, pauseStream, mergeQueue events).
249
+ */
250
+ private forwardCascadeEvent(method: string, params: unknown): void {
251
+ const suffix = matchCascadeSuffix(method);
252
+ if (!suffix) return;
253
+
254
+ // Widen to `string` so case labels for suffixes not yet in the installed
255
+ // git-cascade version (e.g. `stream.paused`, `stream.resumed`,
256
+ // `stream.rolled_back` — added after 0.0.7) still compile. The runtime
257
+ // value is always a string either way; this is a version-skew shim.
258
+ switch (suffix as string) {
259
+ case 'stream.opened': {
260
+ const p = params as StreamOpenedParams;
261
+ this.emit('stream:created', {
262
+ streamId: p.stream_id,
263
+ name: p.name,
264
+ agentId: p.agent_id,
265
+ baseCommit: p.base_commit,
266
+ parentStream: p.parent_stream,
267
+ branchName: p.branch_name,
268
+ metadata: p.metadata,
269
+ });
270
+ break;
271
+ }
272
+ case 'stream.committed': {
273
+ const p = params as StreamCommittedParams;
274
+ this.emit('stream:committed', {
275
+ streamId: p.stream_id,
276
+ commit: p.commit_hash,
277
+ changeId: p.change_id,
278
+ agentId: p.agent_id,
279
+ messageSummary: p.message_summary,
280
+ filesTouched: p.files_touched,
281
+ parentCommit: p.parent_commit,
282
+ metadata: p.metadata,
283
+ });
284
+ break;
285
+ }
286
+ case 'stream.merged': {
287
+ const p = params as StreamMergedParams;
288
+ this.emit('stream:merged', {
289
+ sourceStreamId: p.source_stream_id,
290
+ targetStreamId: p.target_stream_id,
291
+ mergeCommit: p.merge_commit,
292
+ agentId: p.agent_id,
293
+ strategy: p.strategy,
294
+ sourceCommit: p.source_commit,
295
+ metadata: p.metadata,
296
+ });
297
+ break;
298
+ }
299
+ case 'stream.conflicted': {
300
+ const p = params as StreamConflictedParams;
301
+ this.emit('stream:conflicted', {
302
+ streamId: p.stream_id,
303
+ conflictId: p.conflict_id,
304
+ conflictedFiles: p.conflicted_files,
305
+ agentId: p.agent_id,
306
+ conflictingCommit: p.conflicting_commit,
307
+ targetCommit: p.target_commit,
308
+ source: p.source,
309
+ metadata: p.metadata,
310
+ });
311
+ break;
312
+ }
313
+ case 'stream.conflict_resolved': {
314
+ const p = params as import('git-cascade').StreamConflictResolvedParams;
315
+ this.emit('conflict:resolved', {
316
+ streamId: p.stream_id,
317
+ conflictId: p.conflict_id,
318
+ resolutionMethod: p.resolution_method,
319
+ resolvedBy: p.resolved_by,
320
+ resolutionSummary: p.resolution_summary,
321
+ metadata: p.metadata,
322
+ });
323
+ break;
324
+ }
325
+ case 'stream.abandoned': {
326
+ const p = params as StreamAbandonedParams;
327
+ this.emit('stream:abandoned', {
328
+ streamId: p.stream_id,
329
+ reason: p.reason,
330
+ cascade: p.cascade,
331
+ metadata: p.metadata,
332
+ });
333
+ break;
334
+ }
335
+ case 'cascade.rebased': {
336
+ const p = params as CascadeRebasedParams;
337
+ this.emit('cascade:rebased', {
338
+ streamId: p.stream_id,
339
+ agentId: p.agent_id,
340
+ triggeredByStreamId: p.triggered_by_stream_id,
341
+ triggeredByAgentId: p.triggered_by_agent_id,
342
+ newBaseCommit: p.new_base_commit,
343
+ newHead: p.new_head,
344
+ newCommits: p.new_commits,
345
+ metadata: p.metadata,
346
+ });
347
+ break;
348
+ }
349
+ case 'cascade.completed': {
350
+ const p = params as CascadeCompletedParams;
351
+ this.emit('cascade:completed', {
352
+ rootStreamId: p.root_stream_id,
353
+ agentId: p.agent_id,
354
+ strategy: p.strategy,
355
+ updatedStreams: p.updated_streams,
356
+ failedStreams: p.failed_streams,
357
+ skippedStreams: p.skipped_streams,
358
+ deferredStreams: p.deferred_streams,
359
+ metadata: p.metadata,
360
+ });
361
+ break;
362
+ }
363
+ case 'queue.added': {
364
+ const p = params as QueueAddedParams;
365
+ this.emit('mergeQueue:added', {
366
+ entryId: p.entry_id,
367
+ streamId: p.stream_id,
368
+ targetBranch: p.target_branch,
369
+ metadata: p.metadata,
370
+ });
371
+ break;
372
+ }
373
+ case 'queue.ready': {
374
+ const p = params as QueueReadyParams;
375
+ this.emit('mergeQueue:ready', {
376
+ entryId: p.entry_id,
377
+ streamId: p.stream_id,
378
+ targetBranch: p.target_branch,
379
+ });
380
+ break;
381
+ }
382
+ case 'queue.cancelled': {
383
+ const p = params as QueueCancelledParams;
384
+ this.emit('mergeQueue:cancelled', {
385
+ entryId: p.entry_id,
386
+ streamId: p.stream_id,
387
+ targetBranch: p.target_branch,
388
+ reason: p.reason,
389
+ });
390
+ break;
391
+ }
392
+ case 'queue.removed': {
393
+ const p = params as QueueRemovedParams;
394
+ this.emit('mergeQueue:removed', {
395
+ entryId: p.entry_id,
396
+ streamId: p.stream_id,
397
+ targetBranch: p.target_branch,
398
+ outcome: p.outcome,
399
+ });
400
+ break;
401
+ }
402
+ case 'stream.paused': {
403
+ // Tracker fires stream.paused after pauseStream. The adapter also
404
+ // emits stream:paused locally in its pauseStream() wrapper — the
405
+ // bridge uses the local event (not this cascade-forwarded one) for
406
+ // the MAP translation, so this case is mainly to keep the switch
407
+ // exhaustive. No double-emit: the bridge deduplicates by event type.
408
+ break;
409
+ }
410
+ case 'stream.resumed': {
411
+ // Same as stream.paused — adapter.resumeStream() fires the local event.
412
+ break;
413
+ }
414
+ case 'stream.rolled_back': {
415
+ const p = params as { stream_id: string; strategy?: string; target?: string | number; new_head?: string };
416
+ this.emit('stream:rolled_back', {
417
+ streamId: p.stream_id,
418
+ strategy: p.strategy,
419
+ target: p.target,
420
+ newHead: p.new_head,
421
+ });
422
+ break;
423
+ }
424
+ }
425
+ }
426
+
427
+ // ─────────────────────────────────────────────────────────────────────────────
428
+ // Stream Operations
429
+ // ─────────────────────────────────────────────────────────────────────────────
430
+
431
+ /**
432
+ * Create a new stream (integration branch).
433
+ *
434
+ * Note: `stream:created` is emitted by the cascade event forwarder via
435
+ * git-cascade's `x-cascade/stream.opened`. We don't double-emit here.
436
+ *
437
+ * @param options - Stream creation options
438
+ * @returns Stream ID
439
+ */
440
+ createStream(options: CreateStreamOptions): string {
441
+ return this.tracker.createStream(options);
442
+ }
443
+
444
+ /**
445
+ * Get a stream by ID.
446
+ */
447
+ getStream(streamId: string): Stream | null {
448
+ return this.tracker.getStream(streamId);
449
+ }
450
+
451
+ /**
452
+ * List streams with optional filters.
453
+ */
454
+ listStreams(options?: { agentId?: string; status?: StreamStatus }): Stream[] {
455
+ return this.tracker.listStreams(options);
456
+ }
457
+
458
+ /**
459
+ * Update a stream.
460
+ */
461
+ updateStream(
462
+ streamId: string,
463
+ updates: Partial<Pick<Stream, 'name' | 'status' | 'metadata'>>
464
+ ): void {
465
+ this.tracker.updateStream(streamId, updates);
466
+ this.emit('stream:updated', { streamId, updates });
467
+ }
468
+
469
+ /**
470
+ * Abandon a stream.
471
+ *
472
+ * Note: `stream:abandoned` is emitted by the cascade event forwarder.
473
+ */
474
+ abandonStream(
475
+ streamId: string,
476
+ options?: { reason?: string; cascade?: boolean }
477
+ ): void {
478
+ this.tracker.abandonStream(streamId, options);
479
+ }
480
+
481
+ /**
482
+ * Get the git branch name for a stream.
483
+ */
484
+ getStreamBranchName(streamId: string): string {
485
+ return this.tracker.getStreamBranchName(streamId);
486
+ }
487
+
488
+ /**
489
+ * Get the HEAD commit of a stream.
490
+ */
491
+ getStreamHead(streamId: string): string {
492
+ return this.tracker.getStreamHead(streamId);
493
+ }
494
+
495
+ /**
496
+ * Fork a child stream off a parent.
497
+ */
498
+ forkStream(options: ForkStreamOptions): string {
499
+ const streamId = this.tracker.forkStream(options);
500
+ this.emit('stream:forked', {
501
+ streamId,
502
+ parentStreamId: options.parentStreamId,
503
+ name: options.name,
504
+ agentId: options.agentId,
505
+ });
506
+ return streamId;
507
+ }
508
+
509
+ /**
510
+ * Merge source stream into target stream.
511
+ *
512
+ * Note: `stream:merged` is emitted by the cascade event forwarder on success.
513
+ * On conflict, `stream:conflicted` is also forwarded.
514
+ */
515
+ mergeStream(options: MergeStreamOptions): MergeResult {
516
+ return this.tracker.mergeStream(options);
517
+ }
518
+
519
+ /**
520
+ * Rebase a stream onto its parent to pick up new commits.
521
+ */
522
+ syncWithParent(
523
+ streamId: string,
524
+ agentId: string,
525
+ worktree: string,
526
+ onConflict?: ConflictStrategy
527
+ ): RebaseResult {
528
+ return this.tracker.syncWithParent(streamId, agentId, worktree, onConflict);
529
+ }
530
+
531
+ /**
532
+ * Rebase a stream onto a specific target stream.
533
+ */
534
+ rebaseOntoStream(options: RebaseOntoStreamOptions): RebaseResult {
535
+ return this.tracker.rebaseOntoStream(options);
536
+ }
537
+
538
+ /**
539
+ * Async version of rebaseOntoStream — supports async conflict handlers.
540
+ */
541
+ rebaseOntoStreamAsync(options: RebaseOntoStreamOptions): Promise<RebaseResult> {
542
+ return this.tracker.rebaseOntoStreamAsync(options);
543
+ }
544
+
545
+ /**
546
+ * Pause a stream (halt work without abandoning).
547
+ */
548
+ pauseStream(streamId: string, reason?: string): void {
549
+ this.tracker.pauseStream(streamId, reason);
550
+ this.emit('stream:paused', { streamId, reason });
551
+ }
552
+
553
+ /**
554
+ * Resume a paused stream.
555
+ */
556
+ resumeStream(streamId: string): void {
557
+ this.tracker.resumeStream(streamId);
558
+ this.emit('stream:resumed', { streamId });
559
+ }
560
+
561
+ /**
562
+ * Track an existing branch as a stream (local mode — no new `stream/<id>` branch).
563
+ */
564
+ trackExistingBranch(options: Parameters<MultiAgentRepoTracker['trackExistingBranch']>[0]): string {
565
+ return this.tracker.trackExistingBranch(options);
566
+ }
567
+
568
+ /**
569
+ * Get stream hierarchy as a tree.
570
+ */
571
+ getStreamHierarchy(rootStreamId?: string): StreamNode | StreamNode[] {
572
+ return this.tracker.getStreamHierarchy(rootStreamId);
573
+ }
574
+
575
+ /**
576
+ * Get child streams (direct children only).
577
+ */
578
+ getChildStreams(streamId: string): Stream[] {
579
+ return this.tracker.getChildStreams(streamId);
580
+ }
581
+
582
+ /**
583
+ * Find the common ancestor of two streams.
584
+ */
585
+ findCommonAncestor(streamIdA: string, streamIdB: string): string {
586
+ return this.tracker.findCommonAncestor(streamIdA, streamIdB);
587
+ }
588
+
589
+ // ─────────────────────────────────────────────────────────────────────────────
590
+ // Stream Dependencies
591
+ // ─────────────────────────────────────────────────────────────────────────────
592
+
593
+ /**
594
+ * Declare that one stream depends on another.
595
+ */
596
+ addDependency(streamId: string, dependsOnId: string): void {
597
+ this.tracker.addDependency(streamId, dependsOnId);
598
+ }
599
+
600
+ /**
601
+ * Remove a dependency declaration.
602
+ */
603
+ removeDependency(streamId: string, dependsOnId: string): void {
604
+ this.tracker.removeDependency(streamId, dependsOnId);
605
+ }
606
+
607
+ /**
608
+ * Get direct dependencies of a stream.
609
+ */
610
+ getDependencies(streamId: string): string[] {
611
+ return this.tracker.getDependencies(streamId);
612
+ }
613
+
614
+ /**
615
+ * Get direct dependents of a stream.
616
+ */
617
+ getDependents(streamId: string): string[] {
618
+ return this.tracker.getDependents(streamId);
619
+ }
620
+
621
+ // ─────────────────────────────────────────────────────────────────────────────
622
+ // Cascade Rebase (git-cascade 0.0.3+)
623
+ // ─────────────────────────────────────────────────────────────────────────────
624
+
625
+ /**
626
+ * Cascade-rebase all dependents of a root stream.
627
+ *
628
+ * Propagates rebases through the dependency graph. Uses a callback-based
629
+ * worktree provider so not every dependent needs a pre-allocated worktree.
630
+ *
631
+ * @param options - Cascade options including root stream, agent id, and worktree provider
632
+ * @returns CascadeResult with updated/failed/skipped stream lists
633
+ */
634
+ cascadeRebase(
635
+ options: cascadeModule.CascadeRebaseOptions
636
+ ): ReturnType<typeof cascadeModule.cascadeRebase> {
637
+ // Use tracker.cascadeRebase which threads the tracker's emit + eventPrefix
638
+ // into the cascade walk so `cascade.rebased` (per dependent) and
639
+ // `cascade.completed` (at end) both round-trip through our
640
+ // forwardCascadeEvent. No manual emit needed — events are driven by
641
+ // git-cascade 0.0.4+ from inside the walk.
642
+ return this.tracker.cascadeRebase(options);
643
+ }
644
+
645
+ // ─────────────────────────────────────────────────────────────────────────────
646
+ // Worktree Operations
647
+ // ─────────────────────────────────────────────────────────────────────────────
648
+
649
+ /**
650
+ * Create a worktree for an agent.
651
+ *
652
+ * @param options - Worktree creation options
653
+ * @returns Created worktree info
654
+ */
655
+ createWorktree(options: CreateWorktreeOptions): AgentWorktree {
656
+ const worktree = this.tracker.createWorktree(options);
657
+ this.emit('worktree:created', { ...worktree });
658
+ return worktree;
659
+ }
660
+
661
+ /**
662
+ * Get a worktree by agent ID.
663
+ */
664
+ getWorktree(agentId: string): AgentWorktree | null {
665
+ return this.tracker.getWorktree(agentId);
666
+ }
667
+
668
+ /**
669
+ * List all worktrees.
670
+ */
671
+ listWorktrees(): AgentWorktree[] {
672
+ return this.tracker.listWorktrees();
673
+ }
674
+
675
+ /**
676
+ * Update the stream associated with a worktree.
677
+ */
678
+ updateWorktreeStream(agentId: string, streamId: string | null): void {
679
+ this.tracker.updateWorktreeStream(agentId, streamId);
680
+ }
681
+
682
+ /**
683
+ * Deallocate a worktree.
684
+ */
685
+ deallocateWorktree(agentId: string): void {
686
+ this.tracker.deallocateWorktree(agentId);
687
+ this.emit('worktree:deallocated', { agentId });
688
+ }
689
+
690
+ // ─────────────────────────────────────────────────────────────────────────────
691
+ // Worker Task Operations
692
+ // ─────────────────────────────────────────────────────────────────────────────
693
+
694
+ /**
695
+ * Create a worker task under a stream.
696
+ *
697
+ * @param options - Task creation options
698
+ * @returns Task ID
699
+ */
700
+ createTask(options: CreateTaskOptions): string {
701
+ const taskId = this.tracker.createTask(options);
702
+ this.emit('task:created', { taskId, ...options });
703
+ return taskId;
704
+ }
705
+
706
+ /**
707
+ * Get a task by ID.
708
+ */
709
+ getTask(taskId: string): WorkerTask | null {
710
+ return this.tracker.getTask(taskId);
711
+ }
712
+
713
+ /**
714
+ * List tasks for a stream.
715
+ */
716
+ listTasks(streamId: string, options?: ListTasksOptions): WorkerTask[] {
717
+ return this.tracker.listTasks(streamId, options);
718
+ }
719
+
720
+ /**
721
+ * Start a task - assigns agent and creates worker branch.
722
+ *
723
+ * @param options - Start task options
724
+ * @returns Branch name and start commit
725
+ */
726
+ startTask(options: StartTaskOptions): StartTaskResult {
727
+ const result = this.tracker.startTask(options);
728
+ this.emit('task:started', {
729
+ taskId: options.taskId,
730
+ agentId: options.agentId,
731
+ branchName: result.branchName,
732
+ startCommit: result.startCommit,
733
+ });
734
+ return result;
735
+ }
736
+
737
+ /**
738
+ * Complete a task - merges worker branch to stream.
739
+ *
740
+ * @param options - Complete task options
741
+ * @returns Merge result
742
+ */
743
+ completeTask(options: CompleteTaskOptions): CompleteTaskResult {
744
+ const result = this.tracker.completeTask(options);
745
+ this.emit('task:completed', { taskId: options.taskId, ...result });
746
+ return result;
747
+ }
748
+
749
+ /**
750
+ * Abandon a task.
751
+ *
752
+ * @param taskId - Task ID
753
+ * @param options - Options
754
+ */
755
+ abandonTask(taskId: string, options?: { deleteBranch?: boolean }): void {
756
+ this.tracker.abandonTask(taskId, options);
757
+ this.emit('task:abandoned', { taskId, ...options });
758
+ }
759
+
760
+ /**
761
+ * Release a task back to 'open' status.
762
+ */
763
+ releaseTask(taskId: string): void {
764
+ this.tracker.releaseTask(taskId);
765
+ }
766
+
767
+ /**
768
+ * Detect conflicts for a task before completing.
769
+ *
770
+ * @param taskId - Task ID
771
+ * @param worktree - Worktree path
772
+ * @returns Array of conflicting file paths, empty if no conflicts
773
+ */
774
+ detectTaskConflicts(taskId: string, worktree: string): string[] {
775
+ return workerTasks.detectTaskConflicts(
776
+ this.tracker.db,
777
+ this.config.repoPath,
778
+ taskId,
779
+ worktree
780
+ );
781
+ }
782
+
783
+ /**
784
+ * Recover stale tasks that have been in_progress too long.
785
+ *
786
+ * @param thresholdMs - Tasks older than this are considered stale
787
+ * @returns Result with released task IDs
788
+ */
789
+ recoverStaleTasks(thresholdMs: number = 60 * 60 * 1000): { released: string[] } {
790
+ return workerTasks.recoverStaleTasks(this.tracker.db, thresholdMs);
791
+ }
792
+
793
+ // ─────────────────────────────────────────────────────────────────────────────
794
+ // Checkpoint Operations
795
+ // ─────────────────────────────────────────────────────────────────────────────
796
+
797
+ /**
798
+ * Create checkpoints for commits made during a task.
799
+ *
800
+ * Creates a checkpoint for each commit between the task's startCommit and
801
+ * the current HEAD of the task's stream. This captures the work done during
802
+ * the task for future review and merge workflows.
803
+ *
804
+ * @param taskId - Task ID to create checkpoints for
805
+ * @param agentId - Agent ID (used as createdBy)
806
+ * @returns Array of created checkpoints
807
+ */
808
+ createCheckpointsForTask(taskId: string, agentId: string): Checkpoint[] {
809
+ const task = this.getTask(taskId);
810
+ if (!task) {
811
+ console.warn(`[GitCascadeAdapter] Task not found: ${taskId}`);
812
+ return [];
813
+ }
814
+
815
+ if (!task.streamId) {
816
+ console.warn(`[GitCascadeAdapter] Task ${taskId} has no streamId`);
817
+ return [];
818
+ }
819
+
820
+ if (!task.startCommit) {
821
+ console.warn(`[GitCascadeAdapter] Task ${taskId} has no startCommit`);
822
+ return [];
823
+ }
824
+
825
+ try {
826
+ // Create checkpoints from task's startCommit to stream's current HEAD
827
+ const checkpoints = diffStacks.createCheckpointsFromStream(
828
+ this.tracker.db,
829
+ this.config.repoPath,
830
+ task.streamId,
831
+ {
832
+ from: task.startCommit,
833
+ createdBy: agentId,
834
+ }
835
+ );
836
+
837
+ return checkpoints;
838
+ } catch (error) {
839
+ console.error(
840
+ `[GitCascadeAdapter] Failed to create checkpoints for task ${taskId}:`,
841
+ error
842
+ );
843
+ return [];
844
+ }
845
+ }
846
+
847
+ // ─────────────────────────────────────────────────────────────────────────────
848
+ // Commit Operations
849
+ // ─────────────────────────────────────────────────────────────────────────────
850
+
851
+ /**
852
+ * Commit changes in a worktree with Change tracking.
853
+ *
854
+ * @param options - Commit options
855
+ * @returns Commit hash and change ID
856
+ */
857
+ commitChanges(options: {
858
+ streamId: string;
859
+ agentId: string;
860
+ worktree: string;
861
+ message: string;
862
+ /**
863
+ * Optional metadata threaded verbatim to git-cascade's
864
+ * `x-cascade/stream.committed` event. Use `{ task_ref: { resource_id,
865
+ * node_id } }` to bind this commit to an external task (see
866
+ * `SpawnAgentOptions.taskRef`). Each commit can carry a distinct ref —
867
+ * useful for workers handling multiple sub-tasks within one session.
868
+ */
869
+ metadata?: Record<string, unknown>;
870
+ }): { commit: string; changeId: string } {
871
+ return this.tracker.commitChanges(options);
872
+ }
873
+
874
+ // ─────────────────────────────────────────────────────────────────────────────
875
+ // Maintenance Operations
876
+ // ─────────────────────────────────────────────────────────────────────────────
877
+
878
+ /**
879
+ * Clean up old worker branches.
880
+ *
881
+ * Deletes branches for:
882
+ * - Completed tasks older than threshold (default 24h)
883
+ * - Abandoned tasks
884
+ * - Orphaned branches (no task record)
885
+ *
886
+ * @param options - Cleanup options
887
+ * @returns Deleted branches and any errors
888
+ */
889
+ cleanupWorkerBranches(options?: CleanupWorkerBranchesOptions): CleanupResult {
890
+ return workerTasks.cleanupWorkerBranches(
891
+ this.tracker.db,
892
+ this.config.repoPath,
893
+ options
894
+ );
895
+ }
896
+
897
+ /**
898
+ * Delete a specific worker branch.
899
+ *
900
+ * @param branchName - Branch name to delete
901
+ * @returns true if deleted, false if branch didn't exist
902
+ */
903
+ deleteWorkerBranch(branchName: string): boolean {
904
+ try {
905
+ const { execSync } = require('child_process');
906
+ execSync(`git branch -D "${branchName}"`, {
907
+ cwd: this.config.repoPath,
908
+ stdio: 'pipe',
909
+ });
910
+ return true;
911
+ } catch {
912
+ return false;
913
+ }
914
+ }
915
+
916
+ // ─────────────────────────────────────────────────────────────────────────────
917
+ // Change Operations (Change-Id tracking)
918
+ // ─────────────────────────────────────────────────────────────────────────────
919
+
920
+ /**
921
+ * Get a change by ID.
922
+ */
923
+ getChange(changeId: string): Change | null {
924
+ return this.tracker.getChange(changeId);
925
+ }
926
+
927
+ /**
928
+ * Get a change by its current commit hash.
929
+ */
930
+ getChangeByCommit(commit: string): Change | null {
931
+ return this.tracker.getChangeByCommit(commit);
932
+ }
933
+
934
+ /**
935
+ * Get a change by any of its historical commit hashes (survives rebases).
936
+ */
937
+ getChangeByHistoricalCommit(commit: string): Change | null {
938
+ return this.tracker.getChangeByHistoricalCommit(commit);
939
+ }
940
+
941
+ /**
942
+ * List changes for a stream, optionally filtered by status.
943
+ */
944
+ getChangesForStream(
945
+ streamId: string,
946
+ options?: { status?: ChangeStatus }
947
+ ): Change[] {
948
+ return this.tracker.getChangesForStream(streamId, options);
949
+ }
950
+
951
+ /**
952
+ * Mark changes as merged.
953
+ */
954
+ markChangesMerged(changeIds: string[]): void {
955
+ this.tracker.markChangesMerged(changeIds);
956
+ for (const id of changeIds) {
957
+ this.emit('change:merged', { changeId: id });
958
+ }
959
+ }
960
+
961
+ /**
962
+ * Mark a single change as dropped.
963
+ */
964
+ markChangeDropped(changeId: string): void {
965
+ this.tracker.markChangeDropped(changeId);
966
+ this.emit('change:dropped', { changeId });
967
+ }
968
+
969
+ // ─────────────────────────────────────────────────────────────────────────────
970
+ // Merge Queue (git-cascade built-in)
971
+ // ─────────────────────────────────────────────────────────────────────────────
972
+
973
+ /**
974
+ * Add a stream to the merge queue. Local `mergeQueue:added` event fires
975
+ * via forwardCascadeEvent (git-cascade 0.0.7+ emits queue.added natively).
976
+ */
977
+ addToMergeQueue(options: mergeQueueModule.AddToQueueOptions): string {
978
+ return this.tracker.addToMergeQueue(options);
979
+ }
980
+
981
+ /**
982
+ * Get a merge queue entry by id.
983
+ */
984
+ getMergeQueueEntry(entryId: string): mergeQueueModule.MergeQueueEntry | null {
985
+ return this.tracker.getMergeQueueEntry(entryId);
986
+ }
987
+
988
+ /**
989
+ * List merge queue entries with optional filters.
990
+ */
991
+ listMergeQueue(
992
+ options?: {
993
+ targetBranch?: string;
994
+ status?: mergeQueueModule.MergeQueueStatus | mergeQueueModule.MergeQueueStatus[];
995
+ }
996
+ ): mergeQueueModule.MergeQueueEntry[] {
997
+ return this.tracker.getMergeQueue(options);
998
+ }
999
+
1000
+ /**
1001
+ * Mark a queue entry as ready to merge. Local `mergeQueue:ready` event
1002
+ * fires via forwardCascadeEvent (git-cascade 0.0.7+).
1003
+ */
1004
+ markMergeQueueReady(entryId: string): void {
1005
+ this.tracker.markMergeQueueReady(entryId);
1006
+ }
1007
+
1008
+ /**
1009
+ * Cancel a queue entry. Local `mergeQueue:cancelled` event fires via
1010
+ * forwardCascadeEvent (git-cascade 0.0.7+).
1011
+ */
1012
+ cancelMergeQueueEntry(entryId: string): void {
1013
+ this.tracker.cancelMergeQueueEntry(entryId);
1014
+ }
1015
+
1016
+ /**
1017
+ * Remove a queue entry. Local `mergeQueue:removed` event fires via
1018
+ * forwardCascadeEvent (git-cascade 0.0.7+).
1019
+ */
1020
+ removeFromMergeQueue(entryId: string): void {
1021
+ this.tracker.removeFromMergeQueue(entryId);
1022
+ }
1023
+
1024
+ /**
1025
+ * Get the next entry to process for a target branch.
1026
+ */
1027
+ getNextToMerge(targetBranch?: string): mergeQueueModule.MergeQueueEntry | null {
1028
+ return this.tracker.getNextToMerge(targetBranch);
1029
+ }
1030
+
1031
+ /**
1032
+ * Process the merge queue — drains ready entries per provided handler.
1033
+ */
1034
+ processMergeQueue(
1035
+ options: mergeQueueModule.ProcessQueueOptions
1036
+ ): mergeQueueModule.ProcessQueueResult {
1037
+ return this.tracker.processMergeQueue(options);
1038
+ }
1039
+
1040
+ /**
1041
+ * Get a stream's position in the queue (lower = sooner).
1042
+ */
1043
+ getMergeQueuePosition(streamId: string, targetBranch?: string): number | null {
1044
+ return this.tracker.getMergeQueuePosition(streamId, targetBranch);
1045
+ }
1046
+
1047
+ // ─────────────────────────────────────────────────────────────────────────────
1048
+ // Conflict Operations
1049
+ // ─────────────────────────────────────────────────────────────────────────────
1050
+
1051
+ /**
1052
+ * Create a conflict record.
1053
+ *
1054
+ * Usually conflicts are created implicitly by merge/rebase operations.
1055
+ * This is for explicit creation (e.g., an external process detected a
1056
+ * conflict that git-cascade didn't).
1057
+ */
1058
+ createConflict(options: CreateConflictOptions): string {
1059
+ const id = this.tracker.createConflict(options);
1060
+ this.emit('conflict:created', {
1061
+ conflictId: id,
1062
+ streamId: options.streamId,
1063
+ });
1064
+ return id;
1065
+ }
1066
+
1067
+ /**
1068
+ * Get a conflict record by id.
1069
+ */
1070
+ getConflict(conflictId: string): ConflictRecord | null {
1071
+ return this.tracker.getConflict(conflictId);
1072
+ }
1073
+
1074
+ /**
1075
+ * Get the active conflict record for a stream, if any.
1076
+ */
1077
+ getConflictForStream(streamId: string): ConflictRecord | null {
1078
+ return this.tracker.getConflictForStream(streamId);
1079
+ }
1080
+
1081
+ /**
1082
+ * Emit a `stream:pushed` event for trunk-style landing strategies that push
1083
+ * to a remote rather than merging into another stream. Strategies call this
1084
+ * after a successful push so observers (the OpenHive cascade-bridge) can
1085
+ * surface the push as `x-cascade/stream.pushed`.
1086
+ */
1087
+ notifyStreamPushed(args: {
1088
+ streamId: string;
1089
+ agentId: string;
1090
+ pushedCommit: string;
1091
+ remote: string;
1092
+ remoteRef: string;
1093
+ strategy?: string;
1094
+ metadata?: Record<string, unknown>;
1095
+ }): void {
1096
+ this.emit('stream:pushed', {
1097
+ streamId: args.streamId,
1098
+ agentId: args.agentId,
1099
+ pushedCommit: args.pushedCommit,
1100
+ remote: args.remote,
1101
+ remoteRef: args.remoteRef,
1102
+ strategy: args.strategy,
1103
+ metadata: args.metadata,
1104
+ });
1105
+ }
1106
+
1107
+ /**
1108
+ * Mark a conflict as resolved. Routes through git-cascade's tracker so
1109
+ * `x-cascade/stream.conflict_resolved` fires for hub observers (closes
1110
+ * cascade_conflicts.status from pending → resolved on the OpenHive side).
1111
+ */
1112
+ resolveConflict(args: {
1113
+ conflictId: string;
1114
+ resolution: import('git-cascade').ConflictResolution & { summary?: string };
1115
+ metadata?: Record<string, unknown>;
1116
+ }): void {
1117
+ this.tracker.resolveConflict(args.conflictId, args.resolution, {
1118
+ metadata: args.metadata,
1119
+ });
1120
+ }
1121
+
1122
+ /**
1123
+ * Abandon a conflict (stream-level give-up). Emits a
1124
+ * `stream.conflict_resolved` event with method='abandoned' so observers
1125
+ * see the conflict is no longer pending.
1126
+ */
1127
+ abandonConflict(args: {
1128
+ conflictId: string;
1129
+ agentId?: string;
1130
+ reason?: string;
1131
+ metadata?: Record<string, unknown>;
1132
+ }): void {
1133
+ this.tracker.abandonConflict(args.conflictId, {
1134
+ agentId: args.agentId,
1135
+ reason: args.reason,
1136
+ metadata: args.metadata,
1137
+ });
1138
+ }
1139
+
1140
+ // ─────────────────────────────────────────────────────────────────────────────
1141
+ // Reconciliation
1142
+ // ─────────────────────────────────────────────────────────────────────────────
1143
+
1144
+ /**
1145
+ * Check if a stream's database state is in sync with its git branch.
1146
+ */
1147
+ checkStreamSync(streamId: string): reconcileModule.StreamSyncStatus {
1148
+ return this.tracker.checkStreamSync(streamId);
1149
+ }
1150
+
1151
+ /**
1152
+ * Check all active streams for sync status.
1153
+ */
1154
+ checkAllStreamsSync(
1155
+ options?: { streamIds?: string[] }
1156
+ ): reconcileModule.ReconcileCheckResult {
1157
+ return this.tracker.checkAllStreamsSync(options);
1158
+ }
1159
+
1160
+ /**
1161
+ * Reconcile database state with git state. Fixes missing branches, resets
1162
+ * diverged HEAD (per options), etc. Does NOT handle orphan worktrees — the
1163
+ * macro-agent-level reconcile wrapper covers that.
1164
+ */
1165
+ reconcile(
1166
+ options?: reconcileModule.ReconcileOptions
1167
+ ): reconcileModule.ReconcileResult {
1168
+ return this.tracker.reconcile(options);
1169
+ }
1170
+
1171
+ /**
1172
+ * Ensure a stream is in sync before performing an operation.
1173
+ * @throws DesyncError if out of sync unless `force: true`.
1174
+ */
1175
+ ensureStreamInSync(streamId: string, options?: { force?: boolean }): void {
1176
+ this.tracker.ensureStreamInSync(streamId, options);
1177
+ }
1178
+
1179
+ // ─────────────────────────────────────────────────────────────────────────────
1180
+ // Health & Recovery
1181
+ // ─────────────────────────────────────────────────────────────────────────────
1182
+
1183
+ /**
1184
+ * Check system health.
1185
+ */
1186
+ healthCheck(): ReturnType<MultiAgentRepoTracker['healthCheck']> {
1187
+ return this.tracker.healthCheck();
1188
+ }
1189
+
1190
+ // ─────────────────────────────────────────────────────────────────────────────
1191
+ // Lifecycle
1192
+ // ─────────────────────────────────────────────────────────────────────────────
1193
+
1194
+ /**
1195
+ * Close the adapter and release resources.
1196
+ *
1197
+ * Only closes the database if we created it (not if using shared DB).
1198
+ */
1199
+ close(): void {
1200
+ this.eventListeners.clear();
1201
+ this.tracker.close();
1202
+ }
1203
+ }
1204
+
1205
+ /**
1206
+ * Create a GitCascadeAdapter instance.
1207
+ *
1208
+ * @param config - Configuration options
1209
+ * @returns GitCascadeAdapter instance
1210
+ */
1211
+ export function createGitCascadeAdapter(config: GitCascadeConfig): GitCascadeAdapter {
1212
+ return new GitCascadeAdapter(config);
1213
+ }