gsd-pi 2.79.0-dev.ece5fd8ba → 2.80.0-dev.b62439b9f

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 (607) hide show
  1. package/dist/{cli-web-branch.d.ts → cli/cli-web-branch.d.ts} +1 -1
  2. package/dist/{cli-web-branch.js → cli/cli-web-branch.js} +3 -3
  3. package/dist/{cli.js → cli/cli.js} +25 -25
  4. package/dist/{web-mode.js → cli/web-mode.js} +3 -3
  5. package/dist/{extension-registry.js → extension-runtime/extension-registry.js} +1 -1
  6. package/dist/{headless-query.d.ts → headless/headless-query.d.ts} +1 -1
  7. package/dist/{headless-query.js → headless/headless-query.js} +3 -2
  8. package/dist/{headless-recover.js → headless/headless-recover.js} +3 -2
  9. package/dist/{headless.js → headless/headless.js} +3 -3
  10. package/dist/loader.js +10 -10
  11. package/dist/{onboarding.js → onboarding/onboarding.js} +8 -8
  12. package/dist/{welcome-screen.js → onboarding/welcome-screen.js} +1 -1
  13. package/dist/{models-resolver.js → providers/models-resolver.js} +1 -1
  14. package/dist/{remote-questions-config.js → providers/remote-questions-config.js} +1 -1
  15. package/dist/{resource-loader.d.ts → resource-runtime/resource-loader.d.ts} +1 -1
  16. package/dist/{resource-loader.js → resource-runtime/resource-loader.js} +6 -6
  17. package/dist/resources/.managed-resources-content-hash +1 -1
  18. package/dist/resources/extensions/github-sync/templates.js +55 -70
  19. package/dist/resources/extensions/gsd/auto/contracts.js +1 -0
  20. package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +53 -0
  21. package/dist/resources/extensions/gsd/auto/loop.js +362 -523
  22. package/dist/resources/extensions/gsd/auto/orchestrator.js +146 -0
  23. package/dist/resources/extensions/gsd/auto/phases.js +55 -6
  24. package/dist/resources/extensions/gsd/auto/session.js +8 -0
  25. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-dispatch-outcome.js +12 -0
  26. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-iteration.js +24 -0
  27. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-reconcile-outcome.js +33 -0
  28. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-reconcile.js +26 -0
  29. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-retry.js +49 -0
  30. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-verify-outcome.js +25 -0
  31. package/dist/resources/extensions/gsd/auto/workflow-dispatch-claim.js +48 -0
  32. package/dist/resources/extensions/gsd/auto/workflow-dispatch-ledger.js +26 -0
  33. package/dist/resources/extensions/gsd/auto/workflow-iteration-completion.js +10 -0
  34. package/dist/resources/extensions/gsd/auto/workflow-journal-reporter.js +16 -0
  35. package/dist/resources/extensions/gsd/auto/workflow-kernel.js +263 -0
  36. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +36 -0
  37. package/dist/resources/extensions/gsd/auto/workflow-phase-reporter.js +9 -0
  38. package/dist/resources/extensions/gsd/auto/workflow-session-lock.js +35 -0
  39. package/dist/resources/extensions/gsd/auto/workflow-sidecar-iteration.js +24 -0
  40. package/dist/resources/extensions/gsd/auto/workflow-sidecar-queue.js +26 -0
  41. package/dist/resources/extensions/gsd/auto/workflow-turn-reporter.js +36 -0
  42. package/dist/resources/extensions/gsd/auto/workflow-unit-dispatch.js +44 -0
  43. package/dist/resources/extensions/gsd/auto/workflow-worker-heartbeat.js +15 -0
  44. package/dist/resources/extensions/gsd/auto-dashboard.js +3 -0
  45. package/dist/resources/extensions/gsd/auto-prompts.js +168 -3
  46. package/dist/resources/extensions/gsd/auto-recovery.js +45 -52
  47. package/dist/resources/extensions/gsd/auto-runtime-state.js +4 -0
  48. package/dist/resources/extensions/gsd/auto-verification.js +2 -11
  49. package/dist/resources/extensions/gsd/auto-worktree.js +85 -38
  50. package/dist/resources/extensions/gsd/auto.js +162 -5
  51. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +9 -1
  52. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +10 -0
  53. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +11 -5
  54. package/dist/resources/extensions/gsd/commands-ship.js +23 -46
  55. package/dist/resources/extensions/gsd/commands-workflow-templates.js +12 -7
  56. package/dist/resources/extensions/gsd/component-loader.js +5 -11
  57. package/dist/resources/extensions/gsd/custom-workflow-engine.js +25 -1
  58. package/dist/resources/extensions/gsd/db-adapter.js +47 -0
  59. package/dist/resources/extensions/gsd/db-base-schema.js +337 -0
  60. package/dist/resources/extensions/gsd/db-connection-cache.js +31 -0
  61. package/dist/resources/extensions/gsd/db-coordination-schema.js +104 -0
  62. package/dist/resources/extensions/gsd/db-decision-requirement-rows.js +71 -0
  63. package/dist/resources/extensions/gsd/db-gate-rows.js +16 -0
  64. package/dist/resources/extensions/gsd/db-lightweight-query-rows.js +29 -0
  65. package/dist/resources/extensions/gsd/db-memory-fts-schema.js +56 -0
  66. package/dist/resources/extensions/gsd/db-migration-backup.js +22 -0
  67. package/dist/resources/extensions/gsd/db-migration-steps.js +394 -0
  68. package/dist/resources/extensions/gsd/db-milestone-artifact-rows.js +35 -0
  69. package/dist/resources/extensions/gsd/db-open-state.js +32 -0
  70. package/dist/resources/extensions/gsd/db-provider.js +108 -0
  71. package/dist/resources/extensions/gsd/db-runtime-kv-schema.js +27 -0
  72. package/dist/resources/extensions/gsd/db-schema-metadata.js +23 -0
  73. package/dist/resources/extensions/gsd/db-task-slice-rows.js +86 -0
  74. package/dist/resources/extensions/gsd/db-transaction.js +63 -0
  75. package/dist/resources/extensions/gsd/db-verification-evidence-rows.js +3 -0
  76. package/dist/resources/extensions/gsd/db-verification-evidence-schema.js +19 -0
  77. package/dist/resources/extensions/gsd/escalation.js +2 -0
  78. package/dist/resources/extensions/gsd/graph.js +9 -3
  79. package/dist/resources/extensions/gsd/gsd-db.js +215 -1519
  80. package/dist/resources/extensions/gsd/legacy-telemetry.js +70 -0
  81. package/dist/resources/extensions/gsd/markdown-renderer.js +2 -0
  82. package/dist/resources/extensions/gsd/model-router.js +9 -6
  83. package/dist/resources/extensions/gsd/notification-widget.js +21 -3
  84. package/dist/resources/extensions/gsd/post-execution-checks.js +27 -6
  85. package/dist/resources/extensions/gsd/pr-evidence.js +76 -0
  86. package/dist/resources/extensions/gsd/pre-execution-checks.js +2 -0
  87. package/dist/resources/extensions/gsd/process-task-path.js +61 -0
  88. package/dist/resources/extensions/gsd/prompt-loader.js +9 -5
  89. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +32 -30
  90. package/dist/resources/extensions/gsd/prompts/complete-slice.md +23 -34
  91. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +50 -96
  92. package/dist/resources/extensions/gsd/prompts/discuss.md +81 -181
  93. package/dist/resources/extensions/gsd/prompts/execute-task.md +40 -67
  94. package/dist/resources/extensions/gsd/prompts/forensics.md +41 -84
  95. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +29 -39
  96. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +30 -65
  97. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +25 -52
  98. package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +36 -36
  99. package/dist/resources/extensions/gsd/prompts/guided-research-project.md +20 -38
  100. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +45 -59
  101. package/dist/resources/extensions/gsd/prompts/plan-slice.md +25 -87
  102. package/dist/resources/extensions/gsd/prompts/queue.md +46 -53
  103. package/dist/resources/extensions/gsd/prompts/refine-slice.md +23 -23
  104. package/dist/resources/extensions/gsd/prompts/research-slice.md +23 -23
  105. package/dist/resources/extensions/gsd/prompts/rethink.md +10 -10
  106. package/dist/resources/extensions/gsd/prompts/system.md +65 -107
  107. package/dist/resources/extensions/gsd/prompts/triage-captures.md +15 -15
  108. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +24 -24
  109. package/dist/resources/extensions/gsd/prompts/worktree-merge.md +35 -35
  110. package/dist/resources/extensions/gsd/state.js +4 -0
  111. package/dist/resources/extensions/gsd/tools/complete-milestone.js +14 -9
  112. package/dist/resources/extensions/gsd/tools/complete-task.js +2 -0
  113. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +6 -1
  114. package/dist/resources/extensions/gsd/unit-context-composer.js +1 -1
  115. package/dist/resources/extensions/gsd/uok/kernel.js +8 -3
  116. package/dist/resources/extensions/gsd/uok/plan-v2.js +2 -0
  117. package/dist/resources/extensions/gsd/workflow-logger.js +13 -13
  118. package/dist/resources/extensions/gsd/workflow-manifest.js +2 -0
  119. package/dist/resources/extensions/gsd/workflow-projections.js +2 -0
  120. package/dist/resources/extensions/gsd/workflow-templates.js +9 -0
  121. package/dist/resources/extensions/shared/interview-ui.js +15 -4
  122. package/dist/{security-overrides.js → security/security-overrides.js} +1 -1
  123. package/dist/{startup-model-validation.js → startup/startup-model-validation.js} +1 -1
  124. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  125. package/dist/{update-check.js → update/update-check.js} +1 -1
  126. package/dist/web/standalone/.next/BUILD_ID +1 -1
  127. package/dist/web/standalone/.next/app-path-routes-manifest.json +16 -16
  128. package/dist/web/standalone/.next/build-manifest.json +2 -2
  129. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  130. package/dist/web/standalone/.next/required-server-files.json +1 -1
  131. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  132. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  134. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  136. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  138. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  140. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  141. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  142. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  143. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  144. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  145. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  146. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  147. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  151. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  153. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  154. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  155. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  157. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  159. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  161. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  165. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  167. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  169. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  171. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  172. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  173. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  177. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  179. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  180. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  185. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  186. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  188. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  190. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  192. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  193. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  194. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  196. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  198. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  200. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  201. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  202. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  203. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  204. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  205. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  206. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  207. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  208. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +3 -3
  209. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  210. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  211. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  212. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  213. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  214. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  215. package/dist/web/standalone/.next/server/app/index.html +1 -1
  216. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  217. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  218. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  219. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  220. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  221. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  222. package/dist/web/standalone/.next/server/app-paths-manifest.json +16 -16
  223. package/dist/web/standalone/.next/server/chunks/167.js +2 -0
  224. package/dist/web/standalone/.next/server/chunks/7020.js +12 -0
  225. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  226. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  227. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  228. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  229. package/dist/web/standalone/package.json +1 -0
  230. package/dist/web/standalone/server.js +1 -1
  231. package/dist/{worktree-cli.js → worktrees/worktree-cli.js} +3 -2
  232. package/package.json +12 -5
  233. package/packages/contracts/dist/index.d.ts +3 -0
  234. package/packages/contracts/dist/index.d.ts.map +1 -0
  235. package/packages/contracts/dist/index.js +5 -0
  236. package/packages/contracts/dist/index.js.map +1 -0
  237. package/packages/contracts/dist/rpc.d.ts +549 -0
  238. package/packages/contracts/dist/rpc.d.ts.map +1 -0
  239. package/packages/contracts/dist/rpc.js +53 -0
  240. package/packages/contracts/dist/rpc.js.map +1 -0
  241. package/packages/contracts/dist/rpc.test.d.ts +2 -0
  242. package/packages/contracts/dist/rpc.test.d.ts.map +1 -0
  243. package/packages/contracts/dist/rpc.test.js +47 -0
  244. package/packages/contracts/dist/rpc.test.js.map +1 -0
  245. package/packages/contracts/dist/workflow.d.ts +180 -0
  246. package/packages/contracts/dist/workflow.d.ts.map +1 -0
  247. package/packages/contracts/dist/workflow.js +201 -0
  248. package/packages/contracts/dist/workflow.js.map +1 -0
  249. package/packages/contracts/package.json +39 -0
  250. package/packages/contracts/src/index.ts +5 -0
  251. package/packages/contracts/src/rpc.test.ts +72 -0
  252. package/packages/contracts/src/rpc.ts +286 -0
  253. package/packages/contracts/src/workflow.ts +213 -0
  254. package/packages/contracts/tsconfig.json +25 -0
  255. package/packages/daemon/package.json +3 -2
  256. package/packages/daemon/src/event-bridge.test.ts +2 -1
  257. package/packages/daemon/src/event-bridge.ts +1 -1
  258. package/packages/daemon/src/event-formatter.test.ts +1 -2
  259. package/packages/daemon/src/event-formatter.ts +1 -2
  260. package/packages/daemon/src/session-manager.ts +2 -2
  261. package/packages/daemon/src/types.ts +3 -18
  262. package/packages/mcp-server/dist/server.d.ts +13 -0
  263. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  264. package/packages/mcp-server/dist/server.js +77 -0
  265. package/packages/mcp-server/dist/server.js.map +1 -1
  266. package/packages/mcp-server/dist/session-manager.js +1 -1
  267. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  268. package/packages/mcp-server/dist/types.d.ts +3 -11
  269. package/packages/mcp-server/dist/types.d.ts.map +1 -1
  270. package/packages/mcp-server/dist/types.js.map +1 -1
  271. package/packages/mcp-server/dist/workflow-tools.d.ts +1 -1
  272. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  273. package/packages/mcp-server/dist/workflow-tools.js +2 -40
  274. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  275. package/packages/mcp-server/package.json +3 -2
  276. package/packages/mcp-server/src/mcp-server.test.ts +138 -0
  277. package/packages/mcp-server/src/server.ts +99 -1
  278. package/packages/mcp-server/src/session-manager.ts +2 -2
  279. package/packages/mcp-server/src/types.ts +7 -18
  280. package/packages/mcp-server/src/workflow-tools.ts +2 -40
  281. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  282. package/packages/native/package.json +1 -1
  283. package/packages/pi-agent-core/package.json +1 -1
  284. package/packages/pi-ai/package.json +1 -1
  285. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +14 -0
  286. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js +20 -0
  288. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js.map +1 -1
  289. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  290. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +6 -1
  291. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  292. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +9 -1
  293. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  294. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +31 -10
  295. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  296. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +1 -512
  297. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
  298. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js +3 -7
  299. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
  300. package/packages/pi-coding-agent/package.json +2 -1
  301. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +23 -2
  302. package/packages/pi-coding-agent/src/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.ts +31 -0
  303. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +6 -1
  304. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +50 -9
  305. package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +3 -336
  306. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  307. package/packages/pi-tui/dist/__tests__/tui.test.js +25 -3
  308. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  309. package/packages/pi-tui/dist/tui.d.ts +3 -1
  310. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  311. package/packages/pi-tui/dist/tui.js +3 -1
  312. package/packages/pi-tui/dist/tui.js.map +1 -1
  313. package/packages/pi-tui/package.json +1 -1
  314. package/packages/pi-tui/src/__tests__/tui.test.ts +30 -3
  315. package/packages/pi-tui/src/tui.ts +3 -1
  316. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  317. package/packages/rpc-client/README.md +3 -3
  318. package/packages/rpc-client/dist/index.d.ts +1 -1
  319. package/packages/rpc-client/dist/index.d.ts.map +1 -1
  320. package/packages/rpc-client/dist/index.js.map +1 -1
  321. package/packages/rpc-client/dist/rpc-client.d.ts +2 -6
  322. package/packages/rpc-client/dist/rpc-client.d.ts.map +1 -1
  323. package/packages/rpc-client/dist/rpc-client.js.map +1 -1
  324. package/packages/rpc-client/dist/rpc-types.d.ts +1 -565
  325. package/packages/rpc-client/dist/rpc-types.d.ts.map +1 -1
  326. package/packages/rpc-client/dist/rpc-types.js +3 -11
  327. package/packages/rpc-client/dist/rpc-types.js.map +1 -1
  328. package/packages/rpc-client/package.json +4 -1
  329. package/packages/rpc-client/src/index.ts +1 -1
  330. package/packages/rpc-client/src/rpc-client.ts +3 -6
  331. package/packages/rpc-client/src/rpc-types.ts +3 -398
  332. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  333. package/pkg/package.json +1 -1
  334. package/src/resources/extensions/github-sync/templates.ts +59 -84
  335. package/src/resources/extensions/github-sync/tests/templates.test.ts +10 -2
  336. package/src/resources/extensions/gsd/auto/contracts.ts +87 -0
  337. package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +72 -0
  338. package/src/resources/extensions/gsd/auto/loop-deps.ts +10 -3
  339. package/src/resources/extensions/gsd/auto/loop.ts +414 -598
  340. package/src/resources/extensions/gsd/auto/orchestrator.ts +161 -0
  341. package/src/resources/extensions/gsd/auto/phases.ts +82 -8
  342. package/src/resources/extensions/gsd/auto/session.ts +11 -0
  343. package/src/resources/extensions/gsd/auto/workflow-custom-engine-dispatch-outcome.ts +28 -0
  344. package/src/resources/extensions/gsd/auto/workflow-custom-engine-iteration.ts +52 -0
  345. package/src/resources/extensions/gsd/auto/workflow-custom-engine-reconcile-outcome.ts +58 -0
  346. package/src/resources/extensions/gsd/auto/workflow-custom-engine-reconcile.ts +71 -0
  347. package/src/resources/extensions/gsd/auto/workflow-custom-engine-retry.ts +90 -0
  348. package/src/resources/extensions/gsd/auto/workflow-custom-engine-verify-outcome.ts +50 -0
  349. package/src/resources/extensions/gsd/auto/workflow-dispatch-claim.ts +97 -0
  350. package/src/resources/extensions/gsd/auto/workflow-dispatch-ledger.ts +45 -0
  351. package/src/resources/extensions/gsd/auto/workflow-iteration-completion.ts +26 -0
  352. package/src/resources/extensions/gsd/auto/workflow-journal-reporter.ts +33 -0
  353. package/src/resources/extensions/gsd/auto/workflow-kernel.ts +520 -0
  354. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +58 -0
  355. package/src/resources/extensions/gsd/auto/workflow-phase-reporter.ts +22 -0
  356. package/src/resources/extensions/gsd/auto/workflow-session-lock.ts +68 -0
  357. package/src/resources/extensions/gsd/auto/workflow-sidecar-iteration.ts +46 -0
  358. package/src/resources/extensions/gsd/auto/workflow-sidecar-queue.ts +46 -0
  359. package/src/resources/extensions/gsd/auto/workflow-turn-reporter.ts +68 -0
  360. package/src/resources/extensions/gsd/auto/workflow-unit-dispatch.ts +89 -0
  361. package/src/resources/extensions/gsd/auto/workflow-worker-heartbeat.ts +38 -0
  362. package/src/resources/extensions/gsd/auto-dashboard.ts +4 -0
  363. package/src/resources/extensions/gsd/auto-prompts.ts +170 -3
  364. package/src/resources/extensions/gsd/auto-recovery.ts +42 -50
  365. package/src/resources/extensions/gsd/auto-runtime-state.ts +7 -0
  366. package/src/resources/extensions/gsd/auto-verification.ts +5 -1
  367. package/src/resources/extensions/gsd/auto-worktree.ts +83 -36
  368. package/src/resources/extensions/gsd/auto.ts +170 -4
  369. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +14 -1
  370. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +11 -0
  371. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +13 -5
  372. package/src/resources/extensions/gsd/commands-ship.ts +24 -51
  373. package/src/resources/extensions/gsd/commands-workflow-templates.ts +13 -0
  374. package/src/resources/extensions/gsd/component-loader.ts +5 -11
  375. package/src/resources/extensions/gsd/custom-workflow-engine.ts +29 -0
  376. package/src/resources/extensions/gsd/db-adapter.ts +75 -0
  377. package/src/resources/extensions/gsd/db-base-schema.ts +368 -0
  378. package/src/resources/extensions/gsd/db-connection-cache.ts +45 -0
  379. package/src/resources/extensions/gsd/db-coordination-schema.ts +109 -0
  380. package/src/resources/extensions/gsd/db-decision-requirement-rows.ts +77 -0
  381. package/src/resources/extensions/gsd/db-gate-rows.ts +19 -0
  382. package/src/resources/extensions/gsd/db-lightweight-query-rows.ts +50 -0
  383. package/src/resources/extensions/gsd/db-memory-fts-schema.ts +66 -0
  384. package/src/resources/extensions/gsd/db-migration-backup.ts +34 -0
  385. package/src/resources/extensions/gsd/db-migration-steps.ts +434 -0
  386. package/src/resources/extensions/gsd/db-milestone-artifact-rows.ts +70 -0
  387. package/src/resources/extensions/gsd/db-open-state.ts +47 -0
  388. package/src/resources/extensions/gsd/db-provider.ts +148 -0
  389. package/src/resources/extensions/gsd/db-runtime-kv-schema.ts +30 -0
  390. package/src/resources/extensions/gsd/db-schema-metadata.ts +33 -0
  391. package/src/resources/extensions/gsd/db-task-slice-rows.ts +146 -0
  392. package/src/resources/extensions/gsd/db-transaction.ts +76 -0
  393. package/src/resources/extensions/gsd/db-verification-evidence-rows.ts +14 -0
  394. package/src/resources/extensions/gsd/db-verification-evidence-schema.ts +22 -0
  395. package/src/resources/extensions/gsd/escalation.ts +3 -1
  396. package/src/resources/extensions/gsd/graph.ts +12 -5
  397. package/src/resources/extensions/gsd/gsd-db.ts +260 -1659
  398. package/src/resources/extensions/gsd/interrupted-session.ts +1 -0
  399. package/src/resources/extensions/gsd/legacy-telemetry.ts +99 -0
  400. package/src/resources/extensions/gsd/markdown-renderer.ts +4 -1
  401. package/src/resources/extensions/gsd/model-router.ts +10 -6
  402. package/src/resources/extensions/gsd/notification-widget.ts +25 -4
  403. package/src/resources/extensions/gsd/post-execution-checks.ts +35 -7
  404. package/src/resources/extensions/gsd/pr-evidence.ts +124 -0
  405. package/src/resources/extensions/gsd/pre-execution-checks.ts +4 -1
  406. package/src/resources/extensions/gsd/process-task-path.ts +81 -0
  407. package/src/resources/extensions/gsd/prompt-loader.ts +9 -5
  408. package/src/resources/extensions/gsd/prompts/complete-milestone.md +32 -30
  409. package/src/resources/extensions/gsd/prompts/complete-slice.md +23 -34
  410. package/src/resources/extensions/gsd/prompts/discuss-headless.md +50 -96
  411. package/src/resources/extensions/gsd/prompts/discuss.md +81 -181
  412. package/src/resources/extensions/gsd/prompts/execute-task.md +40 -67
  413. package/src/resources/extensions/gsd/prompts/forensics.md +41 -84
  414. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +29 -39
  415. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +30 -65
  416. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +25 -52
  417. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +36 -36
  418. package/src/resources/extensions/gsd/prompts/guided-research-project.md +20 -38
  419. package/src/resources/extensions/gsd/prompts/plan-milestone.md +45 -59
  420. package/src/resources/extensions/gsd/prompts/plan-slice.md +25 -87
  421. package/src/resources/extensions/gsd/prompts/queue.md +46 -53
  422. package/src/resources/extensions/gsd/prompts/refine-slice.md +23 -23
  423. package/src/resources/extensions/gsd/prompts/research-slice.md +23 -23
  424. package/src/resources/extensions/gsd/prompts/rethink.md +10 -10
  425. package/src/resources/extensions/gsd/prompts/system.md +65 -107
  426. package/src/resources/extensions/gsd/prompts/triage-captures.md +15 -15
  427. package/src/resources/extensions/gsd/prompts/validate-milestone.md +24 -24
  428. package/src/resources/extensions/gsd/prompts/worktree-merge.md +35 -35
  429. package/src/resources/extensions/gsd/state.ts +6 -3
  430. package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +32 -0
  431. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +65 -0
  432. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +38 -0
  433. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +353 -0
  434. package/src/resources/extensions/gsd/tests/auto-pr-bugs.test.ts +19 -0
  435. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +39 -0
  436. package/src/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +3 -0
  437. package/src/resources/extensions/gsd/tests/commands-eval-review.test.ts +2 -2
  438. package/src/resources/extensions/gsd/tests/commands-ship-eval-warn.test.ts +3 -2
  439. package/src/resources/extensions/gsd/tests/complete-milestone-prompt-rendering.test.ts +47 -0
  440. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +19 -5
  441. package/src/resources/extensions/gsd/tests/component-loader.test.ts +2 -9
  442. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +144 -0
  443. package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +139 -0
  444. package/src/resources/extensions/gsd/tests/custom-workflow-engine.test.ts +50 -0
  445. package/src/resources/extensions/gsd/tests/dashboard-custom-engine.test.ts +3 -3
  446. package/src/resources/extensions/gsd/tests/db-adapter.test.ts +82 -0
  447. package/src/resources/extensions/gsd/tests/db-base-schema.test.ts +62 -0
  448. package/src/resources/extensions/gsd/tests/db-connection-cache.test.ts +60 -0
  449. package/src/resources/extensions/gsd/tests/db-coordination-schema.test.ts +39 -0
  450. package/src/resources/extensions/gsd/tests/db-decision-requirement-rows.test.ts +135 -0
  451. package/src/resources/extensions/gsd/tests/db-gate-rows.test.ts +53 -0
  452. package/src/resources/extensions/gsd/tests/db-lightweight-query-rows.test.ts +45 -0
  453. package/src/resources/extensions/gsd/tests/db-memory-fts-schema.test.ts +86 -0
  454. package/src/resources/extensions/gsd/tests/db-migration-backup.test.ts +105 -0
  455. package/src/resources/extensions/gsd/tests/db-migration-steps.test.ts +159 -0
  456. package/src/resources/extensions/gsd/tests/db-milestone-artifact-rows.test.ts +53 -0
  457. package/src/resources/extensions/gsd/tests/db-open-state.test.ts +56 -0
  458. package/src/resources/extensions/gsd/tests/db-provider.test.ts +105 -0
  459. package/src/resources/extensions/gsd/tests/db-runtime-kv-schema.test.ts +37 -0
  460. package/src/resources/extensions/gsd/tests/db-schema-metadata.test.ts +115 -0
  461. package/src/resources/extensions/gsd/tests/db-task-slice-rows.test.ts +128 -0
  462. package/src/resources/extensions/gsd/tests/db-transaction.test.ts +110 -0
  463. package/src/resources/extensions/gsd/tests/db-verification-evidence-schema.test.ts +76 -0
  464. package/src/resources/extensions/gsd/tests/discuss-headless-rendering.test.ts +37 -0
  465. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +59 -0
  466. package/src/resources/extensions/gsd/tests/forensics-prompt-rendering.test.ts +36 -0
  467. package/src/resources/extensions/gsd/tests/graph-operations.test.ts +10 -0
  468. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +95 -0
  469. package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +43 -0
  470. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +41 -0
  471. package/src/resources/extensions/gsd/tests/guided-discuss-requirements-prompt-rendering.test.ts +45 -0
  472. package/src/resources/extensions/gsd/tests/headless-answers.test.ts +1 -1
  473. package/src/resources/extensions/gsd/tests/headless-query.test.ts +2 -2
  474. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +79 -0
  475. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +37 -0
  476. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +5 -3
  477. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +134 -0
  478. package/src/resources/extensions/gsd/tests/legacy-component-format-telemetry.test.ts +62 -0
  479. package/src/resources/extensions/gsd/tests/legacy-telemetry.test.ts +144 -0
  480. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +40 -16
  481. package/src/resources/extensions/gsd/tests/model-router.test.ts +33 -12
  482. package/src/resources/extensions/gsd/tests/notification-store.test.ts +8 -0
  483. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +40 -1
  484. package/src/resources/extensions/gsd/tests/paused-session-via-db.test.ts +2 -0
  485. package/src/resources/extensions/gsd/tests/plan-milestone-rendering.test.ts +45 -0
  486. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +65 -16
  487. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +27 -0
  488. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +46 -0
  489. package/src/resources/extensions/gsd/tests/pr-evidence.test.ts +79 -0
  490. package/src/resources/extensions/gsd/tests/process-task-path.test.ts +51 -0
  491. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +16 -1
  492. package/src/resources/extensions/gsd/tests/queue-prompt-rendering.test.ts +37 -0
  493. package/src/resources/extensions/gsd/tests/resource-loader-import-path.test.ts +1 -1
  494. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +32 -9
  495. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +6 -6
  496. package/src/resources/extensions/gsd/tests/uok-kernel-path.test.ts +12 -0
  497. package/src/resources/extensions/gsd/tests/workflow-custom-engine-dispatch-outcome.test.ts +55 -0
  498. package/src/resources/extensions/gsd/tests/workflow-custom-engine-iteration.test.ts +93 -0
  499. package/src/resources/extensions/gsd/tests/workflow-custom-engine-reconcile-outcome.test.ts +108 -0
  500. package/src/resources/extensions/gsd/tests/workflow-custom-engine-reconcile.test.ts +146 -0
  501. package/src/resources/extensions/gsd/tests/workflow-custom-engine-retry.test.ts +136 -0
  502. package/src/resources/extensions/gsd/tests/workflow-custom-engine-verify-outcome.test.ts +95 -0
  503. package/src/resources/extensions/gsd/tests/workflow-dispatch-claim.test.ts +158 -0
  504. package/src/resources/extensions/gsd/tests/workflow-dispatch-ledger.test.ts +82 -0
  505. package/src/resources/extensions/gsd/tests/workflow-iteration-completion.test.ts +44 -0
  506. package/src/resources/extensions/gsd/tests/workflow-journal-reporter.test.ts +49 -0
  507. package/src/resources/extensions/gsd/tests/workflow-kernel.test.ts +607 -0
  508. package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +20 -4
  509. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +8 -7
  510. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +71 -0
  511. package/src/resources/extensions/gsd/tests/workflow-phase-reporter.test.ts +40 -0
  512. package/src/resources/extensions/gsd/tests/workflow-session-lock.test.ts +135 -0
  513. package/src/resources/extensions/gsd/tests/workflow-sidecar-iteration.test.ts +110 -0
  514. package/src/resources/extensions/gsd/tests/workflow-sidecar-queue.test.ts +116 -0
  515. package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +21 -0
  516. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +32 -0
  517. package/src/resources/extensions/gsd/tests/workflow-turn-reporter.test.ts +87 -0
  518. package/src/resources/extensions/gsd/tests/workflow-unit-dispatch.test.ts +160 -0
  519. package/src/resources/extensions/gsd/tests/workflow-worker-heartbeat.test.ts +123 -0
  520. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +18 -34
  521. package/src/resources/extensions/gsd/tools/complete-milestone.ts +15 -9
  522. package/src/resources/extensions/gsd/tools/complete-task.ts +4 -1
  523. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +6 -1
  524. package/src/resources/extensions/gsd/unit-context-composer.ts +1 -1
  525. package/src/resources/extensions/gsd/uok/kernel.ts +10 -3
  526. package/src/resources/extensions/gsd/uok/plan-v2.ts +5 -1
  527. package/src/resources/extensions/gsd/workflow-logger.ts +13 -13
  528. package/src/resources/extensions/gsd/workflow-manifest.ts +6 -15
  529. package/src/resources/extensions/gsd/workflow-projections.ts +5 -1
  530. package/src/resources/extensions/gsd/workflow-templates.ts +11 -0
  531. package/src/resources/extensions/shared/interview-ui.ts +18 -5
  532. package/src/resources/extensions/shared/tests/interview-notes-loop.test.ts +41 -0
  533. package/dist/web/standalone/.next/server/chunks/6336.js +0 -1
  534. package/dist/web/standalone/.next/server/chunks/6897.js +0 -12
  535. /package/dist/{app-paths.d.ts → app/app-paths.d.ts} +0 -0
  536. /package/dist/{app-paths.js → app/app-paths.js} +0 -0
  537. /package/dist/{project-sessions.d.ts → app/project-sessions.d.ts} +0 -0
  538. /package/dist/{project-sessions.js → app/project-sessions.js} +0 -0
  539. /package/dist/{tool-bootstrap.d.ts → app/tool-bootstrap.d.ts} +0 -0
  540. /package/dist/{tool-bootstrap.js → app/tool-bootstrap.js} +0 -0
  541. /package/dist/{cli-auto-routing.d.ts → cli/cli-auto-routing.d.ts} +0 -0
  542. /package/dist/{cli-auto-routing.js → cli/cli-auto-routing.js} +0 -0
  543. /package/dist/{cli-policy.d.ts → cli/cli-policy.d.ts} +0 -0
  544. /package/dist/{cli-policy.js → cli/cli-policy.js} +0 -0
  545. /package/dist/{cli.d.ts → cli/cli.d.ts} +0 -0
  546. /package/dist/{help-text.d.ts → cli/help-text.d.ts} +0 -0
  547. /package/dist/{help-text.js → cli/help-text.js} +0 -0
  548. /package/dist/{logo.d.ts → cli/logo.d.ts} +0 -0
  549. /package/dist/{logo.js → cli/logo.js} +0 -0
  550. /package/dist/{web-mode.d.ts → cli/web-mode.d.ts} +0 -0
  551. /package/dist/{bundled-extension-paths.d.ts → extension-runtime/bundled-extension-paths.d.ts} +0 -0
  552. /package/dist/{bundled-extension-paths.js → extension-runtime/bundled-extension-paths.js} +0 -0
  553. /package/dist/{bundled-resource-path.d.ts → extension-runtime/bundled-resource-path.d.ts} +0 -0
  554. /package/dist/{bundled-resource-path.js → extension-runtime/bundled-resource-path.js} +0 -0
  555. /package/dist/{extension-discovery.d.ts → extension-runtime/extension-discovery.d.ts} +0 -0
  556. /package/dist/{extension-discovery.js → extension-runtime/extension-discovery.js} +0 -0
  557. /package/dist/{extension-registry.d.ts → extension-runtime/extension-registry.d.ts} +0 -0
  558. /package/dist/{extension-sort.d.ts → extension-runtime/extension-sort.d.ts} +0 -0
  559. /package/dist/{extension-sort.js → extension-runtime/extension-sort.js} +0 -0
  560. /package/dist/{extension-validator.d.ts → extension-runtime/extension-validator.d.ts} +0 -0
  561. /package/dist/{extension-validator.js → extension-runtime/extension-validator.js} +0 -0
  562. /package/dist/{headless-answers.d.ts → headless/headless-answers.d.ts} +0 -0
  563. /package/dist/{headless-answers.js → headless/headless-answers.js} +0 -0
  564. /package/dist/{headless-context.d.ts → headless/headless-context.d.ts} +0 -0
  565. /package/dist/{headless-context.js → headless/headless-context.js} +0 -0
  566. /package/dist/{headless-events.d.ts → headless/headless-events.d.ts} +0 -0
  567. /package/dist/{headless-events.js → headless/headless-events.js} +0 -0
  568. /package/dist/{headless-recover.d.ts → headless/headless-recover.d.ts} +0 -0
  569. /package/dist/{headless-types.d.ts → headless/headless-types.d.ts} +0 -0
  570. /package/dist/{headless-types.js → headless/headless-types.js} +0 -0
  571. /package/dist/{headless-ui.d.ts → headless/headless-ui.d.ts} +0 -0
  572. /package/dist/{headless-ui.js → headless/headless-ui.js} +0 -0
  573. /package/dist/{headless.d.ts → headless/headless.d.ts} +0 -0
  574. /package/dist/{mcp-server.d.ts → mcp/mcp-server.d.ts} +0 -0
  575. /package/dist/{mcp-server.js → mcp/mcp-server.js} +0 -0
  576. /package/dist/{onboarding.d.ts → onboarding/onboarding.d.ts} +0 -0
  577. /package/dist/{welcome-screen.d.ts → onboarding/welcome-screen.d.ts} +0 -0
  578. /package/dist/{wizard.d.ts → onboarding/wizard.d.ts} +0 -0
  579. /package/dist/{wizard.js → onboarding/wizard.js} +0 -0
  580. /package/dist/{models-resolver.d.ts → providers/models-resolver.d.ts} +0 -0
  581. /package/dist/{pi-migration.d.ts → providers/pi-migration.d.ts} +0 -0
  582. /package/dist/{pi-migration.js → providers/pi-migration.js} +0 -0
  583. /package/dist/{provider-migrations.d.ts → providers/provider-migrations.d.ts} +0 -0
  584. /package/dist/{provider-migrations.js → providers/provider-migrations.js} +0 -0
  585. /package/dist/{remote-questions-config.d.ts → providers/remote-questions-config.d.ts} +0 -0
  586. /package/dist/{rtk-shared.d.ts → rtk/rtk-shared.d.ts} +0 -0
  587. /package/dist/{rtk-shared.js → rtk/rtk-shared.js} +0 -0
  588. /package/dist/{rtk.d.ts → rtk/rtk.d.ts} +0 -0
  589. /package/dist/{rtk.js → rtk/rtk.js} +0 -0
  590. /package/dist/{security-overrides.d.ts → security/security-overrides.d.ts} +0 -0
  591. /package/dist/{claude-cli-check.d.ts → startup/claude-cli-check.d.ts} +0 -0
  592. /package/dist/{claude-cli-check.js → startup/claude-cli-check.js} +0 -0
  593. /package/dist/{runtime-checks.d.ts → startup/runtime-checks.d.ts} +0 -0
  594. /package/dist/{runtime-checks.js → startup/runtime-checks.js} +0 -0
  595. /package/dist/{startup-model-validation.d.ts → startup/startup-model-validation.d.ts} +0 -0
  596. /package/dist/{startup-timings.d.ts → startup/startup-timings.d.ts} +0 -0
  597. /package/dist/{startup-timings.js → startup/startup-timings.js} +0 -0
  598. /package/dist/{update-check.d.ts → update/update-check.d.ts} +0 -0
  599. /package/dist/{update-cmd.d.ts → update/update-cmd.d.ts} +0 -0
  600. /package/dist/{update-cmd.js → update/update-cmd.js} +0 -0
  601. /package/dist/web/standalone/.next/static/{TzEVJ1Lh8vbez4n4Q9TqQ → -WDXZiiJHGZDbYH75Mu6f}/_buildManifest.js +0 -0
  602. /package/dist/web/standalone/.next/static/{TzEVJ1Lh8vbez4n4Q9TqQ → -WDXZiiJHGZDbYH75Mu6f}/_ssgManifest.js +0 -0
  603. /package/dist/{worktree-cli.d.ts → worktrees/worktree-cli.d.ts} +0 -0
  604. /package/dist/{worktree-name-gen.d.ts → worktrees/worktree-name-gen.d.ts} +0 -0
  605. /package/dist/{worktree-name-gen.js → worktrees/worktree-name-gen.js} +0 -0
  606. /package/dist/{worktree-status-banner.d.ts → worktrees/worktree-status-banner.d.ts} +0 -0
  607. /package/dist/{worktree-status-banner.js → worktrees/worktree-status-banner.js} +0 -0
@@ -9,24 +9,39 @@
9
9
  import { randomUUID } from "node:crypto";
10
10
  import { MAX_LOOP_ITERATIONS, } from "./types.js";
11
11
  import { _clearCurrentResolve } from "./resolve.js";
12
- import { runPreDispatch, runDispatch, runGuards, runUnitPhase, runFinalize, } from "./phases.js";
12
+ import { runPreDispatch, runDispatch, runGuards, runFinalize, } from "./phases.js";
13
13
  import { debugLog } from "../debug-logger.js";
14
14
  import { isInfrastructureError, isTransientCooldownError, getCooldownRetryAfterMs, COOLDOWN_FALLBACK_WAIT_MS, MAX_COOLDOWN_RETRIES } from "./infra-errors.js";
15
15
  import { ModelPolicyDispatchBlockedError } from "../auto-model-selection.js";
16
16
  import { resolveEngine } from "../engine-resolver.js";
17
17
  import { logWarning } from "../workflow-logger.js";
18
- import { gsdRoot } from "../paths.js";
19
- import { heartbeatAutoWorker } from "../db/auto-workers.js";
20
18
  import { recordDispatchClaim, markRunning as markDispatchRunning, markCompleted as markDispatchCompleted, markFailed as markDispatchFailed, getRecentForUnit as getRecentDispatchesForUnit, getRecentUnitKeysForProjectRoot, } from "../db/unit-dispatches.js";
21
19
  import { refreshMilestoneLease } from "../db/milestone-leases.js";
20
+ import { heartbeatAutoWorker } from "../db/auto-workers.js";
22
21
  import { getRuntimeKv, setRuntimeKv } from "../db/runtime-kv.js";
23
- import { atomicWriteSync } from "../atomic-write.js";
24
22
  import { resolveUokFlags } from "../uok/flags.js";
25
23
  import { scheduleSidecarQueue } from "../uok/execution-graph.js";
26
- import { ExecutionGraphScheduler } from "../uok/execution-graph.js";
27
- import { readFileSync, mkdirSync, unlinkSync } from "node:fs";
28
- import { join } from "node:path";
29
24
  import { normalizeRealPath } from "../paths.js";
25
+ import { decideCooldownRecovery, decideDispatchClaim, decideEngineDispatch, decideFinalizeResult, decideInfrastructureError, decideIterationErrorRecovery, decideMemoryPressure, decideModelPolicyBlocked, decideMinRequestInterval, decideWorkflowLoop, formatDispatchExceptionSummary, formatUnhandledDispatchErrorSummary, resolveUnitRequestTimestamp, shouldUseCustomEnginePath, } from "./workflow-kernel.js";
26
+ import { hydrateCustomVerifyRetryCounts, saveCustomVerifyRetryCounts, } from "./custom-verify-retry-store.js";
27
+ import { settleDispatchCompleted, settleDispatchFailed, } from "./workflow-dispatch-ledger.js";
28
+ import { openDispatchClaim } from "./workflow-dispatch-claim.js";
29
+ import { completeWorkflowIteration } from "./workflow-iteration-completion.js";
30
+ import { createWorkflowJournalReporter } from "./workflow-journal-reporter.js";
31
+ import { createWorkflowPhaseReporter } from "./workflow-phase-reporter.js";
32
+ import { createWorkflowTurnReporter } from "./workflow-turn-reporter.js";
33
+ import { validateWorkflowSessionLock } from "./workflow-session-lock.js";
34
+ import { dequeueSidecarItem } from "./workflow-sidecar-queue.js";
35
+ import { maintainWorkerHeartbeat } from "./workflow-worker-heartbeat.js";
36
+ import { measureMemoryPressure } from "./workflow-memory-pressure.js";
37
+ import { buildSidecarIterationData } from "./workflow-sidecar-iteration.js";
38
+ import { createExecutionGraphUnitDispatchDeps, runUnitPhaseViaContract, } from "./workflow-unit-dispatch.js";
39
+ import { handleCustomEngineDispatchOutcome } from "./workflow-custom-engine-dispatch-outcome.js";
40
+ import { buildCustomEngineIterationData } from "./workflow-custom-engine-iteration.js";
41
+ import { handleCustomEngineVerifyRetry } from "./workflow-custom-engine-retry.js";
42
+ import { handleCustomEngineVerifyPause, handleCustomEngineVerifyRetryOutcome, } from "./workflow-custom-engine-verify-outcome.js";
43
+ import { handleCustomEngineReconcile } from "./workflow-custom-engine-reconcile.js";
44
+ import { handleCustomEngineReconcileOutcome } from "./workflow-custom-engine-reconcile-outcome.js";
30
45
  // ── Stuck detection persistence (#3704) ──────────────────────────────────
31
46
  // Phase C migration: stuck-state.json deleted in favor of DB-backed
32
47
  // equivalents. recentUnits is rebuilt from unit_dispatches (Phase B
@@ -70,200 +85,54 @@ function saveStuckState(s, state) {
70
85
  debugLog("autoLoop", { phase: "save-stuck-state-failed", error: err instanceof Error ? err.message : String(err) });
71
86
  }
72
87
  }
73
- // ── Custom workflow verification retry persistence ───────────────────────
74
- // Custom workflows can request verification retries after a step runs. The
75
- // retry budget must survive an auto-mode restart or a failing verifier can
76
- // consume a fresh retry budget every session.
77
- function customVerifyRetryStateDir(s) {
78
- return s.activeRunDir ? join(s.activeRunDir, "runtime") : join(gsdRoot(s.basePath), "runtime");
88
+ function logDispatchLedgerWriteFailure(err) {
89
+ debugLog("autoLoop", {
90
+ phase: "dispatch-ledger-write-failed",
91
+ error: err instanceof Error ? err.message : String(err),
92
+ });
79
93
  }
80
- function customVerifyRetryStatePath(s) {
81
- return join(customVerifyRetryStateDir(s), "custom-verify-retries.json");
94
+ function logDispatchClaimRejected(details) {
95
+ debugLog("autoLoop", {
96
+ phase: "dispatch-claim-rejected",
97
+ ...details,
98
+ });
82
99
  }
83
- function hydrateCustomVerifyRetryCounts(s) {
84
- if (s.verificationRetryCount.size > 0) {
85
- return s.verificationRetryCount;
86
- }
87
- try {
88
- const raw = JSON.parse(readFileSync(customVerifyRetryStatePath(s), "utf-8"));
89
- const counts = raw && typeof raw === "object" && raw.counts && typeof raw.counts === "object"
90
- ? raw.counts
91
- : {};
92
- for (const [key, value] of Object.entries(counts)) {
93
- if (typeof value === "number" && Number.isFinite(value) && value > 0) {
94
- s.verificationRetryCount.set(key, Math.floor(value));
95
- }
96
- }
97
- }
98
- catch (err) {
99
- debugLog("autoLoop", { phase: "load-custom-verify-retries-failed", error: err instanceof Error ? err.message : String(err) });
100
- }
101
- return s.verificationRetryCount;
100
+ function logDispatchClaimFailed(err) {
101
+ debugLog("autoLoop", {
102
+ phase: "dispatch-claim-failed",
103
+ error: err instanceof Error ? err.message : String(err),
104
+ });
102
105
  }
103
- function saveCustomVerifyRetryCounts(s) {
104
- const retryCounts = s.verificationRetryCount;
105
- const filePath = customVerifyRetryStatePath(s);
106
- try {
107
- if (!retryCounts || retryCounts.size === 0) {
108
- unlinkSync(filePath);
109
- return;
110
- }
111
- mkdirSync(customVerifyRetryStateDir(s), { recursive: true });
112
- atomicWriteSync(filePath, JSON.stringify({
113
- counts: Object.fromEntries(retryCounts),
114
- updatedAt: new Date().toISOString(),
115
- }) + "\n");
116
- }
117
- catch (err) {
118
- const code = err && typeof err === "object" && "code" in err ? err.code : undefined;
119
- if (code !== "ENOENT") {
120
- debugLog("autoLoop", { phase: "save-custom-verify-retries-failed", error: err instanceof Error ? err.message : String(err) });
121
- }
122
- }
106
+ function logCustomVerifyRetryLoadFailure(err) {
107
+ debugLog("autoLoop", {
108
+ phase: "load-custom-verify-retries-failed",
109
+ error: err instanceof Error ? err.message : String(err),
110
+ });
123
111
  }
124
- function openDispatchClaim(s, flowId, turnId, iterData) {
125
- if (!s.workerId || s.milestoneLeaseToken === null)
126
- return { kind: "degraded" };
127
- const mid = iterData.mid;
128
- if (!mid)
129
- return { kind: "degraded" };
130
- const recent = getRecentDispatchesForUnit(iterData.unitId, 1);
131
- const attemptN = (recent[0]?.attempt_n ?? 0) + 1;
132
- let claim;
133
- try {
134
- claim = recordDispatchClaim({
135
- traceId: flowId,
136
- turnId,
137
- workerId: s.workerId,
138
- milestoneLeaseToken: s.milestoneLeaseToken,
139
- milestoneId: mid,
140
- sliceId: iterData.state.activeSlice?.id ?? null,
141
- taskId: iterData.state.activeTask?.id ?? null,
142
- unitType: iterData.unitType,
143
- unitId: iterData.unitId,
144
- attemptN,
145
- });
146
- if (!claim.ok) {
147
- debugLog("autoLoop", {
148
- phase: "dispatch-claim-rejected",
149
- unitId: iterData.unitId,
150
- reason: claim.error,
151
- existingId: "existingId" in claim ? claim.existingId : undefined,
152
- existingWorker: "existingWorker" in claim ? claim.existingWorker : undefined,
153
- });
154
- if (claim.error === "already_active") {
155
- return {
156
- kind: "skip",
157
- reason: "already-active",
158
- existingId: claim.existingId,
159
- existingWorker: claim.existingWorker,
160
- };
161
- }
162
- return { kind: "skip", reason: "stale-lease" };
163
- }
164
- markDispatchRunning(claim.dispatchId);
165
- return { kind: "opened", dispatchId: claim.dispatchId };
166
- }
167
- catch (err) {
168
- debugLog("autoLoop", {
169
- phase: "dispatch-claim-failed",
170
- error: err instanceof Error ? err.message : String(err),
171
- });
172
- return { kind: "degraded" };
173
- }
112
+ function logCustomVerifyRetrySaveFailure(err) {
113
+ debugLog("autoLoop", {
114
+ phase: "save-custom-verify-retries-failed",
115
+ error: err instanceof Error ? err.message : String(err),
116
+ });
174
117
  }
175
118
  // ── Memory pressure monitoring (#3331) ──────────────────────────────────
176
119
  // Check heap usage every N iterations and trigger graceful shutdown before
177
120
  // the OS OOM killer sends SIGKILL. The threshold is 90% of the V8 heap
178
121
  // limit (--max-old-space-size or default ~1.5-4GB depending on platform).
179
122
  const MEMORY_CHECK_INTERVAL = 5; // check every 5 iterations
180
- const MEMORY_PRESSURE_THRESHOLD = 0.85; // 85% of heap limit
181
123
  const MAX_CUSTOM_ENGINE_VERIFY_RETRIES = 3;
182
- function checkMemoryPressure() {
183
- const mem = process.memoryUsage();
184
- // v8.getHeapStatistics() gives heap_size_limit but requires import
185
- // Use a conservative estimate: RSS > 3GB is danger zone on most systems
186
- const heapMB = Math.round(mem.heapUsed / 1024 / 1024);
187
- const rssMB = Math.round(mem.rss / 1024 / 1024);
188
- // Try to get the actual V8 heap limit
189
- let limitMB = 4096; // conservative default
190
- try {
191
- const v8 = require("node:v8");
192
- const stats = v8.getHeapStatistics();
193
- limitMB = Math.round(stats.heap_size_limit / 1024 / 1024);
194
- }
195
- catch {
196
- limitMB = 4096; /* v8 stats unavailable — use conservative default */
197
- }
198
- const pct = heapMB / limitMB;
199
- return { pressured: pct > MEMORY_PRESSURE_THRESHOLD, heapMB, limitMB, pct };
200
- }
201
- function resolveDispatchNodeKind(unitType, sidecarItem) {
202
- if (sidecarItem?.kind === "hook")
203
- return "hook";
204
- if (sidecarItem?.kind === "triage")
205
- return "verification";
206
- if (sidecarItem?.kind === "quick-task")
207
- return "team-worker";
208
- if (unitType.startsWith("hook/"))
209
- return "hook";
210
- if (unitType === "reactive-execute")
211
- return "subagent";
212
- if (unitType === "gate-evaluate"
213
- || unitType === "validate-milestone"
214
- || unitType === "run-uat"
215
- || unitType === "complete-slice") {
216
- return "verification";
217
- }
218
- if (unitType === "replan-slice" || unitType === "reassess-roadmap") {
219
- return "reprocess";
220
- }
221
- return "unit";
222
- }
223
124
  async function enforceMinRequestInterval(s, prefs) {
224
125
  const minInterval = prefs?.min_request_interval_ms ?? 0;
225
- if (minInterval > 0 && s.lastRequestTimestamp > 0) {
226
- const elapsed = Date.now() - s.lastRequestTimestamp;
227
- if (elapsed < minInterval) {
228
- const waitMs = minInterval - elapsed;
229
- debugLog("autoLoop", { phase: "rate-limit-wait", waitMs });
230
- await new Promise(r => setTimeout(r, waitMs));
231
- }
126
+ const decision = decideMinRequestInterval({
127
+ minIntervalMs: minInterval,
128
+ lastRequestTimestamp: s.lastRequestTimestamp,
129
+ nowMs: Date.now(),
130
+ });
131
+ if (decision.action === "wait") {
132
+ debugLog("autoLoop", { phase: "rate-limit-wait", waitMs: decision.waitMs });
133
+ await new Promise(r => setTimeout(r, decision.waitMs));
232
134
  }
233
135
  }
234
- async function runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem) {
235
- if (dispatchContract === "legacy-direct") {
236
- return runUnitPhase(ic, iterData, loopState, sidecarItem);
237
- }
238
- const scheduler = new ExecutionGraphScheduler();
239
- let outcome = null;
240
- const executeNode = async () => {
241
- outcome = await runUnitPhase(ic, iterData, loopState, sidecarItem);
242
- };
243
- const kinds = [
244
- "unit",
245
- "hook",
246
- "subagent",
247
- "team-worker",
248
- "verification",
249
- "reprocess",
250
- ];
251
- for (const kind of kinds)
252
- scheduler.registerHandler(kind, executeNode);
253
- const nodeId = `dispatch:${ic.iteration}:${iterData.unitType}:${iterData.unitId}`;
254
- await scheduler.run([
255
- {
256
- id: nodeId,
257
- kind: resolveDispatchNodeKind(iterData.unitType, sidecarItem),
258
- dependsOn: [],
259
- metadata: {
260
- unitType: iterData.unitType,
261
- unitId: iterData.unitId,
262
- },
263
- },
264
- ], { parallel: false, maxWorkers: 1 });
265
- return outcome ?? { action: "break", reason: "scheduler-dispatch-missing-result" };
266
- }
267
136
  /**
268
137
  * Main auto-mode execution loop. Iterates: derive → dispatch → guards →
269
138
  * runUnit → finalize → repeat. Exits when s.active becomes false or a
@@ -276,6 +145,7 @@ export async function autoLoop(ctx, pi, s, deps, options) {
276
145
  debugLog("autoLoop", { phase: "enter" });
277
146
  let iteration = 0;
278
147
  const dispatchContract = options?.dispatchContract ?? "legacy-direct";
148
+ const unitDispatchDeps = createExecutionGraphUnitDispatchDeps();
279
149
  // Load persisted stuck state so counters survive session restarts (#3704)
280
150
  const persisted = loadStuckState(s);
281
151
  const loopState = {
@@ -289,65 +159,65 @@ export async function autoLoop(ctx, pi, s, deps, options) {
289
159
  while (s.active) {
290
160
  iteration++;
291
161
  debugLog("autoLoop", { phase: "loop-top", iteration });
292
- // Phase B: heartbeat the worker registry + active milestone lease so
293
- // janitors and concurrent workers see a live process. Best-effort —
294
- // DB unavailability or stale state must not stop the loop.
295
- if (s.workerId) {
296
- try {
297
- heartbeatAutoWorker(s.workerId);
298
- if (s.currentMilestoneId && s.milestoneLeaseToken) {
299
- refreshMilestoneLease(s.workerId, s.currentMilestoneId, s.milestoneLeaseToken);
300
- }
301
- }
302
- catch (err) {
303
- debugLog("autoLoop", {
304
- phase: "heartbeat-failed",
305
- error: err instanceof Error ? err.message : String(err),
306
- });
307
- }
308
- }
162
+ maintainWorkerHeartbeat(s, {
163
+ heartbeatAutoWorker,
164
+ refreshMilestoneLease,
165
+ logHeartbeatFailure: err => debugLog("autoLoop", {
166
+ phase: "heartbeat-failed",
167
+ error: err instanceof Error ? err.message : String(err),
168
+ }),
169
+ });
309
170
  // ── Journal: per-iteration flow grouping ──
310
171
  const flowId = randomUUID();
311
172
  let seqCounter = 0;
312
173
  const nextSeq = () => ++seqCounter;
174
+ const journalReporter = createWorkflowJournalReporter({
175
+ emitJournalEvent: deps.emitJournalEvent,
176
+ flowId,
177
+ nextSeq,
178
+ });
313
179
  const turnId = randomUUID();
314
180
  s.currentTraceId = flowId;
315
181
  s.currentTurnId = turnId;
316
182
  const turnStartedAt = new Date().toISOString();
317
183
  let observedUnitType;
318
184
  let observedUnitId;
319
- let turnFinished = false;
185
+ const phaseReporter = createWorkflowPhaseReporter({
186
+ observer: deps.uokObserver,
187
+ });
188
+ const turnReporter = createWorkflowTurnReporter({
189
+ observer: deps.uokObserver,
190
+ traceId: flowId,
191
+ turnId,
192
+ iteration,
193
+ basePath: s.basePath,
194
+ startedAt: turnStartedAt,
195
+ clearCurrentTurn: () => {
196
+ s.currentTraceId = null;
197
+ s.currentTurnId = null;
198
+ },
199
+ });
320
200
  const finishTurn = (status, failureClass = "none", error) => {
321
- if (turnFinished)
322
- return;
323
- turnFinished = true;
324
- deps.uokObserver?.onTurnResult({
325
- traceId: flowId,
326
- turnId,
327
- iteration,
201
+ turnReporter.finish({
328
202
  unitType: observedUnitType,
329
203
  unitId: observedUnitId,
330
204
  status,
331
205
  failureClass,
332
- phaseResults: [],
333
206
  error,
334
- startedAt: turnStartedAt,
335
- finishedAt: new Date().toISOString(),
336
207
  });
337
- s.currentTraceId = null;
338
- s.currentTurnId = null;
339
208
  };
340
- deps.uokObserver?.onTurnStart({
341
- traceId: flowId,
342
- turnId,
209
+ turnReporter.start();
210
+ const iterationDecision = decideWorkflowLoop({
211
+ active: s.active,
343
212
  iteration,
344
- basePath: s.basePath,
345
- startedAt: turnStartedAt,
213
+ maxIterations: MAX_LOOP_ITERATIONS,
214
+ hasCommandContext: true,
215
+ sessionLockValid: true,
346
216
  });
347
- if (iteration > MAX_LOOP_ITERATIONS) {
217
+ if (iterationDecision.action === "stop" && iterationDecision.reason === "max-iterations") {
348
218
  debugLog("autoLoop", {
349
219
  phase: "exit",
350
- reason: "max-iterations",
220
+ reason: iterationDecision.reason,
351
221
  iteration,
352
222
  });
353
223
  await deps.stopAuto(ctx, pi, `Safety: loop exceeded ${MAX_LOOP_ITERATIONS} iterations — possible runaway`);
@@ -357,69 +227,80 @@ export async function autoLoop(ctx, pi, s, deps, options) {
357
227
  // ── Memory pressure check (#3331) ──
358
228
  // Graceful shutdown before OOM killer sends SIGKILL.
359
229
  if (iteration % MEMORY_CHECK_INTERVAL === 0) {
360
- const mem = checkMemoryPressure();
230
+ const mem = measureMemoryPressure();
361
231
  debugLog("autoLoop", { phase: "memory-check", ...mem });
362
- if (mem.pressured) {
363
- logWarning("dispatch", `Memory pressure: ${mem.heapMB}MB / ${mem.limitMB}MB (${Math.round(mem.pct * 100)}%) — stopping auto-mode to prevent OOM kill`);
364
- await deps.stopAuto(ctx, pi, `Memory pressure: heap at ${mem.heapMB}MB / ${mem.limitMB}MB (${Math.round(mem.pct * 100)}%). ` +
365
- `Stopping gracefully to prevent OOM kill after ${iteration} iterations. ` +
366
- `Resume with /gsd auto to continue from where you left off.`);
367
- finishTurn("stopped", "timeout", "memory-pressure");
232
+ const memoryDecision = decideMemoryPressure({ ...mem, iteration });
233
+ if (memoryDecision.action === "stop") {
234
+ logWarning("dispatch", memoryDecision.warningMessage);
235
+ await deps.stopAuto(ctx, pi, memoryDecision.stopMessage);
236
+ finishTurn("stopped", "timeout", memoryDecision.turnError);
368
237
  break;
369
238
  }
370
239
  }
371
- if (!s.cmdCtx) {
240
+ const commandContextDecision = decideWorkflowLoop({
241
+ active: s.active,
242
+ iteration,
243
+ maxIterations: MAX_LOOP_ITERATIONS,
244
+ hasCommandContext: Boolean(s.cmdCtx),
245
+ sessionLockValid: true,
246
+ });
247
+ if (commandContextDecision.action === "stop" && commandContextDecision.reason === "missing-command-context") {
372
248
  debugLog("autoLoop", { phase: "exit", reason: "no-cmdCtx" });
373
- finishTurn("stopped", "manual-attention", "missing-command-context");
249
+ finishTurn("stopped", "manual-attention", commandContextDecision.reason);
374
250
  break;
375
251
  }
376
252
  let dispatchId = null;
377
253
  let dispatchSettled = false;
254
+ const completeIteration = () => {
255
+ completeWorkflowIteration({
256
+ get consecutiveErrors() { return consecutiveErrors; },
257
+ set consecutiveErrors(value) { consecutiveErrors = value; },
258
+ get consecutiveCooldowns() { return consecutiveCooldowns; },
259
+ set consecutiveCooldowns(value) { consecutiveCooldowns = value; },
260
+ recentErrorMessages,
261
+ }, {
262
+ emitIterationEnd: () => journalReporter.emit("iteration-end", { iteration }),
263
+ saveStuckState: () => saveStuckState(s, loopState),
264
+ logIterationComplete: () => debugLog("autoLoop", { phase: "iteration-complete", iteration }),
265
+ });
266
+ };
378
267
  try {
379
268
  // ── Blanket try/catch: one bad iteration must not kill the session
380
269
  const prefs = deps.loadEffectiveGSDPreferences()?.preferences;
381
270
  const uokFlags = resolveUokFlags(prefs);
382
- // ── Check sidecar queue before deriveState ──
383
- let sidecarItem;
384
- if (s.sidecarQueue.length > 0) {
385
- if (uokFlags.executionGraph && s.sidecarQueue.length > 1) {
386
- try {
387
- s.sidecarQueue = await scheduleSidecarQueue(s.sidecarQueue);
388
- }
389
- catch (err) {
390
- logWarning("dispatch", `sidecar queue scheduling failed: ${err instanceof Error ? err.message : String(err)}`);
391
- }
392
- }
393
- sidecarItem = s.sidecarQueue.shift();
394
- debugLog("autoLoop", {
395
- phase: "sidecar-dequeue",
396
- kind: sidecarItem.kind,
397
- unitType: sidecarItem.unitType,
398
- unitId: sidecarItem.unitId,
399
- });
400
- deps.emitJournalEvent({ ts: new Date().toISOString(), flowId, seq: nextSeq(), eventType: "sidecar-dequeue", data: { kind: sidecarItem.kind, unitType: sidecarItem.unitType, unitId: sidecarItem.unitId } });
401
- }
402
- const sessionLockBase = deps.lockBase();
403
- if (sessionLockBase) {
404
- const lockStatus = deps.validateSessionLock(sessionLockBase);
405
- if (!lockStatus.valid) {
406
- debugLog("autoLoop", {
271
+ const sessionLockOutcome = validateWorkflowSessionLock({
272
+ active: s.active,
273
+ iteration,
274
+ maxIterations: MAX_LOOP_ITERATIONS,
275
+ deps: {
276
+ lockBase: deps.lockBase,
277
+ validateSessionLock: deps.validateSessionLock,
278
+ handleLostSessionLock: lockStatus => deps.handleLostSessionLock(ctx, lockStatus),
279
+ logInvalidSessionLock: details => debugLog("autoLoop", {
407
280
  phase: "session-lock-invalid",
408
- reason: lockStatus.failureReason ?? "unknown",
409
- existingPid: lockStatus.existingPid,
410
- expectedPid: lockStatus.expectedPid,
411
- });
412
- deps.handleLostSessionLock(ctx, lockStatus);
413
- debugLog("autoLoop", {
281
+ ...details,
282
+ }),
283
+ logSessionLockExit: details => debugLog("autoLoop", {
414
284
  phase: "exit",
415
- reason: "session-lock-lost",
416
- detail: lockStatus.failureReason ?? "unknown",
417
- });
418
- break;
419
- }
285
+ ...details,
286
+ }),
287
+ },
288
+ });
289
+ if (sessionLockOutcome.action === "stop" && sessionLockOutcome.reason === "session-lock-lost") {
290
+ finishTurn("stopped", "manual-attention", sessionLockOutcome.reason);
291
+ break;
420
292
  }
293
+ // ── Check sidecar queue before deriveState ──
294
+ const sidecarItem = await dequeueSidecarItem({
295
+ queue: s.sidecarQueue,
296
+ executionGraphEnabled: uokFlags.executionGraph,
297
+ scheduleQueue: scheduleSidecarQueue,
298
+ warnSchedulingFailure: message => logWarning("dispatch", `sidecar queue scheduling failed: ${message}`),
299
+ logDequeue: payload => debugLog("autoLoop", { phase: "sidecar-dequeue", ...payload }),
300
+ emitDequeue: payload => journalReporter.emit("sidecar-dequeue", payload),
301
+ });
421
302
  const ic = { ctx, pi, s, deps, prefs, iteration, flowId, nextSeq };
422
- deps.emitJournalEvent({ ts: new Date().toISOString(), flowId, seq: nextSeq(), eventType: "iteration-start", data: { iteration } });
303
+ journalReporter.emit("iteration-start", { iteration });
423
304
  let iterData;
424
305
  // ── Custom engine path ──────────────────────────────────────────────
425
306
  // When activeEngineId is a non-dev value, bypass runPreDispatch and
@@ -429,7 +310,11 @@ export async function autoLoop(ctx, pi, s, deps, options) {
429
310
  //
430
311
  // GSD_ENGINE_BYPASS=1 skips the engine layer entirely — falls through
431
312
  // to the dev path below.
432
- if (s.activeEngineId != null && s.activeEngineId !== "dev" && !sidecarItem && process.env.GSD_ENGINE_BYPASS !== "1") {
313
+ if (shouldUseCustomEnginePath({
314
+ activeEngineId: s.activeEngineId,
315
+ hasSidecarItem: Boolean(sidecarItem),
316
+ engineBypass: process.env.GSD_ENGINE_BYPASS === "1",
317
+ })) {
433
318
  debugLog("autoLoop", { phase: "custom-engine-derive", iteration, engineId: s.activeEngineId });
434
319
  const { engine, policy } = resolveEngine({
435
320
  activeEngineId: s.activeEngineId,
@@ -447,48 +332,53 @@ export async function autoLoop(ctx, pi, s, deps, options) {
447
332
  isComplete: engineState.isComplete,
448
333
  });
449
334
  if (engineState.isComplete) {
335
+ finishTurn("completed");
450
336
  await deps.stopAuto(ctx, pi, "Workflow complete");
451
337
  break;
452
338
  }
453
339
  debugLog("autoLoop", { phase: "custom-engine-dispatch", iteration });
454
340
  const dispatch = await engine.resolveDispatch(engineState, { basePath: s.basePath });
455
- if (dispatch.action === "stop") {
456
- await deps.stopAuto(ctx, pi, dispatch.reason ?? "Engine stopped");
341
+ const engineDispatchDecision = decideEngineDispatch(dispatch.action === "stop"
342
+ ? { action: "stop", reason: dispatch.reason }
343
+ : { action: dispatch.action });
344
+ const dispatchFlow = await handleCustomEngineDispatchOutcome({
345
+ decision: engineDispatchDecision,
346
+ deps: {
347
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
348
+ },
349
+ });
350
+ if (dispatchFlow.action === "break") {
351
+ finishTurn("stopped", "manual-attention", "custom-engine-dispatch-stop");
457
352
  break;
458
353
  }
459
- if (dispatch.action === "skip") {
354
+ if (dispatchFlow.action === "continue") {
355
+ finishTurn("skipped");
460
356
  continue;
461
357
  }
462
358
  // dispatch.action === "dispatch"
359
+ if (dispatch.action !== "dispatch") {
360
+ finishTurn("skipped");
361
+ continue;
362
+ }
463
363
  const step = dispatch.step;
464
- const gsdState = await deps.deriveState(s.canonicalProjectRoot);
465
- debugLog("autoLoop", {
466
- phase: "post-derive",
467
- site: "custom-engine-gsd-state",
364
+ iterData = await buildCustomEngineIterationData({
365
+ step,
468
366
  basePath: s.basePath,
469
367
  canonicalProjectRoot: s.canonicalProjectRoot,
470
- derivedPhase: gsdState.phase,
471
- activeUnit: gsdState.activeTask?.id ?? gsdState.activeSlice?.id ?? gsdState.activeMilestone?.id,
368
+ currentMilestoneId: s.currentMilestoneId,
369
+ deriveState: deps.deriveState,
370
+ logPostDerive: details => debugLog("autoLoop", {
371
+ phase: "post-derive",
372
+ ...details,
373
+ }),
472
374
  });
473
- iterData = {
474
- unitType: step.unitType,
475
- unitId: step.unitId,
476
- prompt: step.prompt,
477
- finalPrompt: step.prompt,
478
- pauseAfterUatDispatch: false,
479
- state: gsdState,
480
- mid: s.currentMilestoneId ?? "workflow",
481
- midTitle: "Workflow",
482
- isRetry: false,
483
- previousTier: undefined,
484
- };
485
375
  observedUnitType = iterData.unitType;
486
376
  observedUnitId = iterData.unitId;
487
377
  // ── Progress widget (mirrors dev path in runDispatch) ──
488
378
  deps.updateProgressWidget(ctx, iterData.unitType, iterData.unitId, iterData.state);
489
379
  // ── Guards (shared with dev path) ──
490
380
  const guardsResult = await runGuards(ic, s.currentMilestoneId ?? "workflow");
491
- deps.uokObserver?.onPhaseResult("guard", guardsResult.action, {
381
+ phaseReporter.report("guard", guardsResult.action, {
492
382
  unitType: iterData.unitType,
493
383
  unitId: iterData.unitId,
494
384
  });
@@ -498,13 +388,13 @@ export async function autoLoop(ctx, pi, s, deps, options) {
498
388
  }
499
389
  // ── Unit execution (shared with dev path) ──
500
390
  await enforceMinRequestInterval(s, prefs);
501
- const unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState);
391
+ const unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, undefined, unitDispatchDeps);
502
392
  if (unitPhaseResult.action === "next") {
503
- const requestTimestamp = unitPhaseResult.data.requestDispatchedAt ?? unitPhaseResult.data.unitStartedAt;
504
- if (typeof requestTimestamp === "number")
393
+ const requestTimestamp = resolveUnitRequestTimestamp(unitPhaseResult.data);
394
+ if (requestTimestamp !== undefined)
505
395
  s.lastRequestTimestamp = requestTimestamp;
506
396
  }
507
- deps.uokObserver?.onPhaseResult("unit", unitPhaseResult.action, {
397
+ phaseReporter.report("unit", unitPhaseResult.action, {
508
398
  unitType: iterData.unitType,
509
399
  unitId: iterData.unitId,
510
400
  });
@@ -516,103 +406,94 @@ export async function autoLoop(ctx, pi, s, deps, options) {
516
406
  debugLog("autoLoop", { phase: "custom-engine-verify", iteration, unitId: iterData.unitId });
517
407
  const verifyResult = await policy.verify(iterData.unitType, iterData.unitId, { basePath: s.basePath });
518
408
  if (verifyResult === "pause") {
519
- await deps.pauseAuto(ctx, pi);
520
- deps.uokObserver?.onPhaseResult("custom-engine", "pause", {
409
+ const verifyFlow = await handleCustomEngineVerifyPause({
521
410
  unitType: iterData.unitType,
522
411
  unitId: iterData.unitId,
412
+ deps: {
413
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
414
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
415
+ reportPause: details => phaseReporter.report("custom-engine", "pause", details),
416
+ finishTurn,
417
+ },
523
418
  });
524
- finishTurn("paused", "manual-attention", "custom-engine-verify-pause");
525
- break;
419
+ if (verifyFlow.action === "break")
420
+ break;
526
421
  }
527
422
  if (verifyResult === "retry") {
528
- const recoveryKey = `${iterData.unitType}/${iterData.unitId}`;
529
- const retryCounts = hydrateCustomVerifyRetryCounts(s);
530
- const attempts = (retryCounts.get(recoveryKey) ?? 0) + 1;
531
- retryCounts.set(recoveryKey, attempts);
532
- saveCustomVerifyRetryCounts(s);
533
- debugLog("autoLoop", { phase: "custom-engine-verify-retry", iteration, unitId: iterData.unitId, attempts });
534
- deps.uokObserver?.onPhaseResult("custom-engine", "retry", {
423
+ const retryOutcome = await handleCustomEngineVerifyRetry({
424
+ session: s,
535
425
  unitType: iterData.unitType,
536
426
  unitId: iterData.unitId,
537
- attempts,
427
+ basePath: s.basePath,
428
+ iteration,
429
+ maxRetries: MAX_CUSTOM_ENGINE_VERIFY_RETRIES,
430
+ deps: {
431
+ hydrateRetryCounts: () => hydrateCustomVerifyRetryCounts(s, {
432
+ logFailure: logCustomVerifyRetryLoadFailure,
433
+ }),
434
+ saveRetryCounts: () => saveCustomVerifyRetryCounts(s, {
435
+ logFailure: logCustomVerifyRetrySaveFailure,
436
+ }),
437
+ recover: (unitType, unitId, options) => policy.recover(unitType, unitId, options),
438
+ logRetry: details => debugLog("autoLoop", {
439
+ phase: "custom-engine-verify-retry",
440
+ ...details,
441
+ }),
442
+ reportRetry: details => phaseReporter.report("custom-engine", "retry", details),
443
+ },
538
444
  });
539
- if (attempts > MAX_CUSTOM_ENGINE_VERIFY_RETRIES) {
540
- const recovery = await policy.recover(iterData.unitType, iterData.unitId, { basePath: s.basePath });
541
- if (recovery.outcome === "pause") {
542
- await deps.pauseAuto(ctx, pi);
543
- finishTurn("paused", "manual-attention", recovery.reason ?? "custom-engine-verify-retry-exhausted");
544
- break;
545
- }
546
- if (recovery.outcome === "skip") {
547
- await deps.stopAuto(ctx, pi, recovery.reason ??
548
- `Custom workflow verification for ${iterData.unitId} requested skip after retry exhaustion, but the custom engine cannot reconcile skipped steps.`);
549
- finishTurn("stopped", "manual-attention", "custom-engine-verify-retry-exhausted");
550
- break;
551
- }
552
- const exhaustedReason = `Custom workflow verification for ${iterData.unitId} requested retry ${attempts} times without passing.`;
553
- await deps.stopAuto(ctx, pi, recovery.outcome === "stop" && recovery.reason ? recovery.reason : exhaustedReason);
554
- finishTurn("stopped", "manual-attention", "custom-engine-verify-retry-exhausted");
445
+ const retryFlow = await handleCustomEngineVerifyRetryOutcome({
446
+ outcome: retryOutcome,
447
+ deps: {
448
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
449
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
450
+ reportPause: details => phaseReporter.report("custom-engine", "pause", details),
451
+ finishTurn,
452
+ },
453
+ });
454
+ if (retryFlow.action === "break")
555
455
  break;
556
- }
557
- finishTurn("retry");
558
456
  continue;
559
457
  }
560
458
  // Verification passed — mark step complete
561
- s.verificationRetryCount?.delete(`${iterData.unitType}/${iterData.unitId}`);
562
- saveCustomVerifyRetryCounts(s);
563
- debugLog("autoLoop", { phase: "custom-engine-reconcile", iteration, unitId: iterData.unitId });
564
- const reconcileResult = await engine.reconcile(engineState, {
565
- unitType: iterData.unitType,
566
- unitId: iterData.unitId,
567
- startedAt: s.currentUnit?.startedAt ?? Date.now(),
568
- finishedAt: Date.now(),
459
+ const reconcileOutcome = await handleCustomEngineReconcile({
460
+ session: s,
461
+ engineState,
462
+ iterData,
463
+ iteration,
464
+ deps: {
465
+ saveRetryCounts: () => saveCustomVerifyRetryCounts(s, {
466
+ logFailure: logCustomVerifyRetrySaveFailure,
467
+ }),
468
+ logReconcile: details => debugLog("autoLoop", {
469
+ phase: "custom-engine-reconcile",
470
+ ...details,
471
+ }),
472
+ reconcile: (state, completedStep) => engine.reconcile(state, completedStep),
473
+ now: () => Date.now(),
474
+ clearUnitTimeout: deps.clearUnitTimeout,
475
+ completeIteration,
476
+ },
569
477
  });
570
- deps.clearUnitTimeout();
571
- consecutiveErrors = 0;
572
- consecutiveCooldowns = 0;
573
- recentErrorMessages.length = 0;
574
- deps.emitJournalEvent({ ts: new Date().toISOString(), flowId, seq: nextSeq(), eventType: "iteration-end", data: { iteration } });
575
- saveStuckState(s, loopState); // persist across session restarts (#3704)
576
- debugLog("autoLoop", { phase: "iteration-complete", iteration });
577
- if (reconcileResult.outcome === "milestone-complete") {
578
- await deps.stopAuto(ctx, pi, "Workflow complete");
579
- deps.uokObserver?.onPhaseResult("custom-engine", "milestone-complete", {
580
- unitType: iterData.unitType,
581
- unitId: iterData.unitId,
582
- });
583
- finishTurn("completed");
584
- break;
585
- }
586
- if (reconcileResult.outcome === "pause") {
587
- await deps.pauseAuto(ctx, pi);
588
- deps.uokObserver?.onPhaseResult("custom-engine", "pause", {
589
- unitType: iterData.unitType,
590
- unitId: iterData.unitId,
591
- });
592
- finishTurn("paused", "manual-attention");
593
- break;
594
- }
595
- if (reconcileResult.outcome === "stop") {
596
- await deps.stopAuto(ctx, pi, reconcileResult.reason ?? "Engine stopped");
597
- deps.uokObserver?.onPhaseResult("custom-engine", "stop", {
598
- unitType: iterData.unitType,
599
- unitId: iterData.unitId,
600
- reason: reconcileResult.reason,
601
- });
602
- finishTurn("stopped", "manual-attention", reconcileResult.reason);
603
- break;
604
- }
605
- deps.uokObserver?.onPhaseResult("custom-engine", "continue", {
478
+ const reconcileFlow = await handleCustomEngineReconcileOutcome({
479
+ outcome: reconcileOutcome,
606
480
  unitType: iterData.unitType,
607
481
  unitId: iterData.unitId,
482
+ deps: {
483
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
484
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
485
+ report: (action, details) => phaseReporter.report("custom-engine", action, details),
486
+ finishTurn,
487
+ },
608
488
  });
609
- finishTurn("completed");
489
+ if (reconcileFlow.action === "break")
490
+ break;
610
491
  continue;
611
492
  }
612
493
  if (!sidecarItem) {
613
494
  // ── Phase 1: Pre-dispatch ─────────────────────────────────────────
614
495
  const preDispatchResult = await runPreDispatch(ic, loopState);
615
- deps.uokObserver?.onPhaseResult("pre-dispatch", preDispatchResult.action);
496
+ phaseReporter.report("pre-dispatch", preDispatchResult.action);
616
497
  if (preDispatchResult.action === "break") {
617
498
  finishTurn("stopped", "manual-attention", "pre-dispatch-break");
618
499
  break;
@@ -624,14 +505,14 @@ export async function autoLoop(ctx, pi, s, deps, options) {
624
505
  const preData = preDispatchResult.data;
625
506
  // ── Phase 2: Guards ───────────────────────────────────────────────
626
507
  const guardsResult = await runGuards(ic, preData.mid);
627
- deps.uokObserver?.onPhaseResult("guard", guardsResult.action);
508
+ phaseReporter.report("guard", guardsResult.action);
628
509
  if (guardsResult.action === "break") {
629
510
  finishTurn("stopped", "manual-attention", "guard-break");
630
511
  break;
631
512
  }
632
513
  // ── Phase 3: Dispatch ─────────────────────────────────────────────
633
514
  const dispatchResult = await runDispatch(ic, preData, loopState);
634
- deps.uokObserver?.onPhaseResult("dispatch", dispatchResult.action);
515
+ phaseReporter.report("dispatch", dispatchResult.action);
635
516
  if (dispatchResult.action === "break") {
636
517
  finishTurn("stopped", "manual-attention", "dispatch-break");
637
518
  break;
@@ -645,30 +526,19 @@ export async function autoLoop(ctx, pi, s, deps, options) {
645
526
  observedUnitId = iterData.unitId;
646
527
  }
647
528
  else {
648
- // ── Sidecar path: use values from the sidecar item directly ──
649
- const sidecarState = await deps.deriveState(s.canonicalProjectRoot);
650
- debugLog("autoLoop", {
651
- phase: "post-derive",
652
- site: "sidecar",
529
+ iterData = await buildSidecarIterationData({
530
+ sidecarItem,
653
531
  basePath: s.basePath,
654
532
  canonicalProjectRoot: s.canonicalProjectRoot,
655
- derivedPhase: sidecarState.phase,
656
- activeUnit: sidecarState.activeTask?.id ?? sidecarState.activeSlice?.id ?? sidecarState.activeMilestone?.id,
533
+ deriveState: deps.deriveState,
534
+ logPostDerive: details => debugLog("autoLoop", {
535
+ phase: "post-derive",
536
+ ...details,
537
+ }),
657
538
  });
658
- iterData = {
659
- unitType: sidecarItem.unitType,
660
- unitId: sidecarItem.unitId,
661
- prompt: sidecarItem.prompt,
662
- finalPrompt: sidecarItem.prompt,
663
- pauseAfterUatDispatch: false,
664
- state: sidecarState,
665
- mid: sidecarState.activeMilestone?.id,
666
- midTitle: sidecarState.activeMilestone?.title,
667
- isRetry: false, previousTier: undefined,
668
- };
669
539
  observedUnitType = iterData.unitType;
670
540
  observedUnitId = iterData.unitId;
671
- deps.uokObserver?.onPhaseResult("dispatch", "sidecar", {
541
+ phaseReporter.report("dispatch", "sidecar", {
672
542
  unitType: iterData.unitType,
673
543
  unitId: iterData.unitId,
674
544
  sidecarKind: sidecarItem.kind,
@@ -681,52 +551,51 @@ export async function autoLoop(ctx, pi, s, deps, options) {
681
551
  // null when DB unavailable, no worker registered, or no active lease
682
552
  // — those degraded paths fall through to the existing single-worker
683
553
  // semantics with no ledger entry, preserving back-compat.
684
- const dispatchClaim = openDispatchClaim(s, flowId, turnId, iterData);
685
- if (dispatchClaim.kind === "skip") {
686
- finishTurn("skipped", "execution", dispatchClaim.reason);
554
+ const dispatchClaim = openDispatchClaim(s, flowId, turnId, iterData, {
555
+ getRecentDispatchesForUnit,
556
+ recordDispatchClaim,
557
+ markDispatchRunning,
558
+ logClaimRejected: logDispatchClaimRejected,
559
+ logClaimFailed: logDispatchClaimFailed,
560
+ });
561
+ const dispatchDecision = decideDispatchClaim(dispatchClaim.kind === "opened"
562
+ ? { kind: "opened", dispatchId: dispatchClaim.dispatchId }
563
+ : dispatchClaim.kind === "skip"
564
+ ? { kind: "skip", reason: dispatchClaim.reason }
565
+ : { kind: "degraded" });
566
+ if (dispatchDecision.action === "skip") {
567
+ finishTurn("skipped", "execution", dispatchDecision.reason);
687
568
  continue;
688
569
  }
689
- dispatchId = dispatchClaim.kind === "opened" ? dispatchClaim.dispatchId : null;
570
+ dispatchId = dispatchDecision.dispatchId;
690
571
  let unitPhaseResult;
691
572
  try {
692
- unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem);
573
+ unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem, unitDispatchDeps);
693
574
  }
694
575
  catch (err) {
695
576
  if (err instanceof ModelPolicyDispatchBlockedError) {
696
577
  throw err;
697
578
  }
698
- if (dispatchId !== null) {
699
- try {
700
- markDispatchFailed(dispatchId, {
701
- errorSummary: `exception:${err instanceof Error ? err.message : String(err)}`,
702
- });
703
- dispatchSettled = true;
704
- }
705
- catch (ledgerErr) {
706
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: ledgerErr instanceof Error ? ledgerErr.message : String(ledgerErr) });
707
- }
708
- }
579
+ dispatchSettled = settleDispatchFailed(dispatchId, formatDispatchExceptionSummary({ error: err }), {
580
+ markFailed: markDispatchFailed,
581
+ logWriteFailure: logDispatchLedgerWriteFailure,
582
+ }) || dispatchSettled;
709
583
  throw err;
710
584
  }
711
585
  if (unitPhaseResult.action === "next") {
712
- const requestTimestamp = unitPhaseResult.data.requestDispatchedAt ?? unitPhaseResult.data.unitStartedAt;
713
- if (typeof requestTimestamp === "number")
586
+ const requestTimestamp = resolveUnitRequestTimestamp(unitPhaseResult.data);
587
+ if (requestTimestamp !== undefined)
714
588
  s.lastRequestTimestamp = requestTimestamp;
715
589
  }
716
- deps.uokObserver?.onPhaseResult("unit", unitPhaseResult.action, {
590
+ phaseReporter.report("unit", unitPhaseResult.action, {
717
591
  unitType: iterData.unitType,
718
592
  unitId: iterData.unitId,
719
593
  });
720
594
  if (unitPhaseResult.action === "break") {
721
- if (dispatchId !== null) {
722
- try {
723
- markDispatchFailed(dispatchId, { errorSummary: "unit-break" });
724
- dispatchSettled = true;
725
- }
726
- catch (err) {
727
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: err instanceof Error ? err.message : String(err) });
728
- }
729
- }
595
+ dispatchSettled = settleDispatchFailed(dispatchId, "unit-break", {
596
+ markFailed: markDispatchFailed,
597
+ logWriteFailure: logDispatchLedgerWriteFailure,
598
+ }) || dispatchSettled;
730
599
  finishTurn("stopped", "execution", "unit-break");
731
600
  break;
732
601
  }
@@ -736,88 +605,57 @@ export async function autoLoop(ctx, pi, s, deps, options) {
736
605
  finalizeResult = await runFinalize(ic, iterData, loopState, sidecarItem);
737
606
  }
738
607
  catch (err) {
739
- if (dispatchId !== null) {
740
- try {
741
- markDispatchFailed(dispatchId, {
742
- errorSummary: `exception:${err instanceof Error ? err.message : String(err)}`,
743
- });
744
- dispatchSettled = true;
745
- }
746
- catch (ledgerErr) {
747
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: ledgerErr instanceof Error ? ledgerErr.message : String(ledgerErr) });
748
- }
749
- }
608
+ dispatchSettled = settleDispatchFailed(dispatchId, formatDispatchExceptionSummary({ error: err }), {
609
+ markFailed: markDispatchFailed,
610
+ logWriteFailure: logDispatchLedgerWriteFailure,
611
+ }) || dispatchSettled;
750
612
  throw err;
751
613
  }
752
- deps.uokObserver?.onPhaseResult("finalize", finalizeResult.action, {
614
+ phaseReporter.report("finalize", finalizeResult.action, {
753
615
  unitType: iterData.unitType,
754
616
  unitId: iterData.unitId,
755
617
  });
756
- if (finalizeResult.action === "break") {
757
- const finalizeFailureClass = finalizeResult.reason === "git-closeout-failure"
758
- ? "git"
759
- : "closeout";
760
- if (dispatchId !== null) {
761
- try {
762
- markDispatchFailed(dispatchId, { errorSummary: `finalize-break:${finalizeResult.reason ?? "unknown"}` });
763
- dispatchSettled = true;
764
- }
765
- catch (err) {
766
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: err instanceof Error ? err.message : String(err) });
767
- }
768
- }
769
- finishTurn("stopped", finalizeFailureClass, "finalize-break");
618
+ const finalizeDecision = decideFinalizeResult(finalizeResult.action === "break"
619
+ ? { action: "break", reason: finalizeResult.reason }
620
+ : finalizeResult.action === "continue"
621
+ ? { action: "continue" }
622
+ : { action: "next" });
623
+ if (finalizeDecision.action === "stop") {
624
+ dispatchSettled = settleDispatchFailed(dispatchId, finalizeDecision.ledgerErrorSummary, {
625
+ markFailed: markDispatchFailed,
626
+ logWriteFailure: logDispatchLedgerWriteFailure,
627
+ }) || dispatchSettled;
628
+ finishTurn("stopped", finalizeDecision.failureClass, finalizeDecision.turnError);
770
629
  break;
771
630
  }
772
- if (finalizeResult.action === "continue") {
773
- if (dispatchId !== null) {
774
- try {
775
- markDispatchFailed(dispatchId, { errorSummary: "finalize-retry" });
776
- dispatchSettled = true;
777
- }
778
- catch (err) {
779
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: err instanceof Error ? err.message : String(err) });
780
- }
781
- }
631
+ if (finalizeDecision.action === "retry") {
632
+ dispatchSettled = settleDispatchFailed(dispatchId, finalizeDecision.ledgerErrorSummary, {
633
+ markFailed: markDispatchFailed,
634
+ logWriteFailure: logDispatchLedgerWriteFailure,
635
+ }) || dispatchSettled;
782
636
  finishTurn("retry");
783
637
  continue;
784
638
  }
785
- if (dispatchId !== null) {
786
- try {
787
- markDispatchCompleted(dispatchId);
788
- dispatchSettled = true;
789
- }
790
- catch (err) {
791
- debugLog("autoLoop", { phase: "dispatch-ledger-write-failed", error: err instanceof Error ? err.message : String(err) });
792
- }
793
- }
794
- consecutiveErrors = 0; // Iteration completed successfully
795
- consecutiveCooldowns = 0;
796
- recentErrorMessages.length = 0;
797
- deps.emitJournalEvent({ ts: new Date().toISOString(), flowId, seq: nextSeq(), eventType: "iteration-end", data: { iteration } });
798
- saveStuckState(s, loopState); // persist across session restarts (#4382)
799
- debugLog("autoLoop", { phase: "iteration-complete", iteration });
639
+ dispatchSettled = settleDispatchCompleted(dispatchId, {
640
+ markCompleted: markDispatchCompleted,
641
+ logWriteFailure: logDispatchLedgerWriteFailure,
642
+ }) || dispatchSettled;
643
+ completeIteration();
800
644
  finishTurn("completed");
801
645
  }
802
646
  catch (loopErr) {
803
647
  // ── Blanket catch: absorb unexpected exceptions, apply graduated recovery ──
804
648
  const msg = loopErr instanceof Error ? loopErr.message : String(loopErr);
805
649
  if (dispatchId !== null && !dispatchSettled && !(loopErr instanceof ModelPolicyDispatchBlockedError)) {
806
- try {
807
- markDispatchFailed(dispatchId, { errorSummary: `unhandled-error:${msg.slice(0, 200)}` });
808
- dispatchSettled = true;
809
- }
810
- catch (err) {
811
- debugLog("autoLoop", {
812
- phase: "dispatch-ledger-write-failed",
813
- error: err instanceof Error ? err.message : String(err),
814
- });
815
- }
650
+ dispatchSettled = settleDispatchFailed(dispatchId, formatUnhandledDispatchErrorSummary({ error: loopErr }), {
651
+ markFailed: markDispatchFailed,
652
+ logWriteFailure: logDispatchLedgerWriteFailure,
653
+ }) || dispatchSettled;
816
654
  }
817
655
  // Always emit iteration-end on error so the journal records iteration
818
656
  // completion even on failure (#2344). Without this, errors in
819
657
  // runFinalize leave the journal incomplete, making diagnosis harder.
820
- deps.emitJournalEvent({ ts: new Date().toISOString(), flowId, seq: nextSeq(), eventType: "iteration-end", data: { iteration, error: msg } });
658
+ journalReporter.emit("iteration-end", { iteration, error: msg });
821
659
  // ── Pre-send model-policy block: not a retryable error (#4959 / #4850) ──
822
660
  // The model-policy gate runs before the prompt is sent. When every
823
661
  // candidate model is denied (cross-provider disabled + flat-rate
@@ -827,6 +665,12 @@ export async function autoLoop(ctx, pi, s, deps, options) {
827
665
  // instead, with the per-model deny reasons surfaced from the typed
828
666
  // error.
829
667
  if (loopErr instanceof ModelPolicyDispatchBlockedError) {
668
+ const policyDecision = decideModelPolicyBlocked({
669
+ unitType: loopErr.unitType,
670
+ unitId: loopErr.unitId,
671
+ errorMessage: msg,
672
+ reasons: loopErr.reasons,
673
+ });
830
674
  debugLog("autoLoop", {
831
675
  phase: "model-policy-blocked",
832
676
  iteration,
@@ -834,20 +678,8 @@ export async function autoLoop(ctx, pi, s, deps, options) {
834
678
  unitId: loopErr.unitId,
835
679
  reasons: loopErr.reasons,
836
680
  });
837
- ctx.ui.notify(`Auto-mode paused: model-policy denied dispatch for ${loopErr.unitType}/${loopErr.unitId}. ${msg}`, "error");
838
- deps.emitJournalEvent({
839
- ts: new Date().toISOString(),
840
- flowId,
841
- seq: nextSeq(),
842
- eventType: "unit-end",
843
- data: {
844
- unitType: loopErr.unitType,
845
- unitId: loopErr.unitId,
846
- status: "blocked",
847
- reason: "model-policy-dispatch-blocked",
848
- reasons: loopErr.reasons,
849
- },
850
- });
681
+ ctx.ui.notify(policyDecision.notifyMessage, "error");
682
+ journalReporter.emit("unit-end", policyDecision.journalData);
851
683
  // Carry the blocked unit identity into the turn-result observer:
852
684
  // the throw originated inside dispatch, so observedUnitType/Id were
853
685
  // not assigned by the success path at lines 453/631/647 — but the
@@ -855,7 +687,7 @@ export async function autoLoop(ctx, pi, s, deps, options) {
855
687
  observedUnitType = loopErr.unitType;
856
688
  observedUnitId = loopErr.unitId;
857
689
  await deps.pauseAuto(ctx, pi);
858
- finishTurn("paused", "manual-attention", msg);
690
+ finishTurn(policyDecision.turnStatus, policyDecision.failureClass, msg);
859
691
  // Do NOT increment consecutiveErrors — the failure is configuration,
860
692
  // not a transient runtime fault.
861
693
  break;
@@ -865,15 +697,19 @@ export async function autoLoop(ctx, pi, s, deps, options) {
865
697
  // LLM budget on guaranteed failures.
866
698
  const infraCode = isInfrastructureError(loopErr);
867
699
  if (infraCode) {
700
+ const infraDecision = decideInfrastructureError({
701
+ code: infraCode,
702
+ errorMessage: msg,
703
+ });
868
704
  debugLog("autoLoop", {
869
705
  phase: "infrastructure-error",
870
706
  iteration,
871
707
  code: infraCode,
872
708
  error: msg,
873
709
  });
874
- ctx.ui.notify(`Auto-mode stopped: infrastructure error ${infraCode} — ${msg}`, "error");
875
- await deps.stopAuto(ctx, pi, `Infrastructure error (${infraCode}): not recoverable by retry`);
876
- finishTurn("failed", "execution", msg);
710
+ ctx.ui.notify(infraDecision.notifyMessage, "error");
711
+ await deps.stopAuto(ctx, pi, infraDecision.stopMessage);
712
+ finishTurn(infraDecision.turnStatus, infraDecision.failureClass, msg);
877
713
  break;
878
714
  }
879
715
  // ── Credential cooldown: wait and retry with bounded budget ──
@@ -885,6 +721,12 @@ export async function autoLoop(ctx, pi, s, deps, options) {
885
721
  if (isTransientCooldownError(loopErr)) {
886
722
  consecutiveCooldowns++;
887
723
  const retryAfterMs = getCooldownRetryAfterMs(loopErr);
724
+ const cooldownDecision = decideCooldownRecovery({
725
+ consecutiveCooldowns,
726
+ maxCooldownRetries: MAX_COOLDOWN_RETRIES,
727
+ retryAfterMs,
728
+ fallbackWaitMs: COOLDOWN_FALLBACK_WAIT_MS,
729
+ });
888
730
  debugLog("autoLoop", {
889
731
  phase: "cooldown-wait",
890
732
  iteration,
@@ -892,16 +734,14 @@ export async function autoLoop(ctx, pi, s, deps, options) {
892
734
  retryAfterMs,
893
735
  error: msg,
894
736
  });
895
- if (consecutiveCooldowns > MAX_COOLDOWN_RETRIES) {
896
- ctx.ui.notify(`Auto-mode stopped: ${consecutiveCooldowns} consecutive credential cooldowns — rate limit or quota may be persistently exhausted.`, "error");
897
- await deps.stopAuto(ctx, pi, `${consecutiveCooldowns} consecutive credential cooldowns exceeded retry budget`);
737
+ if (cooldownDecision.action === "stop") {
738
+ ctx.ui.notify(cooldownDecision.notifyMessage, "error");
739
+ finishTurn("stopped", "timeout", msg);
740
+ await deps.stopAuto(ctx, pi, cooldownDecision.stopMessage);
898
741
  break;
899
742
  }
900
- const waitMs = (retryAfterMs !== undefined && retryAfterMs > 0 && retryAfterMs <= 60_000)
901
- ? retryAfterMs + 500 // Use structured hint + small buffer
902
- : COOLDOWN_FALLBACK_WAIT_MS;
903
- ctx.ui.notify(`Credentials in cooldown (${consecutiveCooldowns}/${MAX_COOLDOWN_RETRIES}) — waiting ${Math.round(waitMs / 1000)}s before retrying.`, "warning");
904
- await new Promise(resolve => setTimeout(resolve, waitMs));
743
+ ctx.ui.notify(cooldownDecision.notifyMessage, "warning");
744
+ await new Promise(resolve => setTimeout(resolve, cooldownDecision.waitMs));
905
745
  finishTurn("retry", "timeout", msg);
906
746
  continue; // Retry iteration without incrementing consecutiveErrors
907
747
  }
@@ -913,26 +753,25 @@ export async function autoLoop(ctx, pi, s, deps, options) {
913
753
  consecutiveErrors,
914
754
  error: msg,
915
755
  });
916
- if (consecutiveErrors >= 3) {
917
- // 3+ consecutive: hard stop — something is fundamentally broken
918
- const errorHistory = recentErrorMessages
919
- .map((m, i) => ` ${i + 1}. ${m}`)
920
- .join("\n");
921
- ctx.ui.notify(`Auto-mode stopped: ${consecutiveErrors} consecutive iteration failures:\n${errorHistory}`, "error");
922
- await deps.stopAuto(ctx, pi, `${consecutiveErrors} consecutive iteration failures`);
923
- finishTurn("failed", "execution", msg);
756
+ const errorDecision = decideIterationErrorRecovery({
757
+ consecutiveErrors,
758
+ recentErrorMessages,
759
+ currentErrorMessage: msg,
760
+ });
761
+ if (errorDecision.action === "stop") {
762
+ ctx.ui.notify(errorDecision.notifyMessage, "error");
763
+ await deps.stopAuto(ctx, pi, errorDecision.stopMessage);
764
+ finishTurn(errorDecision.turnStatus, "execution", msg);
924
765
  break;
925
766
  }
926
- else if (consecutiveErrors === 2) {
927
- // 2nd consecutive: try invalidating caches + re-deriving state
928
- ctx.ui.notify(`Iteration error (attempt ${consecutiveErrors}): ${msg}. Invalidating caches and retrying.`, "warning");
767
+ if (errorDecision.action === "invalidate-and-retry") {
768
+ ctx.ui.notify(errorDecision.notifyMessage, "warning");
929
769
  deps.invalidateAllCaches();
930
770
  }
931
771
  else {
932
- // 1st error: log and retry — transient failures happen
933
- ctx.ui.notify(`Iteration error: ${msg}. Retrying.`, "warning");
772
+ ctx.ui.notify(errorDecision.notifyMessage, "warning");
934
773
  }
935
- finishTurn("retry", "execution", msg);
774
+ finishTurn(errorDecision.turnStatus, "execution", msg);
936
775
  }
937
776
  }
938
777
  _clearCurrentResolve();