gsd-pi 2.80.0 → 2.81.0

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 (1172) hide show
  1. package/README.md +35 -59
  2. package/dist/claude-cli-check.d.ts +30 -0
  3. package/dist/claude-cli-check.js +18 -7
  4. package/dist/cli.js +0 -19
  5. package/dist/headless-query.d.ts +10 -0
  6. package/dist/headless-query.js +6 -4
  7. package/dist/loader-entrypoint.d.ts +8 -0
  8. package/dist/loader-entrypoint.js +27 -0
  9. package/dist/loader.js +2 -11
  10. package/dist/mcp-server.d.ts +1 -0
  11. package/dist/mcp-server.js +6 -3
  12. package/dist/resources/.managed-resources-content-hash +1 -1
  13. package/dist/resources/GSD-WORKFLOW.md +2 -2
  14. package/dist/resources/extensions/claude-code-cli/readiness.js +18 -7
  15. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +40 -3
  16. package/dist/resources/extensions/github-sync/sync.js +4 -1
  17. package/dist/resources/extensions/github-sync/templates.js +90 -74
  18. package/dist/resources/extensions/gsd/auto/contracts.js +2 -0
  19. package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +53 -0
  20. package/dist/resources/extensions/gsd/auto/loop.js +462 -525
  21. package/dist/resources/extensions/gsd/auto/orchestrator.js +48 -4
  22. package/dist/resources/extensions/gsd/auto/phases.js +311 -154
  23. package/dist/resources/extensions/gsd/auto/resolve.js +29 -0
  24. package/dist/resources/extensions/gsd/auto/run-unit.js +41 -45
  25. package/dist/resources/extensions/gsd/auto/session.js +9 -1
  26. package/dist/resources/extensions/gsd/auto/unit-runner-events.js +7 -0
  27. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-dispatch-outcome.js +12 -0
  28. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-iteration.js +24 -0
  29. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-reconcile-outcome.js +33 -0
  30. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-reconcile.js +26 -0
  31. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-retry.js +49 -0
  32. package/dist/resources/extensions/gsd/auto/workflow-custom-engine-verify-outcome.js +25 -0
  33. package/dist/resources/extensions/gsd/auto/workflow-dispatch-claim.js +80 -0
  34. package/dist/resources/extensions/gsd/auto/workflow-dispatch-ledger.js +26 -0
  35. package/dist/resources/extensions/gsd/auto/workflow-iteration-completion.js +10 -0
  36. package/dist/resources/extensions/gsd/auto/workflow-journal-reporter.js +16 -0
  37. package/dist/resources/extensions/gsd/auto/workflow-kernel.js +263 -0
  38. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +36 -0
  39. package/dist/resources/extensions/gsd/auto/workflow-phase-reporter.js +9 -0
  40. package/dist/resources/extensions/gsd/auto/workflow-session-lock.js +35 -0
  41. package/dist/resources/extensions/gsd/auto/workflow-sidecar-iteration.js +24 -0
  42. package/dist/resources/extensions/gsd/auto/workflow-sidecar-queue.js +26 -0
  43. package/dist/resources/extensions/gsd/auto/workflow-turn-reporter.js +36 -0
  44. package/dist/resources/extensions/gsd/auto/workflow-unit-dispatch.js +44 -0
  45. package/dist/resources/extensions/gsd/auto/workflow-worker-heartbeat.js +23 -0
  46. package/dist/resources/extensions/gsd/auto-dashboard.js +72 -15
  47. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +5 -32
  48. package/dist/resources/extensions/gsd/auto-dispatch.js +26 -0
  49. package/dist/resources/extensions/gsd/auto-post-unit.js +122 -88
  50. package/dist/resources/extensions/gsd/auto-prompts.js +382 -20
  51. package/dist/resources/extensions/gsd/auto-recovery.js +197 -9
  52. package/dist/resources/extensions/gsd/auto-start.js +254 -17
  53. package/dist/resources/extensions/gsd/auto-supervisor.js +8 -1
  54. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +2 -2
  55. package/dist/resources/extensions/gsd/auto-verification.js +2 -11
  56. package/dist/resources/extensions/gsd/auto-worktree.js +316 -374
  57. package/dist/resources/extensions/gsd/auto.js +273 -87
  58. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +136 -11
  59. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +54 -37
  60. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +37 -10
  61. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +30 -20
  62. package/dist/resources/extensions/gsd/bootstrap/journal-tools.js +4 -1
  63. package/dist/resources/extensions/gsd/bootstrap/memory-tools.js +6 -4
  64. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +5 -3
  65. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +1 -1
  66. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +349 -57
  67. package/dist/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.js +4 -0
  68. package/dist/resources/extensions/gsd/bootstrap/system-context.js +82 -23
  69. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +147 -2
  70. package/dist/resources/extensions/gsd/clean-root-preflight.js +65 -9
  71. package/dist/resources/extensions/gsd/commands/dispatcher.js +5 -0
  72. package/dist/resources/extensions/gsd/commands/handlers/ops.js +2 -2
  73. package/dist/resources/extensions/gsd/commands-config.js +1 -1
  74. package/dist/resources/extensions/gsd/commands-eval-review.js +2 -2
  75. package/dist/resources/extensions/gsd/commands-extract-learnings.js +17 -12
  76. package/dist/resources/extensions/gsd/commands-handlers.js +23 -9
  77. package/dist/resources/extensions/gsd/commands-ship.js +23 -46
  78. package/dist/resources/extensions/gsd/commands-workflow-templates.js +12 -7
  79. package/dist/resources/extensions/gsd/component-loader.js +5 -11
  80. package/dist/resources/extensions/gsd/context-budget.js +37 -2
  81. package/dist/resources/extensions/gsd/crash-recovery.js +56 -10
  82. package/dist/resources/extensions/gsd/custom-workflow-engine.js +25 -1
  83. package/dist/resources/extensions/gsd/dashboard-overlay.js +1 -1
  84. package/dist/resources/extensions/gsd/db/unit-dispatches.js +92 -0
  85. package/dist/resources/extensions/gsd/db-adapter.js +47 -0
  86. package/dist/resources/extensions/gsd/db-base-schema.js +353 -0
  87. package/dist/resources/extensions/gsd/db-connection-cache.js +31 -0
  88. package/dist/resources/extensions/gsd/db-coordination-schema.js +104 -0
  89. package/dist/resources/extensions/gsd/db-decision-requirement-rows.js +71 -0
  90. package/dist/resources/extensions/gsd/db-gate-rows.js +16 -0
  91. package/dist/resources/extensions/gsd/db-lightweight-query-rows.js +29 -0
  92. package/dist/resources/extensions/gsd/db-memory-fts-schema.js +56 -0
  93. package/dist/resources/extensions/gsd/db-migration-backup.js +22 -0
  94. package/dist/resources/extensions/gsd/db-migration-steps.js +416 -0
  95. package/dist/resources/extensions/gsd/db-milestone-artifact-rows.js +35 -0
  96. package/dist/resources/extensions/gsd/db-open-state.js +32 -0
  97. package/dist/resources/extensions/gsd/db-provider.js +108 -0
  98. package/dist/resources/extensions/gsd/db-runtime-kv-schema.js +27 -0
  99. package/dist/resources/extensions/gsd/db-schema-metadata.js +23 -0
  100. package/dist/resources/extensions/gsd/db-task-slice-rows.js +86 -0
  101. package/dist/resources/extensions/gsd/db-transaction.js +63 -0
  102. package/dist/resources/extensions/gsd/db-verification-evidence-rows.js +3 -0
  103. package/dist/resources/extensions/gsd/db-verification-evidence-schema.js +19 -0
  104. package/dist/resources/extensions/gsd/detection.js +106 -0
  105. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +2 -0
  106. package/dist/resources/extensions/gsd/escalation.js +2 -0
  107. package/dist/resources/extensions/gsd/git-service.js +73 -5
  108. package/dist/resources/extensions/gsd/graph.js +9 -3
  109. package/dist/resources/extensions/gsd/gsd-db.js +331 -1535
  110. package/dist/resources/extensions/gsd/guided-flow.js +149 -48
  111. package/dist/resources/extensions/gsd/health-widget.js +3 -0
  112. package/dist/resources/extensions/gsd/init-wizard.js +4 -1
  113. package/dist/resources/extensions/gsd/legacy-telemetry.js +70 -0
  114. package/dist/resources/extensions/gsd/markdown-renderer.js +2 -0
  115. package/dist/resources/extensions/gsd/memory-store.js +69 -12
  116. package/dist/resources/extensions/gsd/migrate/command.js +40 -1
  117. package/dist/resources/extensions/gsd/migration-auto-check.js +87 -0
  118. package/dist/resources/extensions/gsd/model-router.js +9 -6
  119. package/dist/resources/extensions/gsd/native-git-bridge.js +32 -8
  120. package/dist/resources/extensions/gsd/notification-widget.js +21 -3
  121. package/dist/resources/extensions/gsd/orphan-stash-audit.js +101 -0
  122. package/dist/resources/extensions/gsd/parallel-orchestrator.js +13 -3
  123. package/dist/resources/extensions/gsd/planning-path-scope.js +26 -0
  124. package/dist/resources/extensions/gsd/post-execution-checks.js +2 -0
  125. package/dist/resources/extensions/gsd/pr-evidence.js +117 -0
  126. package/dist/resources/extensions/gsd/pre-execution-checks.js +24 -0
  127. package/dist/resources/extensions/gsd/process-task-path.js +61 -0
  128. package/dist/resources/extensions/gsd/prompt-loader.js +37 -7
  129. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +41 -35
  130. package/dist/resources/extensions/gsd/prompts/complete-slice.md +23 -34
  131. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +50 -96
  132. package/dist/resources/extensions/gsd/prompts/discuss.md +81 -181
  133. package/dist/resources/extensions/gsd/prompts/execute-task.md +42 -67
  134. package/dist/resources/extensions/gsd/prompts/forensics.md +41 -84
  135. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +29 -39
  136. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +30 -65
  137. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +25 -52
  138. package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +36 -36
  139. package/dist/resources/extensions/gsd/prompts/guided-research-project.md +20 -38
  140. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +1 -1
  141. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +47 -59
  142. package/dist/resources/extensions/gsd/prompts/plan-slice.md +25 -87
  143. package/dist/resources/extensions/gsd/prompts/queue.md +46 -53
  144. package/dist/resources/extensions/gsd/prompts/quick-task.md +1 -5
  145. package/dist/resources/extensions/gsd/prompts/refine-slice.md +23 -23
  146. package/dist/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  147. package/dist/resources/extensions/gsd/prompts/research-slice.md +23 -23
  148. package/dist/resources/extensions/gsd/prompts/rethink.md +10 -10
  149. package/dist/resources/extensions/gsd/prompts/system.md +65 -107
  150. package/dist/resources/extensions/gsd/prompts/triage-captures.md +15 -15
  151. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +24 -24
  152. package/dist/resources/extensions/gsd/prompts/worktree-merge.md +35 -35
  153. package/dist/resources/extensions/gsd/quick.js +34 -2
  154. package/dist/resources/extensions/gsd/recovery-classification.js +94 -0
  155. package/dist/resources/extensions/gsd/safety/evidence-collector.js +10 -2
  156. package/dist/resources/extensions/gsd/slice-cadence.js +45 -2
  157. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +15 -9
  158. package/dist/resources/extensions/gsd/state-reconciliation.js +27 -0
  159. package/dist/resources/extensions/gsd/state.js +4 -0
  160. package/dist/resources/extensions/gsd/tool-contract.js +50 -0
  161. package/dist/resources/extensions/gsd/tools/complete-milestone.js +20 -16
  162. package/dist/resources/extensions/gsd/tools/complete-task.js +3 -1
  163. package/dist/resources/extensions/gsd/tools/context-mode-tool-result.js +15 -0
  164. package/dist/resources/extensions/gsd/tools/exec-search-tool.js +5 -0
  165. package/dist/resources/extensions/gsd/tools/exec-tool.js +3 -15
  166. package/dist/resources/extensions/gsd/tools/memory-tools.js +1 -0
  167. package/dist/resources/extensions/gsd/tools/plan-slice.js +9 -0
  168. package/dist/resources/extensions/gsd/tools/plan-task.js +9 -0
  169. package/dist/resources/extensions/gsd/tools/resume-tool.js +5 -0
  170. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +7 -2
  171. package/dist/resources/extensions/gsd/unit-context-composer.js +12 -3
  172. package/dist/resources/extensions/gsd/unit-runtime.js +22 -0
  173. package/dist/resources/extensions/gsd/uok/kernel.js +8 -3
  174. package/dist/resources/extensions/gsd/uok/plan-v2.js +2 -0
  175. package/dist/resources/extensions/gsd/workflow-logger.js +13 -13
  176. package/dist/resources/extensions/gsd/workflow-manifest.js +2 -0
  177. package/dist/resources/extensions/gsd/workflow-projections.js +2 -0
  178. package/dist/resources/extensions/gsd/workflow-protocol.js +131 -0
  179. package/dist/resources/extensions/gsd/workflow-templates.js +9 -0
  180. package/dist/resources/extensions/gsd/working-output-messages.js +64 -0
  181. package/dist/resources/extensions/gsd/worktree-lifecycle.js +958 -0
  182. package/dist/resources/extensions/gsd/worktree-manager.js +16 -14
  183. package/dist/resources/extensions/gsd/worktree-safety.js +119 -0
  184. package/dist/resources/extensions/gsd/worktree-state-projection.js +317 -0
  185. package/dist/resources/skills/web-quality-audit/scripts/analyze.sh +0 -0
  186. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  187. package/dist/web/standalone/.next/BUILD_ID +1 -1
  188. package/dist/web/standalone/.next/app-path-routes-manifest.json +3 -3
  189. package/dist/web/standalone/.next/build-manifest.json +3 -3
  190. package/dist/web/standalone/.next/react-loadable-manifest.json +2 -2
  191. package/dist/web/standalone/.next/required-server-files.json +1 -1
  192. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  193. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  194. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  195. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  196. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  197. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  198. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  199. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  200. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  201. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  202. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  203. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  204. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  205. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  206. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  207. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  208. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  209. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  210. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  211. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  212. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  213. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  214. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  215. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  216. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +1 -1
  217. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  218. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  219. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  220. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  221. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  222. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  223. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  224. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  225. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  226. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  227. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  228. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  229. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  230. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  231. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  232. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  233. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  234. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  235. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  236. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  237. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  238. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  239. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  240. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  241. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  242. package/dist/web/standalone/.next/server/app/api/notifications/route.js +1 -1
  243. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  244. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  245. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  246. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  247. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  248. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  249. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  250. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  251. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  252. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  253. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  254. package/dist/web/standalone/.next/server/app/api/session/events/route.js +1 -1
  255. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  256. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  257. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  258. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  259. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  260. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  261. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  262. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  263. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  264. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  265. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  266. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  267. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  268. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  269. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  270. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  271. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  272. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  273. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  274. package/dist/web/standalone/.next/server/app/index.html +1 -1
  275. package/dist/web/standalone/.next/server/app/index.rsc +2 -2
  276. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  277. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
  278. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  279. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  280. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  281. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  282. package/dist/web/standalone/.next/server/app-paths-manifest.json +3 -3
  283. package/dist/web/standalone/.next/server/chunks/167.js +2 -0
  284. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  285. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  286. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  287. package/dist/web/standalone/.next/server/middleware.js +2 -2
  288. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  289. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  290. package/dist/web/standalone/.next/static/chunks/8359.e059d86b255fce1c.js +10 -0
  291. package/dist/web/standalone/.next/static/chunks/app/{page-ff639266d978f2a0.js → page-200592a7f3baf579.js} +1 -1
  292. package/dist/web/standalone/.next/static/chunks/{webpack-d82dbee6356c1733.js → webpack-de742b64187e13fe.js} +1 -1
  293. package/dist/web/standalone/package.json +1 -0
  294. package/dist/web/standalone/server.js +1 -1
  295. package/dist/welcome-screen.d.ts +2 -0
  296. package/dist/welcome-screen.js +9 -7
  297. package/package.json +20 -9
  298. package/packages/contracts/dist/index.d.ts +3 -0
  299. package/packages/contracts/dist/index.d.ts.map +1 -0
  300. package/packages/contracts/dist/index.js +5 -0
  301. package/packages/contracts/dist/index.js.map +1 -0
  302. package/packages/contracts/dist/rpc.d.ts +549 -0
  303. package/packages/contracts/dist/rpc.d.ts.map +1 -0
  304. package/packages/contracts/dist/rpc.js +53 -0
  305. package/packages/contracts/dist/rpc.js.map +1 -0
  306. package/packages/contracts/dist/rpc.test.d.ts +2 -0
  307. package/packages/contracts/dist/rpc.test.d.ts.map +1 -0
  308. package/packages/contracts/dist/rpc.test.js +47 -0
  309. package/packages/contracts/dist/rpc.test.js.map +1 -0
  310. package/packages/contracts/dist/workflow.d.ts +180 -0
  311. package/packages/contracts/dist/workflow.d.ts.map +1 -0
  312. package/packages/contracts/dist/workflow.js +201 -0
  313. package/packages/contracts/dist/workflow.js.map +1 -0
  314. package/packages/contracts/package.json +39 -0
  315. package/packages/contracts/src/index.ts +5 -0
  316. package/packages/contracts/src/rpc.test.ts +72 -0
  317. package/packages/contracts/src/rpc.ts +286 -0
  318. package/packages/contracts/src/workflow.ts +213 -0
  319. package/packages/contracts/tsconfig.json +25 -0
  320. package/packages/daemon/package.json +3 -2
  321. package/packages/daemon/src/event-bridge.test.ts +2 -1
  322. package/packages/daemon/src/event-bridge.ts +1 -1
  323. package/packages/daemon/src/event-formatter.test.ts +1 -2
  324. package/packages/daemon/src/event-formatter.ts +1 -2
  325. package/packages/daemon/src/session-manager.ts +2 -2
  326. package/packages/daemon/src/types.ts +3 -18
  327. package/packages/mcp-server/dist/server.d.ts +13 -0
  328. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  329. package/packages/mcp-server/dist/server.js +77 -0
  330. package/packages/mcp-server/dist/server.js.map +1 -1
  331. package/packages/mcp-server/dist/session-manager.js +1 -1
  332. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  333. package/packages/mcp-server/dist/types.d.ts +3 -11
  334. package/packages/mcp-server/dist/types.d.ts.map +1 -1
  335. package/packages/mcp-server/dist/types.js.map +1 -1
  336. package/packages/mcp-server/dist/workflow-tools.d.ts +1 -1
  337. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  338. package/packages/mcp-server/dist/workflow-tools.js +24 -57
  339. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  340. package/packages/mcp-server/package.json +3 -2
  341. package/packages/mcp-server/src/mcp-server.test.ts +138 -0
  342. package/packages/mcp-server/src/server.ts +99 -1
  343. package/packages/mcp-server/src/session-manager.ts +2 -2
  344. package/packages/mcp-server/src/types.ts +7 -18
  345. package/packages/mcp-server/src/workflow-tools.test.ts +75 -2
  346. package/packages/mcp-server/src/workflow-tools.ts +32 -56
  347. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  348. package/packages/native/package.json +1 -1
  349. package/packages/native/tsconfig.tsbuildinfo +1 -1
  350. package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
  351. package/packages/pi-agent-core/dist/agent-loop.js +4 -1
  352. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  353. package/packages/pi-agent-core/dist/agent.d.ts +9 -2
  354. package/packages/pi-agent-core/dist/agent.d.ts.map +1 -1
  355. package/packages/pi-agent-core/dist/agent.js +43 -11
  356. package/packages/pi-agent-core/dist/agent.js.map +1 -1
  357. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  358. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  359. package/packages/pi-agent-core/dist/index.js +2 -0
  360. package/packages/pi-agent-core/dist/index.js.map +1 -1
  361. package/packages/pi-agent-core/dist/token-audit.d.ts +47 -0
  362. package/packages/pi-agent-core/dist/token-audit.d.ts.map +1 -0
  363. package/packages/pi-agent-core/dist/token-audit.js +221 -0
  364. package/packages/pi-agent-core/dist/token-audit.js.map +1 -0
  365. package/packages/pi-agent-core/dist/types.d.ts +31 -0
  366. package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
  367. package/packages/pi-agent-core/dist/types.js.map +1 -1
  368. package/packages/pi-agent-core/package.json +1 -1
  369. package/packages/pi-agent-core/src/agent-loop.test.ts +128 -0
  370. package/packages/pi-agent-core/src/agent-loop.ts +4 -1
  371. package/packages/pi-agent-core/src/agent.ts +52 -11
  372. package/packages/pi-agent-core/src/index.ts +2 -0
  373. package/packages/pi-agent-core/src/token-audit.test.ts +189 -0
  374. package/packages/pi-agent-core/src/token-audit.ts +287 -0
  375. package/packages/pi-agent-core/src/types.ts +26 -10
  376. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  377. package/packages/pi-ai/dist/models/fake-model.d.ts +12 -0
  378. package/packages/pi-ai/dist/models/fake-model.d.ts.map +1 -0
  379. package/packages/pi-ai/dist/models/fake-model.js +27 -0
  380. package/packages/pi-ai/dist/models/fake-model.js.map +1 -0
  381. package/packages/pi-ai/dist/models/index.d.ts.map +1 -1
  382. package/packages/pi-ai/dist/models/index.js +8 -0
  383. package/packages/pi-ai/dist/models/index.js.map +1 -1
  384. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +35 -13
  385. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
  386. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js +21 -11
  387. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js.map +1 -1
  388. package/packages/pi-ai/dist/providers/anthropic.d.ts +7 -0
  389. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  390. package/packages/pi-ai/dist/providers/anthropic.js +9 -7
  391. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  392. package/packages/pi-ai/dist/providers/fake.d.ts +42 -0
  393. package/packages/pi-ai/dist/providers/fake.d.ts.map +1 -0
  394. package/packages/pi-ai/dist/providers/fake.js +319 -0
  395. package/packages/pi-ai/dist/providers/fake.js.map +1 -0
  396. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +23 -14
  397. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -1
  398. package/packages/pi-ai/dist/providers/register-builtins.d.ts.map +1 -1
  399. package/packages/pi-ai/dist/providers/register-builtins.js +24 -0
  400. package/packages/pi-ai/dist/providers/register-builtins.js.map +1 -1
  401. package/packages/pi-ai/dist/types.d.ts +2 -0
  402. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  403. package/packages/pi-ai/dist/types.js.map +1 -1
  404. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js +48 -21
  405. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js.map +1 -1
  406. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js +22 -21
  407. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js.map +1 -1
  408. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js +22 -21
  409. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js.map +1 -1
  410. package/packages/pi-ai/package.json +1 -1
  411. package/packages/pi-ai/src/models/fake-model.ts +30 -0
  412. package/packages/pi-ai/src/models/index.ts +9 -0
  413. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +39 -25
  414. package/packages/pi-ai/src/providers/anthropic-bearer-auth.test.ts +26 -22
  415. package/packages/pi-ai/src/providers/anthropic.ts +22 -9
  416. package/packages/pi-ai/src/providers/fake.ts +376 -0
  417. package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +34 -21
  418. package/packages/pi-ai/src/providers/register-builtins.ts +23 -0
  419. package/packages/pi-ai/src/types.ts +3 -0
  420. package/packages/pi-ai/src/utils/oauth/github-copilot.test.ts +56 -22
  421. package/packages/pi-ai/src/utils/oauth/google-antigravity.test.ts +24 -28
  422. package/packages/pi-ai/src/utils/oauth/google-gemini-cli.test.ts +24 -28
  423. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  424. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +36 -1
  425. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  426. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +30 -1
  427. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  428. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +21 -2
  429. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  430. package/packages/pi-coding-agent/dist/core/agent-session.js +94 -16
  431. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  432. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +78 -0
  433. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  434. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +11 -0
  435. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  436. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +9 -0
  437. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  438. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.d.ts +2 -0
  439. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.d.ts.map +1 -0
  440. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.js +103 -0
  441. package/packages/pi-coding-agent/dist/core/compaction-threshold.test.js.map +1 -0
  442. package/packages/pi-coding-agent/dist/core/db-snapshot.d.ts +15 -0
  443. package/packages/pi-coding-agent/dist/core/db-snapshot.d.ts.map +1 -0
  444. package/packages/pi-coding-agent/dist/core/db-snapshot.js +66 -0
  445. package/packages/pi-coding-agent/dist/core/db-snapshot.js.map +1 -0
  446. package/packages/pi-coding-agent/dist/core/db-snapshot.test.d.ts +2 -0
  447. package/packages/pi-coding-agent/dist/core/db-snapshot.test.d.ts.map +1 -0
  448. package/packages/pi-coding-agent/dist/core/db-snapshot.test.js +24 -0
  449. package/packages/pi-coding-agent/dist/core/db-snapshot.test.js.map +1 -0
  450. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  451. package/packages/pi-coding-agent/dist/core/extensions/loader.js +8 -0
  452. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  453. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +5 -0
  454. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  455. package/packages/pi-coding-agent/dist/core/extensions/runner.js +20 -7
  456. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  457. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js +102 -3
  458. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js.map +1 -1
  459. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +60 -4
  460. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  461. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  462. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js +2 -0
  463. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js.map +1 -1
  464. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  465. package/packages/pi-coding-agent/dist/core/model-registry.js +5 -0
  466. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  467. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.d.ts +2 -0
  468. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.d.ts.map +1 -0
  469. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.js +46 -0
  470. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.js.map +1 -0
  471. package/packages/pi-coding-agent/dist/core/sdk.d.ts +10 -2
  472. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  473. package/packages/pi-coding-agent/dist/core/sdk.js +81 -4
  474. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  475. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +24 -0
  476. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  477. package/packages/pi-coding-agent/dist/core/settings-manager.js +33 -0
  478. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  479. package/packages/pi-coding-agent/dist/core/skill-tool.test.js +22 -0
  480. package/packages/pi-coding-agent/dist/core/skill-tool.test.js.map +1 -1
  481. package/packages/pi-coding-agent/dist/core/slash-commands.d.ts.map +1 -1
  482. package/packages/pi-coding-agent/dist/core/slash-commands.js +1 -0
  483. package/packages/pi-coding-agent/dist/core/slash-commands.js.map +1 -1
  484. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +6 -7
  485. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  486. package/packages/pi-coding-agent/dist/core/system-prompt.js +2 -3
  487. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  488. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js +22 -56
  489. package/packages/pi-coding-agent/dist/core/tools/bash-spawn-windows.test.js.map +1 -1
  490. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +1 -0
  491. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  492. package/packages/pi-coding-agent/dist/core/tools/bash.js +3 -1
  493. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  494. package/packages/pi-coding-agent/dist/core/tools/edit.d.ts +2 -0
  495. package/packages/pi-coding-agent/dist/core/tools/edit.d.ts.map +1 -1
  496. package/packages/pi-coding-agent/dist/core/tools/edit.js +12 -1
  497. package/packages/pi-coding-agent/dist/core/tools/edit.js.map +1 -1
  498. package/packages/pi-coding-agent/dist/core/tools/find.d.ts +2 -0
  499. package/packages/pi-coding-agent/dist/core/tools/find.d.ts.map +1 -1
  500. package/packages/pi-coding-agent/dist/core/tools/find.js +14 -6
  501. package/packages/pi-coding-agent/dist/core/tools/find.js.map +1 -1
  502. package/packages/pi-coding-agent/dist/core/tools/grep.d.ts +2 -0
  503. package/packages/pi-coding-agent/dist/core/tools/grep.d.ts.map +1 -1
  504. package/packages/pi-coding-agent/dist/core/tools/grep.js +12 -3
  505. package/packages/pi-coding-agent/dist/core/tools/grep.js.map +1 -1
  506. package/packages/pi-coding-agent/dist/core/tools/hashline-read.d.ts +2 -0
  507. package/packages/pi-coding-agent/dist/core/tools/hashline-read.d.ts.map +1 -1
  508. package/packages/pi-coding-agent/dist/core/tools/hashline-read.js +3 -1
  509. package/packages/pi-coding-agent/dist/core/tools/hashline-read.js.map +1 -1
  510. package/packages/pi-coding-agent/dist/core/tools/index.d.ts +2 -1
  511. package/packages/pi-coding-agent/dist/core/tools/index.d.ts.map +1 -1
  512. package/packages/pi-coding-agent/dist/core/tools/index.js +1 -0
  513. package/packages/pi-coding-agent/dist/core/tools/index.js.map +1 -1
  514. package/packages/pi-coding-agent/dist/core/tools/ls.d.ts +2 -0
  515. package/packages/pi-coding-agent/dist/core/tools/ls.d.ts.map +1 -1
  516. package/packages/pi-coding-agent/dist/core/tools/ls.js +10 -3
  517. package/packages/pi-coding-agent/dist/core/tools/ls.js.map +1 -1
  518. package/packages/pi-coding-agent/dist/core/tools/read.d.ts +2 -0
  519. package/packages/pi-coding-agent/dist/core/tools/read.d.ts.map +1 -1
  520. package/packages/pi-coding-agent/dist/core/tools/read.js +3 -1
  521. package/packages/pi-coding-agent/dist/core/tools/read.js.map +1 -1
  522. package/packages/pi-coding-agent/dist/core/tools/spawn-shell-windows.test.js +7 -62
  523. package/packages/pi-coding-agent/dist/core/tools/spawn-shell-windows.test.js.map +1 -1
  524. package/packages/pi-coding-agent/dist/core/tools/tool-target-metadata.test.d.ts +2 -0
  525. package/packages/pi-coding-agent/dist/core/tools/tool-target-metadata.test.d.ts.map +1 -0
  526. package/packages/pi-coding-agent/dist/core/tools/tool-target-metadata.test.js +115 -0
  527. package/packages/pi-coding-agent/dist/core/tools/tool-target-metadata.test.js.map +1 -0
  528. package/packages/pi-coding-agent/dist/core/tools/tool-target.d.ts +19 -0
  529. package/packages/pi-coding-agent/dist/core/tools/tool-target.d.ts.map +1 -0
  530. package/packages/pi-coding-agent/dist/core/tools/tool-target.js +20 -0
  531. package/packages/pi-coding-agent/dist/core/tools/tool-target.js.map +1 -0
  532. package/packages/pi-coding-agent/dist/core/tools/write.d.ts +4 -0
  533. package/packages/pi-coding-agent/dist/core/tools/write.d.ts.map +1 -1
  534. package/packages/pi-coding-agent/dist/core/tools/write.js +9 -1
  535. package/packages/pi-coding-agent/dist/core/tools/write.js.map +1 -1
  536. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js +6 -4
  537. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js.map +1 -1
  538. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +154 -14
  539. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  540. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.d.ts +26 -0
  541. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.d.ts.map +1 -0
  542. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.js +112 -0
  543. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.js.map +1 -0
  544. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.test.d.ts +2 -0
  545. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.test.d.ts.map +1 -0
  546. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.test.js +51 -0
  547. package/packages/pi-coding-agent/dist/modes/interactive/components/adaptive-layout.test.js.map +1 -0
  548. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  549. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  550. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
  551. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +10 -9
  552. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
  553. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +3 -0
  554. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
  555. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +11 -0
  556. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
  557. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js +51 -6
  558. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js.map +1 -1
  559. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +22 -0
  560. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  561. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +262 -20
  562. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  563. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  564. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +86 -2
  565. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  566. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js +111 -1
  567. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js.map +1 -1
  568. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +1 -0
  569. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  570. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +1 -0
  571. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  572. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  573. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +14 -1
  574. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  575. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +77 -11
  576. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  577. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts.map +1 -1
  578. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +20 -0
  579. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
  580. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.test.d.ts +2 -0
  581. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.test.d.ts.map +1 -0
  582. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.test.js +79 -0
  583. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.test.js.map +1 -0
  584. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.d.ts +12 -0
  585. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.d.ts.map +1 -1
  586. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.js +13 -0
  587. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.js.map +1 -1
  588. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts +1 -1
  589. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  590. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js +18 -1
  591. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js.map +1 -1
  592. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts.map +1 -1
  593. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js +36 -27
  594. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js.map +1 -1
  595. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.d.ts +11 -0
  596. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.d.ts.map +1 -0
  597. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.js +18 -0
  598. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.js.map +1 -0
  599. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.test.d.ts +2 -0
  600. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.test.d.ts.map +1 -0
  601. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.test.js +48 -0
  602. package/packages/pi-coding-agent/dist/modes/interactive/tui-mode.test.js.map +1 -0
  603. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +1 -1
  604. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
  605. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +1 -512
  606. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
  607. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js +3 -7
  608. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
  609. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage-safety-guard.test.d.ts +2 -0
  610. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage-safety-guard.test.d.ts.map +1 -0
  611. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage-safety-guard.test.js +28 -0
  612. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage-safety-guard.test.js.map +1 -0
  613. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.d.ts.map +1 -1
  614. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.js +3 -2
  615. package/packages/pi-coding-agent/dist/resources/extensions/memory/storage.js.map +1 -1
  616. package/packages/pi-coding-agent/package.json +2 -1
  617. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +40 -1
  618. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +40 -1
  619. package/packages/pi-coding-agent/src/core/agent-session.ts +102 -16
  620. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +91 -0
  621. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +18 -0
  622. package/packages/pi-coding-agent/src/core/compaction-threshold.test.ts +121 -0
  623. package/packages/pi-coding-agent/src/core/db-snapshot.test.ts +32 -0
  624. package/packages/pi-coding-agent/src/core/db-snapshot.ts +66 -0
  625. package/packages/pi-coding-agent/src/core/extensions/loader.ts +10 -0
  626. package/packages/pi-coding-agent/src/core/extensions/runner.test.ts +113 -3
  627. package/packages/pi-coding-agent/src/core/extensions/runner.ts +24 -6
  628. package/packages/pi-coding-agent/src/core/extensions/types.ts +63 -2
  629. package/packages/pi-coding-agent/src/core/hooks-runner.test.ts +2 -0
  630. package/packages/pi-coding-agent/src/core/model-registry.ts +4 -0
  631. package/packages/pi-coding-agent/src/core/sdk-tool-filter.test.ts +60 -0
  632. package/packages/pi-coding-agent/src/core/sdk.ts +92 -4
  633. package/packages/pi-coding-agent/src/core/settings-manager.ts +51 -1
  634. package/packages/pi-coding-agent/src/core/skill-tool.test.ts +28 -0
  635. package/packages/pi-coding-agent/src/core/slash-commands.ts +1 -0
  636. package/packages/pi-coding-agent/src/core/system-prompt.ts +8 -10
  637. package/packages/pi-coding-agent/src/core/tools/bash-spawn-windows.test.ts +22 -66
  638. package/packages/pi-coding-agent/src/core/tools/bash.ts +4 -1
  639. package/packages/pi-coding-agent/src/core/tools/edit.ts +13 -1
  640. package/packages/pi-coding-agent/src/core/tools/find.ts +15 -6
  641. package/packages/pi-coding-agent/src/core/tools/grep.ts +13 -3
  642. package/packages/pi-coding-agent/src/core/tools/hashline-read.ts +4 -1
  643. package/packages/pi-coding-agent/src/core/tools/index.ts +8 -0
  644. package/packages/pi-coding-agent/src/core/tools/ls.ts +11 -3
  645. package/packages/pi-coding-agent/src/core/tools/read.ts +4 -1
  646. package/packages/pi-coding-agent/src/core/tools/spawn-shell-windows.test.ts +11 -72
  647. package/packages/pi-coding-agent/src/core/tools/tool-target-metadata.test.ts +127 -0
  648. package/packages/pi-coding-agent/src/core/tools/tool-target.ts +48 -0
  649. package/packages/pi-coding-agent/src/core/tools/write.ts +14 -2
  650. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.ts +7 -5
  651. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +217 -16
  652. package/packages/pi-coding-agent/src/modes/interactive/components/adaptive-layout.test.ts +59 -0
  653. package/packages/pi-coding-agent/src/modes/interactive/components/adaptive-layout.ts +160 -0
  654. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +1 -0
  655. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +10 -9
  656. package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +15 -0
  657. package/packages/pi-coding-agent/src/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.ts +71 -9
  658. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +299 -20
  659. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.test.ts +118 -1
  660. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +108 -3
  661. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +1 -0
  662. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +1 -1
  663. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +101 -11
  664. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.test.ts +95 -0
  665. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +24 -1
  666. package/packages/pi-coding-agent/src/modes/interactive/theme/theme-schema.ts +13 -0
  667. package/packages/pi-coding-agent/src/modes/interactive/theme/theme.ts +32 -2
  668. package/packages/pi-coding-agent/src/modes/interactive/theme/themes.ts +36 -27
  669. package/packages/pi-coding-agent/src/modes/interactive/tui-mode.test.ts +65 -0
  670. package/packages/pi-coding-agent/src/modes/interactive/tui-mode.ts +29 -0
  671. package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +1 -1
  672. package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +3 -336
  673. package/packages/pi-coding-agent/src/resources/extensions/memory/storage-safety-guard.test.ts +31 -0
  674. package/packages/pi-coding-agent/src/resources/extensions/memory/storage.ts +3 -2
  675. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  676. package/packages/pi-tui/dist/__tests__/style.test.d.ts +2 -0
  677. package/packages/pi-tui/dist/__tests__/style.test.d.ts.map +1 -0
  678. package/packages/pi-tui/dist/__tests__/style.test.js +63 -0
  679. package/packages/pi-tui/dist/__tests__/style.test.js.map +1 -0
  680. package/packages/pi-tui/dist/__tests__/tui.test.js +24 -3
  681. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  682. package/packages/pi-tui/dist/index.d.ts +1 -0
  683. package/packages/pi-tui/dist/index.d.ts.map +1 -1
  684. package/packages/pi-tui/dist/index.js +2 -0
  685. package/packages/pi-tui/dist/index.js.map +1 -1
  686. package/packages/pi-tui/dist/style.d.ts +41 -0
  687. package/packages/pi-tui/dist/style.d.ts.map +1 -0
  688. package/packages/pi-tui/dist/style.js +158 -0
  689. package/packages/pi-tui/dist/style.js.map +1 -0
  690. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  691. package/packages/pi-tui/dist/tui.js +19 -8
  692. package/packages/pi-tui/dist/tui.js.map +1 -1
  693. package/packages/pi-tui/package.json +1 -1
  694. package/packages/pi-tui/src/__tests__/style.test.ts +76 -0
  695. package/packages/pi-tui/src/__tests__/tui.test.ts +29 -3
  696. package/packages/pi-tui/src/index.ts +9 -0
  697. package/packages/pi-tui/src/style.ts +225 -0
  698. package/packages/pi-tui/src/tui.ts +21 -8
  699. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  700. package/packages/rpc-client/README.md +10 -3
  701. package/packages/rpc-client/dist/index.d.ts +1 -1
  702. package/packages/rpc-client/dist/index.d.ts.map +1 -1
  703. package/packages/rpc-client/dist/index.js.map +1 -1
  704. package/packages/rpc-client/dist/rpc-client.d.ts +2 -6
  705. package/packages/rpc-client/dist/rpc-client.d.ts.map +1 -1
  706. package/packages/rpc-client/dist/rpc-client.js.map +1 -1
  707. package/packages/rpc-client/dist/rpc-types.d.ts +1 -565
  708. package/packages/rpc-client/dist/rpc-types.d.ts.map +1 -1
  709. package/packages/rpc-client/dist/rpc-types.js +3 -11
  710. package/packages/rpc-client/dist/rpc-types.js.map +1 -1
  711. package/packages/rpc-client/package.json +4 -1
  712. package/packages/rpc-client/src/index.ts +1 -1
  713. package/packages/rpc-client/src/rpc-client.ts +3 -6
  714. package/packages/rpc-client/src/rpc-types.ts +3 -398
  715. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  716. package/pkg/dist/modes/interactive/theme/theme-schema.d.ts +12 -0
  717. package/pkg/dist/modes/interactive/theme/theme-schema.d.ts.map +1 -1
  718. package/pkg/dist/modes/interactive/theme/theme-schema.js +13 -0
  719. package/pkg/dist/modes/interactive/theme/theme-schema.js.map +1 -1
  720. package/pkg/dist/modes/interactive/theme/theme.d.ts +1 -1
  721. package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  722. package/pkg/dist/modes/interactive/theme/theme.js +18 -1
  723. package/pkg/dist/modes/interactive/theme/theme.js.map +1 -1
  724. package/pkg/dist/modes/interactive/theme/themes.d.ts.map +1 -1
  725. package/pkg/dist/modes/interactive/theme/themes.js +36 -27
  726. package/pkg/dist/modes/interactive/theme/themes.js.map +1 -1
  727. package/pkg/package.json +1 -1
  728. package/src/resources/GSD-WORKFLOW.md +2 -2
  729. package/src/resources/extensions/claude-code-cli/readiness.ts +25 -7
  730. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +42 -3
  731. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +67 -0
  732. package/src/resources/extensions/github-sync/sync.ts +8 -1
  733. package/src/resources/extensions/github-sync/templates.ts +93 -88
  734. package/src/resources/extensions/github-sync/tests/inline-code.test.ts +66 -0
  735. package/src/resources/extensions/github-sync/tests/sync-source.test.ts +6 -18
  736. package/src/resources/extensions/github-sync/tests/templates.test.ts +10 -2
  737. package/src/resources/extensions/gsd/auto/contracts.ts +19 -2
  738. package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +72 -0
  739. package/src/resources/extensions/gsd/auto/loop-deps.ts +11 -11
  740. package/src/resources/extensions/gsd/auto/loop.ts +540 -602
  741. package/src/resources/extensions/gsd/auto/orchestrator.ts +52 -4
  742. package/src/resources/extensions/gsd/auto/phases.ts +427 -220
  743. package/src/resources/extensions/gsd/auto/resolve.ts +42 -1
  744. package/src/resources/extensions/gsd/auto/run-unit.ts +52 -44
  745. package/src/resources/extensions/gsd/auto/session.ts +9 -1
  746. package/src/resources/extensions/gsd/auto/types.ts +3 -0
  747. package/src/resources/extensions/gsd/auto/unit-runner-events.ts +15 -0
  748. package/src/resources/extensions/gsd/auto/workflow-custom-engine-dispatch-outcome.ts +28 -0
  749. package/src/resources/extensions/gsd/auto/workflow-custom-engine-iteration.ts +52 -0
  750. package/src/resources/extensions/gsd/auto/workflow-custom-engine-reconcile-outcome.ts +58 -0
  751. package/src/resources/extensions/gsd/auto/workflow-custom-engine-reconcile.ts +71 -0
  752. package/src/resources/extensions/gsd/auto/workflow-custom-engine-retry.ts +90 -0
  753. package/src/resources/extensions/gsd/auto/workflow-custom-engine-verify-outcome.ts +50 -0
  754. package/src/resources/extensions/gsd/auto/workflow-dispatch-claim.ts +159 -0
  755. package/src/resources/extensions/gsd/auto/workflow-dispatch-ledger.ts +45 -0
  756. package/src/resources/extensions/gsd/auto/workflow-iteration-completion.ts +26 -0
  757. package/src/resources/extensions/gsd/auto/workflow-journal-reporter.ts +33 -0
  758. package/src/resources/extensions/gsd/auto/workflow-kernel.ts +520 -0
  759. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +58 -0
  760. package/src/resources/extensions/gsd/auto/workflow-phase-reporter.ts +22 -0
  761. package/src/resources/extensions/gsd/auto/workflow-session-lock.ts +68 -0
  762. package/src/resources/extensions/gsd/auto/workflow-sidecar-iteration.ts +46 -0
  763. package/src/resources/extensions/gsd/auto/workflow-sidecar-queue.ts +46 -0
  764. package/src/resources/extensions/gsd/auto/workflow-turn-reporter.ts +68 -0
  765. package/src/resources/extensions/gsd/auto/workflow-unit-dispatch.ts +89 -0
  766. package/src/resources/extensions/gsd/auto/workflow-worker-heartbeat.ts +51 -0
  767. package/src/resources/extensions/gsd/auto-dashboard.ts +81 -8
  768. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -34
  769. package/src/resources/extensions/gsd/auto-dispatch.ts +33 -0
  770. package/src/resources/extensions/gsd/auto-post-unit.ts +134 -90
  771. package/src/resources/extensions/gsd/auto-prompts.ts +398 -19
  772. package/src/resources/extensions/gsd/auto-recovery.ts +207 -7
  773. package/src/resources/extensions/gsd/auto-start.ts +326 -25
  774. package/src/resources/extensions/gsd/auto-supervisor.ts +7 -0
  775. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +2 -2
  776. package/src/resources/extensions/gsd/auto-verification.ts +5 -1
  777. package/src/resources/extensions/gsd/auto-worktree.ts +344 -396
  778. package/src/resources/extensions/gsd/auto.ts +336 -89
  779. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +162 -11
  780. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +56 -37
  781. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +36 -10
  782. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +32 -19
  783. package/src/resources/extensions/gsd/bootstrap/journal-tools.ts +5 -1
  784. package/src/resources/extensions/gsd/bootstrap/memory-tools.ts +7 -4
  785. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +6 -3
  786. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +1 -1
  787. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +406 -57
  788. package/src/resources/extensions/gsd/bootstrap/sanitize-complete-milestone.ts +4 -0
  789. package/src/resources/extensions/gsd/bootstrap/system-context.ts +90 -22
  790. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +153 -2
  791. package/src/resources/extensions/gsd/clean-root-preflight.ts +72 -9
  792. package/src/resources/extensions/gsd/commands/dispatcher.ts +6 -0
  793. package/src/resources/extensions/gsd/commands/handlers/ops.ts +4 -2
  794. package/src/resources/extensions/gsd/commands-config.ts +1 -1
  795. package/src/resources/extensions/gsd/commands-eval-review.ts +2 -2
  796. package/src/resources/extensions/gsd/commands-extract-learnings.ts +17 -12
  797. package/src/resources/extensions/gsd/commands-handlers.ts +34 -15
  798. package/src/resources/extensions/gsd/commands-ship.ts +24 -51
  799. package/src/resources/extensions/gsd/commands-workflow-templates.ts +13 -0
  800. package/src/resources/extensions/gsd/component-loader.ts +5 -11
  801. package/src/resources/extensions/gsd/context-budget.ts +44 -2
  802. package/src/resources/extensions/gsd/crash-recovery.ts +67 -10
  803. package/src/resources/extensions/gsd/custom-workflow-engine.ts +29 -0
  804. package/src/resources/extensions/gsd/dashboard-overlay.ts +1 -1
  805. package/src/resources/extensions/gsd/db/unit-dispatches.ts +107 -0
  806. package/src/resources/extensions/gsd/db-adapter.ts +75 -0
  807. package/src/resources/extensions/gsd/db-base-schema.ts +385 -0
  808. package/src/resources/extensions/gsd/db-connection-cache.ts +45 -0
  809. package/src/resources/extensions/gsd/db-coordination-schema.ts +109 -0
  810. package/src/resources/extensions/gsd/db-decision-requirement-rows.ts +77 -0
  811. package/src/resources/extensions/gsd/db-gate-rows.ts +19 -0
  812. package/src/resources/extensions/gsd/db-lightweight-query-rows.ts +50 -0
  813. package/src/resources/extensions/gsd/db-memory-fts-schema.ts +66 -0
  814. package/src/resources/extensions/gsd/db-migration-backup.ts +34 -0
  815. package/src/resources/extensions/gsd/db-migration-steps.ts +459 -0
  816. package/src/resources/extensions/gsd/db-milestone-artifact-rows.ts +70 -0
  817. package/src/resources/extensions/gsd/db-open-state.ts +47 -0
  818. package/src/resources/extensions/gsd/db-provider.ts +148 -0
  819. package/src/resources/extensions/gsd/db-runtime-kv-schema.ts +30 -0
  820. package/src/resources/extensions/gsd/db-schema-metadata.ts +33 -0
  821. package/src/resources/extensions/gsd/db-task-slice-rows.ts +146 -0
  822. package/src/resources/extensions/gsd/db-transaction.ts +76 -0
  823. package/src/resources/extensions/gsd/db-verification-evidence-rows.ts +14 -0
  824. package/src/resources/extensions/gsd/db-verification-evidence-schema.ts +22 -0
  825. package/src/resources/extensions/gsd/detection.ts +128 -0
  826. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +3 -0
  827. package/src/resources/extensions/gsd/escalation.ts +3 -1
  828. package/src/resources/extensions/gsd/git-service.ts +87 -10
  829. package/src/resources/extensions/gsd/graph.ts +12 -5
  830. package/src/resources/extensions/gsd/gsd-db.ts +399 -1677
  831. package/src/resources/extensions/gsd/guided-flow.ts +175 -55
  832. package/src/resources/extensions/gsd/health-widget.ts +3 -0
  833. package/src/resources/extensions/gsd/init-wizard.ts +5 -1
  834. package/src/resources/extensions/gsd/legacy-telemetry.ts +99 -0
  835. package/src/resources/extensions/gsd/markdown-renderer.ts +4 -1
  836. package/src/resources/extensions/gsd/memory-store.ts +77 -12
  837. package/src/resources/extensions/gsd/migrate/command.ts +47 -1
  838. package/src/resources/extensions/gsd/migration-auto-check.ts +129 -0
  839. package/src/resources/extensions/gsd/model-router.ts +10 -6
  840. package/src/resources/extensions/gsd/native-git-bridge.ts +39 -6
  841. package/src/resources/extensions/gsd/notification-widget.ts +25 -4
  842. package/src/resources/extensions/gsd/orphan-stash-audit.ts +117 -0
  843. package/src/resources/extensions/gsd/parallel-orchestrator.ts +13 -3
  844. package/src/resources/extensions/gsd/planning-path-scope.ts +35 -0
  845. package/src/resources/extensions/gsd/post-execution-checks.ts +4 -1
  846. package/src/resources/extensions/gsd/pr-evidence.ts +182 -0
  847. package/src/resources/extensions/gsd/pre-execution-checks.ts +27 -1
  848. package/src/resources/extensions/gsd/preferences-types.ts +1 -1
  849. package/src/resources/extensions/gsd/process-task-path.ts +81 -0
  850. package/src/resources/extensions/gsd/prompt-loader.ts +36 -7
  851. package/src/resources/extensions/gsd/prompts/complete-milestone.md +41 -35
  852. package/src/resources/extensions/gsd/prompts/complete-slice.md +23 -34
  853. package/src/resources/extensions/gsd/prompts/discuss-headless.md +50 -96
  854. package/src/resources/extensions/gsd/prompts/discuss.md +81 -181
  855. package/src/resources/extensions/gsd/prompts/execute-task.md +42 -67
  856. package/src/resources/extensions/gsd/prompts/forensics.md +41 -84
  857. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +29 -39
  858. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +30 -65
  859. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +25 -52
  860. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +36 -36
  861. package/src/resources/extensions/gsd/prompts/guided-research-project.md +20 -38
  862. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +1 -1
  863. package/src/resources/extensions/gsd/prompts/plan-milestone.md +47 -59
  864. package/src/resources/extensions/gsd/prompts/plan-slice.md +25 -87
  865. package/src/resources/extensions/gsd/prompts/queue.md +46 -53
  866. package/src/resources/extensions/gsd/prompts/quick-task.md +1 -5
  867. package/src/resources/extensions/gsd/prompts/refine-slice.md +23 -23
  868. package/src/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  869. package/src/resources/extensions/gsd/prompts/research-slice.md +23 -23
  870. package/src/resources/extensions/gsd/prompts/rethink.md +10 -10
  871. package/src/resources/extensions/gsd/prompts/system.md +65 -107
  872. package/src/resources/extensions/gsd/prompts/triage-captures.md +15 -15
  873. package/src/resources/extensions/gsd/prompts/validate-milestone.md +24 -24
  874. package/src/resources/extensions/gsd/prompts/worktree-merge.md +35 -35
  875. package/src/resources/extensions/gsd/quick.ts +37 -2
  876. package/src/resources/extensions/gsd/recovery-classification.ts +122 -0
  877. package/src/resources/extensions/gsd/safety/evidence-collector.ts +11 -2
  878. package/src/resources/extensions/gsd/slice-cadence.ts +49 -2
  879. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +23 -9
  880. package/src/resources/extensions/gsd/state-reconciliation.ts +57 -0
  881. package/src/resources/extensions/gsd/state.ts +6 -3
  882. package/src/resources/extensions/gsd/tests/agent-end-retry.test.ts +59 -89
  883. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +47 -172
  884. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +0 -35
  885. package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +7 -1
  886. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +133 -9
  887. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +660 -40
  888. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +80 -59
  889. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +119 -2
  890. package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +3 -47
  891. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +76 -18
  892. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +175 -11
  893. package/src/resources/extensions/gsd/tests/auto-pr-bugs.test.ts +54 -76
  894. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +67 -26
  895. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +184 -2
  896. package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +32 -30
  897. package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +32 -128
  898. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +20 -54
  899. package/src/resources/extensions/gsd/tests/auto-start-cold-db-bootstrap.test.ts +20 -30
  900. package/src/resources/extensions/gsd/tests/auto-start-index-lock.test.ts +17 -29
  901. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +142 -0
  902. package/src/resources/extensions/gsd/tests/auto-start-time-persistence.test.ts +21 -39
  903. package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +15 -24
  904. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +44 -29
  905. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +39 -51
  906. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +159 -213
  907. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +15 -32
  908. package/src/resources/extensions/gsd/tests/browser-teardown.test.ts +0 -41
  909. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +97 -2
  910. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +34 -27
  911. package/src/resources/extensions/gsd/tests/cmux.test.ts +51 -53
  912. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +39 -61
  913. package/src/resources/extensions/gsd/tests/commands-config.test.ts +26 -19
  914. package/src/resources/extensions/gsd/tests/commands-eval-review.test.ts +2 -2
  915. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +9 -0
  916. package/src/resources/extensions/gsd/tests/commands-ship-eval-warn.test.ts +3 -2
  917. package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +14 -1
  918. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +31 -0
  919. package/src/resources/extensions/gsd/tests/complete-milestone-prompt-rendering.test.ts +47 -0
  920. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +159 -5
  921. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +3 -2
  922. package/src/resources/extensions/gsd/tests/complete-task-normalize-lists.test.ts +29 -33
  923. package/src/resources/extensions/gsd/tests/completed-units-metrics-sync.test.ts +45 -108
  924. package/src/resources/extensions/gsd/tests/component-loader.test.ts +2 -9
  925. package/src/resources/extensions/gsd/tests/context-budget.test.ts +10 -1
  926. package/src/resources/extensions/gsd/tests/context-store.test.ts +7 -1
  927. package/src/resources/extensions/gsd/tests/crash-handler-secondary.test.ts +90 -31
  928. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +22 -0
  929. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +162 -10
  930. package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +139 -0
  931. package/src/resources/extensions/gsd/tests/custom-workflow-engine.test.ts +50 -0
  932. package/src/resources/extensions/gsd/tests/cwd-fallback-hardening.test.ts +138 -0
  933. package/src/resources/extensions/gsd/tests/dashboard-custom-engine.test.ts +4 -68
  934. package/src/resources/extensions/gsd/tests/db-adapter.test.ts +82 -0
  935. package/src/resources/extensions/gsd/tests/db-base-schema.test.ts +62 -0
  936. package/src/resources/extensions/gsd/tests/db-connection-cache.test.ts +60 -0
  937. package/src/resources/extensions/gsd/tests/db-coordination-schema.test.ts +39 -0
  938. package/src/resources/extensions/gsd/tests/db-decision-requirement-rows.test.ts +135 -0
  939. package/src/resources/extensions/gsd/tests/db-gate-rows.test.ts +53 -0
  940. package/src/resources/extensions/gsd/tests/db-lightweight-query-rows.test.ts +45 -0
  941. package/src/resources/extensions/gsd/tests/db-memory-fts-schema.test.ts +86 -0
  942. package/src/resources/extensions/gsd/tests/db-migration-backup.test.ts +105 -0
  943. package/src/resources/extensions/gsd/tests/db-migration-steps.integration.test.ts +428 -0
  944. package/src/resources/extensions/gsd/tests/db-migration-steps.test.ts +159 -0
  945. package/src/resources/extensions/gsd/tests/db-milestone-artifact-rows.test.ts +53 -0
  946. package/src/resources/extensions/gsd/tests/db-open-state.test.ts +56 -0
  947. package/src/resources/extensions/gsd/tests/db-provider.test.ts +105 -0
  948. package/src/resources/extensions/gsd/tests/db-runtime-kv-schema.test.ts +37 -0
  949. package/src/resources/extensions/gsd/tests/db-schema-metadata.test.ts +115 -0
  950. package/src/resources/extensions/gsd/tests/db-task-slice-rows.test.ts +128 -0
  951. package/src/resources/extensions/gsd/tests/db-transaction.test.ts +110 -0
  952. package/src/resources/extensions/gsd/tests/db-verification-evidence-schema.test.ts +76 -0
  953. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +6 -11
  954. package/src/resources/extensions/gsd/tests/deferred-milestone-dir-4996.test.ts +14 -65
  955. package/src/resources/extensions/gsd/tests/detection.test.ts +140 -0
  956. package/src/resources/extensions/gsd/tests/discuss-headless-rendering.test.ts +37 -0
  957. package/src/resources/extensions/gsd/tests/discuss-tool-scoping.test.ts +44 -37
  958. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +58 -40
  959. package/src/resources/extensions/gsd/tests/dispatch-guard-closed-status.test.ts +25 -15
  960. package/src/resources/extensions/gsd/tests/dispatch-rule-coverage.test.ts +313 -0
  961. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +35 -17
  962. package/src/resources/extensions/gsd/tests/error-success-mask.test.ts +16 -21
  963. package/src/resources/extensions/gsd/tests/est-annotation-timeout.test.ts +15 -82
  964. package/src/resources/extensions/gsd/tests/exec-history.test.ts +15 -0
  965. package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +65 -0
  966. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +62 -0
  967. package/src/resources/extensions/gsd/tests/fast-forward-reused-milestone-branch.test.ts +219 -0
  968. package/src/resources/extensions/gsd/tests/finalize-survivor-branch.test.ts +151 -0
  969. package/src/resources/extensions/gsd/tests/fixtures/pr-body/commands-ship-basic.md +52 -0
  970. package/src/resources/extensions/gsd/tests/fixtures/pr-body/commands-ship-empty-optionals.md +42 -0
  971. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-no-blockers.md +55 -0
  972. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-with-blockers.md +60 -0
  973. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +2 -20
  974. package/src/resources/extensions/gsd/tests/forensics-prompt-rendering.test.ts +36 -0
  975. package/src/resources/extensions/gsd/tests/frontmatter-parse-noise.test.ts +18 -26
  976. package/src/resources/extensions/gsd/tests/graph-operations.test.ts +10 -0
  977. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +44 -0
  978. package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +43 -0
  979. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +41 -0
  980. package/src/resources/extensions/gsd/tests/guided-discuss-requirements-prompt-rendering.test.ts +45 -0
  981. package/src/resources/extensions/gsd/tests/has-pending-deep-stage.test.ts +33 -1
  982. package/src/resources/extensions/gsd/tests/init-skip-git.test.ts +9 -12
  983. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +77 -0
  984. package/src/resources/extensions/gsd/tests/integration/commands-eval-review.integration.test.ts +4 -2
  985. package/src/resources/extensions/gsd/tests/integration/git-locale.test.ts +31 -20
  986. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +66 -0
  987. package/src/resources/extensions/gsd/tests/integration/milestone-transition-worktree.test.ts +0 -47
  988. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +5 -3
  989. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +60 -202
  990. package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +13 -56
  991. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +248 -10
  992. package/src/resources/extensions/gsd/tests/journal-query-tool.test.ts +32 -0
  993. package/src/resources/extensions/gsd/tests/knowledge.test.ts +47 -0
  994. package/src/resources/extensions/gsd/tests/lazy-pi-tui-import.test.ts +44 -6
  995. package/src/resources/extensions/gsd/tests/legacy-component-format-telemetry.test.ts +62 -0
  996. package/src/resources/extensions/gsd/tests/legacy-telemetry.test.ts +144 -0
  997. package/src/resources/extensions/gsd/tests/memory-decay-factor.test.ts +90 -0
  998. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +53 -43
  999. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +37 -7
  1000. package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +48 -0
  1001. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +127 -0
  1002. package/src/resources/extensions/gsd/tests/milestone-merge-stash-restore.test.ts +267 -0
  1003. package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +88 -98
  1004. package/src/resources/extensions/gsd/tests/model-router.test.ts +33 -12
  1005. package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +70 -278
  1006. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +34 -2
  1007. package/src/resources/extensions/gsd/tests/needs-remediation-revalidation.test.ts +37 -30
  1008. package/src/resources/extensions/gsd/tests/note-captures-executed.test.ts +32 -28
  1009. package/src/resources/extensions/gsd/tests/notification-store.test.ts +8 -0
  1010. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +40 -1
  1011. package/src/resources/extensions/gsd/tests/originalbase-path-comparison.test.ts +44 -9
  1012. package/src/resources/extensions/gsd/tests/orphan-merge-bootstrap.test.ts +144 -0
  1013. package/src/resources/extensions/gsd/tests/orphan-stash-audit.test.ts +201 -0
  1014. package/src/resources/extensions/gsd/tests/parallel-orchestrator-fast-forward.test.ts +113 -0
  1015. package/src/resources/extensions/gsd/tests/phantom-ghost-detection.test.ts +24 -37
  1016. package/src/resources/extensions/gsd/tests/phantom-milestone-default-queued.test.ts +9 -24
  1017. package/src/resources/extensions/gsd/tests/phases-merge-error-stops-auto.test.ts +95 -75
  1018. package/src/resources/extensions/gsd/tests/plan-milestone-rendering.test.ts +45 -0
  1019. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +65 -16
  1020. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +50 -0
  1021. package/src/resources/extensions/gsd/tests/plan-task.test.ts +21 -0
  1022. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +36 -22
  1023. package/src/resources/extensions/gsd/tests/pr-evidence-equivalence.test.ts +102 -0
  1024. package/src/resources/extensions/gsd/tests/pr-evidence-hardening.test.ts +165 -0
  1025. package/src/resources/extensions/gsd/tests/pr-evidence.test.ts +79 -0
  1026. package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +36 -30
  1027. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +45 -5
  1028. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +74 -4
  1029. package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +20 -22
  1030. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +130 -32
  1031. package/src/resources/extensions/gsd/tests/process-task-path.test.ts +51 -0
  1032. package/src/resources/extensions/gsd/tests/project-root-cwd-crash.test.ts +18 -36
  1033. package/src/resources/extensions/gsd/tests/projection-no-plan-overwrite.test.ts +35 -73
  1034. package/src/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +76 -138
  1035. package/src/resources/extensions/gsd/tests/prompt-duplication-cuts.test.ts +230 -0
  1036. package/src/resources/extensions/gsd/tests/prompt-path-audit.test.ts +40 -0
  1037. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +70 -91
  1038. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +59 -161
  1039. package/src/resources/extensions/gsd/tests/query-tools-db-open.test.ts +33 -29
  1040. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +22 -196
  1041. package/src/resources/extensions/gsd/tests/queue-prompt-rendering.test.ts +37 -0
  1042. package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +23 -93
  1043. package/src/resources/extensions/gsd/tests/quick-external-gsd.test.ts +40 -0
  1044. package/src/resources/extensions/gsd/tests/quick-turn-end-cleanup.test.ts +50 -79
  1045. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +27 -13
  1046. package/src/resources/extensions/gsd/tests/remote-questions.test.ts +151 -251
  1047. package/src/resources/extensions/gsd/tests/resource-loader-import-path.test.ts +41 -29
  1048. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +58 -69
  1049. package/src/resources/extensions/gsd/tests/resume-dispatch-worktree.test.ts +32 -164
  1050. package/src/resources/extensions/gsd/tests/right-sized-workflow-prompts.test.ts +192 -0
  1051. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +57 -41
  1052. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +91 -0
  1053. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +29 -0
  1054. package/src/resources/extensions/gsd/tests/schema-v27-v28-sequence.test.ts +156 -0
  1055. package/src/resources/extensions/gsd/tests/select-resumable-milestone.test.ts +96 -0
  1056. package/src/resources/extensions/gsd/tests/session-model-override.test.ts +14 -9
  1057. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +77 -0
  1058. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +222 -0
  1059. package/src/resources/extensions/gsd/tests/show-config-command.test.ts +44 -42
  1060. package/src/resources/extensions/gsd/tests/signal-handlers.test.ts +27 -0
  1061. package/src/resources/extensions/gsd/tests/skip-slice-state-rebuild.test.ts +56 -24
  1062. package/src/resources/extensions/gsd/tests/skipped-validation-db-atomicity.test.ts +51 -11
  1063. package/src/resources/extensions/gsd/tests/slice-cadence.test.ts +23 -0
  1064. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +66 -50
  1065. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +68 -107
  1066. package/src/resources/extensions/gsd/tests/slice-sequence-insert.test.ts +115 -42
  1067. package/src/resources/extensions/gsd/tests/smart-entry-complete.test.ts +21 -59
  1068. package/src/resources/extensions/gsd/tests/smart-entry-draft.test.ts +25 -116
  1069. package/src/resources/extensions/gsd/tests/sqlite-unavailable-gate.test.ts +21 -57
  1070. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +29 -76
  1071. package/src/resources/extensions/gsd/tests/stale-lockfile-recovery.test.ts +33 -24
  1072. package/src/resources/extensions/gsd/tests/stale-slice-rows.test.ts +39 -30
  1073. package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +49 -1
  1074. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +101 -2
  1075. package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -4
  1076. package/src/resources/extensions/gsd/tests/status-db-open.test.ts +35 -40
  1077. package/src/resources/extensions/gsd/tests/stop-auto-merge-back.test.ts +48 -46
  1078. package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +14 -102
  1079. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +78 -232
  1080. package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +32 -35
  1081. package/src/resources/extensions/gsd/tests/system-context-memory.test.ts +112 -0
  1082. package/src/resources/extensions/gsd/tests/system-context-message-routing.test.ts +7 -9
  1083. package/src/resources/extensions/gsd/tests/token-profile.test.ts +84 -309
  1084. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +291 -0
  1085. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +32 -9
  1086. package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +134 -341
  1087. package/src/resources/extensions/gsd/tests/tui-header-lifecycle.test.ts +210 -0
  1088. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +140 -8
  1089. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +8 -25
  1090. package/src/resources/extensions/gsd/tests/unit-dispatches.test.ts +80 -1
  1091. package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +37 -0
  1092. package/src/resources/extensions/gsd/tests/unstructured-continue-context-injection.test.ts +5 -99
  1093. package/src/resources/extensions/gsd/tests/uok-kernel-path.test.ts +12 -0
  1094. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +43 -36
  1095. package/src/resources/extensions/gsd/tests/visualizer-data.test.ts +84 -444
  1096. package/src/resources/extensions/gsd/tests/workflow-custom-engine-dispatch-outcome.test.ts +55 -0
  1097. package/src/resources/extensions/gsd/tests/workflow-custom-engine-iteration.test.ts +93 -0
  1098. package/src/resources/extensions/gsd/tests/workflow-custom-engine-reconcile-outcome.test.ts +108 -0
  1099. package/src/resources/extensions/gsd/tests/workflow-custom-engine-reconcile.test.ts +146 -0
  1100. package/src/resources/extensions/gsd/tests/workflow-custom-engine-retry.test.ts +136 -0
  1101. package/src/resources/extensions/gsd/tests/workflow-custom-engine-verify-outcome.test.ts +95 -0
  1102. package/src/resources/extensions/gsd/tests/workflow-dispatch-claim.test.ts +300 -0
  1103. package/src/resources/extensions/gsd/tests/workflow-dispatch-ledger.test.ts +82 -0
  1104. package/src/resources/extensions/gsd/tests/workflow-iteration-completion.test.ts +44 -0
  1105. package/src/resources/extensions/gsd/tests/workflow-journal-reporter.test.ts +49 -0
  1106. package/src/resources/extensions/gsd/tests/workflow-kernel.test.ts +607 -0
  1107. package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +44 -189
  1108. package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +20 -4
  1109. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -57
  1110. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +71 -0
  1111. package/src/resources/extensions/gsd/tests/workflow-phase-reporter.test.ts +40 -0
  1112. package/src/resources/extensions/gsd/tests/workflow-protocol-excerpt.test.ts +99 -0
  1113. package/src/resources/extensions/gsd/tests/workflow-session-lock.test.ts +135 -0
  1114. package/src/resources/extensions/gsd/tests/workflow-sidecar-iteration.test.ts +110 -0
  1115. package/src/resources/extensions/gsd/tests/workflow-sidecar-queue.test.ts +116 -0
  1116. package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +21 -0
  1117. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +35 -0
  1118. package/src/resources/extensions/gsd/tests/workflow-turn-reporter.test.ts +87 -0
  1119. package/src/resources/extensions/gsd/tests/workflow-unit-dispatch.test.ts +160 -0
  1120. package/src/resources/extensions/gsd/tests/workflow-worker-heartbeat.test.ts +154 -0
  1121. package/src/resources/extensions/gsd/tests/working-output-messages.test.ts +93 -0
  1122. package/src/resources/extensions/gsd/tests/worktree-db-same-file.test.ts +21 -44
  1123. package/src/resources/extensions/gsd/tests/worktree-expected-warnings.test.ts +27 -26
  1124. package/src/resources/extensions/gsd/tests/worktree-health-dispatch.test.ts +38 -6
  1125. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +50 -31
  1126. package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +434 -0
  1127. package/src/resources/extensions/gsd/tests/worktree-main-branch.test.ts +20 -18
  1128. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +7 -0
  1129. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +9 -2
  1130. package/src/resources/extensions/gsd/tests/worktree-path-injection.test.ts +22 -19
  1131. package/src/resources/extensions/gsd/tests/worktree-project-root-degrade.test.ts +66 -0
  1132. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +373 -76
  1133. package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +327 -0
  1134. package/src/resources/extensions/gsd/tests/worktree-state-projection.test.ts +120 -0
  1135. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +17 -33
  1136. package/src/resources/extensions/gsd/tests/worktree-write-gate.test.ts +179 -0
  1137. package/src/resources/extensions/gsd/tests/write-gate.test.ts +40 -1
  1138. package/src/resources/extensions/gsd/tests/zero-slice-roadmap-guided.test.ts +19 -13
  1139. package/src/resources/extensions/gsd/tool-contract.ts +82 -0
  1140. package/src/resources/extensions/gsd/tools/complete-milestone.ts +29 -24
  1141. package/src/resources/extensions/gsd/tools/complete-task.ts +5 -2
  1142. package/src/resources/extensions/gsd/tools/context-mode-tool-result.ts +25 -0
  1143. package/src/resources/extensions/gsd/tools/exec-search-tool.ts +7 -7
  1144. package/src/resources/extensions/gsd/tools/exec-tool.ts +4 -23
  1145. package/src/resources/extensions/gsd/tools/memory-tools.ts +1 -0
  1146. package/src/resources/extensions/gsd/tools/plan-slice.ts +13 -0
  1147. package/src/resources/extensions/gsd/tools/plan-task.ts +10 -0
  1148. package/src/resources/extensions/gsd/tools/resume-tool.ts +7 -7
  1149. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +7 -2
  1150. package/src/resources/extensions/gsd/unit-context-composer.ts +19 -4
  1151. package/src/resources/extensions/gsd/unit-runtime.ts +25 -0
  1152. package/src/resources/extensions/gsd/uok/kernel.ts +10 -3
  1153. package/src/resources/extensions/gsd/uok/plan-v2.ts +5 -1
  1154. package/src/resources/extensions/gsd/workflow-logger.ts +13 -13
  1155. package/src/resources/extensions/gsd/workflow-manifest.ts +6 -15
  1156. package/src/resources/extensions/gsd/workflow-projections.ts +5 -1
  1157. package/src/resources/extensions/gsd/workflow-protocol.ts +160 -0
  1158. package/src/resources/extensions/gsd/workflow-templates.ts +11 -0
  1159. package/src/resources/extensions/gsd/working-output-messages.ts +120 -0
  1160. package/src/resources/extensions/gsd/worktree-lifecycle.ts +1255 -0
  1161. package/src/resources/extensions/gsd/worktree-manager.ts +15 -4
  1162. package/src/resources/extensions/gsd/worktree-safety.ts +282 -0
  1163. package/src/resources/extensions/gsd/worktree-state-projection.ts +404 -0
  1164. package/src/resources/skills/create-gsd-extension/templates/templates.test.ts +86 -40
  1165. package/src/resources/skills/web-quality-audit/scripts/analyze.sh +0 -0
  1166. package/dist/resources/extensions/gsd/worktree-resolver.js +0 -733
  1167. package/dist/web/standalone/.next/server/chunks/6336.js +0 -1
  1168. package/dist/web/standalone/.next/static/chunks/8336.6f6f30e410419aff.js +0 -10
  1169. package/src/resources/extensions/gsd/tests/sync-layer-scope.test.ts +0 -434
  1170. package/src/resources/extensions/gsd/worktree-resolver.ts +0 -909
  1171. /package/dist/web/standalone/.next/static/{V-3Ehy4B24f9FCGiLPWIM → drLMkgfHQ8lzS229_HWYR}/_buildManifest.js +0 -0
  1172. /package/dist/web/standalone/.next/static/{V-3Ehy4B24f9FCGiLPWIM → drLMkgfHQ8lzS229_HWYR}/_ssgManifest.js +0 -0
@@ -9,24 +9,41 @@
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
- import { refreshMilestoneLease } from "../db/milestone-leases.js";
19
+ import { claimMilestoneLease, 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 { emitOpenUnitEndForUnit } from "../crash-recovery.js";
29
+ import { writeUnitRuntimeRecord } from "../unit-runtime.js";
30
+ import { ensureDispatchLease, openDispatchClaim } from "./workflow-dispatch-claim.js";
31
+ import { completeWorkflowIteration } from "./workflow-iteration-completion.js";
32
+ import { createWorkflowJournalReporter } from "./workflow-journal-reporter.js";
33
+ import { createWorkflowPhaseReporter } from "./workflow-phase-reporter.js";
34
+ import { createWorkflowTurnReporter } from "./workflow-turn-reporter.js";
35
+ import { validateWorkflowSessionLock } from "./workflow-session-lock.js";
36
+ import { dequeueSidecarItem } from "./workflow-sidecar-queue.js";
37
+ import { maintainWorkerHeartbeat } from "./workflow-worker-heartbeat.js";
38
+ import { measureMemoryPressure } from "./workflow-memory-pressure.js";
39
+ import { buildSidecarIterationData } from "./workflow-sidecar-iteration.js";
40
+ import { createExecutionGraphUnitDispatchDeps, runUnitPhaseViaContract, } from "./workflow-unit-dispatch.js";
41
+ import { handleCustomEngineDispatchOutcome } from "./workflow-custom-engine-dispatch-outcome.js";
42
+ import { buildCustomEngineIterationData } from "./workflow-custom-engine-iteration.js";
43
+ import { handleCustomEngineVerifyRetry } from "./workflow-custom-engine-retry.js";
44
+ import { handleCustomEngineVerifyPause, handleCustomEngineVerifyRetryOutcome, } from "./workflow-custom-engine-verify-outcome.js";
45
+ import { handleCustomEngineReconcile } from "./workflow-custom-engine-reconcile.js";
46
+ import { handleCustomEngineReconcileOutcome } from "./workflow-custom-engine-reconcile-outcome.js";
30
47
  // ── Stuck detection persistence (#3704) ──────────────────────────────────
31
48
  // Phase C migration: stuck-state.json deleted in favor of DB-backed
32
49
  // equivalents. recentUnits is rebuilt from unit_dispatches (Phase B
@@ -70,199 +87,83 @@ function saveStuckState(s, state) {
70
87
  debugLog("autoLoop", { phase: "save-stuck-state-failed", error: err instanceof Error ? err.message : String(err) });
71
88
  }
72
89
  }
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");
90
+ function logDispatchLedgerWriteFailure(err) {
91
+ debugLog("autoLoop", {
92
+ phase: "dispatch-ledger-write-failed",
93
+ error: err instanceof Error ? err.message : String(err),
94
+ });
79
95
  }
80
- function customVerifyRetryStatePath(s) {
81
- return join(customVerifyRetryStateDir(s), "custom-verify-retries.json");
96
+ function logDispatchClaimRejected(details) {
97
+ debugLog("autoLoop", {
98
+ phase: "dispatch-claim-rejected",
99
+ ...details,
100
+ });
82
101
  }
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;
102
+ function logDispatchClaimFailed(err) {
103
+ debugLog("autoLoop", {
104
+ phase: "dispatch-claim-failed",
105
+ error: err instanceof Error ? err.message : String(err),
106
+ });
102
107
  }
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
- }
108
+ function logDispatchLeaseRecovered(details) {
109
+ debugLog("autoLoop", {
110
+ phase: details.recovered ? "dispatch-lease-recovered" : "dispatch-lease-acquired",
111
+ ...details,
112
+ });
123
113
  }
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
- }
114
+ function logDispatchLeaseRecoveryFailed(details) {
115
+ debugLog("autoLoop", {
116
+ phase: "dispatch-lease-recovery-failed",
117
+ ...details,
118
+ });
119
+ }
120
+ function logCustomVerifyRetryLoadFailure(err) {
121
+ debugLog("autoLoop", {
122
+ phase: "load-custom-verify-retries-failed",
123
+ error: err instanceof Error ? err.message : String(err),
124
+ });
125
+ }
126
+ function logCustomVerifyRetrySaveFailure(err) {
127
+ debugLog("autoLoop", {
128
+ phase: "save-custom-verify-retries-failed",
129
+ error: err instanceof Error ? err.message : String(err),
130
+ });
174
131
  }
175
132
  // ── Memory pressure monitoring (#3331) ──────────────────────────────────
176
133
  // Check heap usage every N iterations and trigger graceful shutdown before
177
134
  // the OS OOM killer sends SIGKILL. The threshold is 90% of the V8 heap
178
135
  // limit (--max-old-space-size or default ~1.5-4GB depending on platform).
179
136
  const MEMORY_CHECK_INTERVAL = 5; // check every 5 iterations
180
- const MEMORY_PRESSURE_THRESHOLD = 0.85; // 85% of heap limit
181
137
  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
138
  async function enforceMinRequestInterval(s, prefs) {
224
139
  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
- }
140
+ const decision = decideMinRequestInterval({
141
+ minIntervalMs: minInterval,
142
+ lastRequestTimestamp: s.lastRequestTimestamp,
143
+ nowMs: Date.now(),
144
+ });
145
+ if (decision.action === "wait") {
146
+ debugLog("autoLoop", { phase: "rate-limit-wait", waitMs: decision.waitMs });
147
+ await new Promise(r => setTimeout(r, decision.waitMs));
232
148
  }
233
149
  }
234
- async function runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem) {
235
- if (dispatchContract === "legacy-direct") {
236
- return runUnitPhase(ic, iterData, loopState, sidecarItem);
150
+ function closeOutCrashedUnit(s, iterData, err) {
151
+ const summary = formatDispatchExceptionSummary({ error: err });
152
+ try {
153
+ emitOpenUnitEndForUnit(s.basePath, iterData.unitType, iterData.unitId, "cancelled", {
154
+ message: summary,
155
+ category: "unit-exception",
156
+ isTransient: false,
157
+ });
158
+ writeUnitRuntimeRecord(s.basePath, iterData.unitType, iterData.unitId, s.currentUnit?.startedAt ?? Date.now(), {
159
+ phase: "crashed",
160
+ lastProgressAt: Date.now(),
161
+ lastProgressKind: "unit-exception",
162
+ });
163
+ }
164
+ catch (closeoutErr) {
165
+ logWarning("dispatch", `unit crash closeout failed: ${closeoutErr instanceof Error ? closeoutErr.message : String(closeoutErr)}`);
237
166
  }
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
167
  }
267
168
  /**
268
169
  * Main auto-mode execution loop. Iterates: derive → dispatch → guards →
@@ -276,6 +177,7 @@ export async function autoLoop(ctx, pi, s, deps, options) {
276
177
  debugLog("autoLoop", { phase: "enter" });
277
178
  let iteration = 0;
278
179
  const dispatchContract = options?.dispatchContract ?? "legacy-direct";
180
+ const unitDispatchDeps = createExecutionGraphUnitDispatchDeps();
279
181
  // Load persisted stuck state so counters survive session restarts (#3704)
280
182
  const persisted = loadStuckState(s);
281
183
  const loopState = {
@@ -289,65 +191,69 @@ export async function autoLoop(ctx, pi, s, deps, options) {
289
191
  while (s.active) {
290
192
  iteration++;
291
193
  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
- }
194
+ maintainWorkerHeartbeat(s, {
195
+ heartbeatAutoWorker,
196
+ refreshMilestoneLease,
197
+ logHeartbeatFailure: err => debugLog("autoLoop", {
198
+ phase: "heartbeat-failed",
199
+ error: err instanceof Error ? err.message : String(err),
200
+ }),
201
+ logLeaseRefreshMiss: details => debugLog("autoLoop", {
202
+ phase: "lease-refresh-missed",
203
+ ...details,
204
+ }),
205
+ });
309
206
  // ── Journal: per-iteration flow grouping ──
310
207
  const flowId = randomUUID();
311
208
  let seqCounter = 0;
312
209
  const nextSeq = () => ++seqCounter;
210
+ const journalReporter = createWorkflowJournalReporter({
211
+ emitJournalEvent: deps.emitJournalEvent,
212
+ flowId,
213
+ nextSeq,
214
+ });
313
215
  const turnId = randomUUID();
314
216
  s.currentTraceId = flowId;
315
217
  s.currentTurnId = turnId;
316
218
  const turnStartedAt = new Date().toISOString();
317
219
  let observedUnitType;
318
220
  let observedUnitId;
319
- let turnFinished = false;
221
+ const phaseReporter = createWorkflowPhaseReporter({
222
+ observer: deps.uokObserver,
223
+ });
224
+ const turnReporter = createWorkflowTurnReporter({
225
+ observer: deps.uokObserver,
226
+ traceId: flowId,
227
+ turnId,
228
+ iteration,
229
+ basePath: s.basePath,
230
+ startedAt: turnStartedAt,
231
+ clearCurrentTurn: () => {
232
+ s.currentTraceId = null;
233
+ s.currentTurnId = null;
234
+ },
235
+ });
320
236
  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,
237
+ turnReporter.finish({
328
238
  unitType: observedUnitType,
329
239
  unitId: observedUnitId,
330
240
  status,
331
241
  failureClass,
332
- phaseResults: [],
333
242
  error,
334
- startedAt: turnStartedAt,
335
- finishedAt: new Date().toISOString(),
336
243
  });
337
- s.currentTraceId = null;
338
- s.currentTurnId = null;
339
244
  };
340
- deps.uokObserver?.onTurnStart({
341
- traceId: flowId,
342
- turnId,
245
+ turnReporter.start();
246
+ const iterationDecision = decideWorkflowLoop({
247
+ active: s.active,
343
248
  iteration,
344
- basePath: s.basePath,
345
- startedAt: turnStartedAt,
249
+ maxIterations: MAX_LOOP_ITERATIONS,
250
+ hasCommandContext: true,
251
+ sessionLockValid: true,
346
252
  });
347
- if (iteration > MAX_LOOP_ITERATIONS) {
253
+ if (iterationDecision.action === "stop" && iterationDecision.reason === "max-iterations") {
348
254
  debugLog("autoLoop", {
349
255
  phase: "exit",
350
- reason: "max-iterations",
256
+ reason: iterationDecision.reason,
351
257
  iteration,
352
258
  });
353
259
  await deps.stopAuto(ctx, pi, `Safety: loop exceeded ${MAX_LOOP_ITERATIONS} iterations — possible runaway`);
@@ -357,69 +263,84 @@ export async function autoLoop(ctx, pi, s, deps, options) {
357
263
  // ── Memory pressure check (#3331) ──
358
264
  // Graceful shutdown before OOM killer sends SIGKILL.
359
265
  if (iteration % MEMORY_CHECK_INTERVAL === 0) {
360
- const mem = checkMemoryPressure();
266
+ const mem = measureMemoryPressure();
361
267
  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");
268
+ const memoryDecision = decideMemoryPressure({ ...mem, iteration });
269
+ if (memoryDecision.action === "stop") {
270
+ logWarning("dispatch", memoryDecision.warningMessage);
271
+ await deps.stopAuto(ctx, pi, memoryDecision.stopMessage);
272
+ finishTurn("stopped", "timeout", memoryDecision.turnError);
368
273
  break;
369
274
  }
370
275
  }
371
- if (!s.cmdCtx) {
276
+ const commandContextDecision = decideWorkflowLoop({
277
+ active: s.active,
278
+ iteration,
279
+ maxIterations: MAX_LOOP_ITERATIONS,
280
+ hasCommandContext: Boolean(s.cmdCtx),
281
+ sessionLockValid: true,
282
+ });
283
+ if (commandContextDecision.action === "stop" && commandContextDecision.reason === "missing-command-context") {
372
284
  debugLog("autoLoop", { phase: "exit", reason: "no-cmdCtx" });
373
- finishTurn("stopped", "manual-attention", "missing-command-context");
285
+ finishTurn("stopped", "manual-attention", commandContextDecision.reason);
374
286
  break;
375
287
  }
376
288
  let dispatchId = null;
377
289
  let dispatchSettled = false;
290
+ const completeIteration = () => {
291
+ completeWorkflowIteration({
292
+ get consecutiveErrors() { return consecutiveErrors; },
293
+ set consecutiveErrors(value) { consecutiveErrors = value; },
294
+ get consecutiveCooldowns() { return consecutiveCooldowns; },
295
+ set consecutiveCooldowns(value) { consecutiveCooldowns = value; },
296
+ recentErrorMessages,
297
+ }, {
298
+ emitIterationEnd: () => journalReporter.emit("iteration-end", { iteration }),
299
+ saveStuckState: () => saveStuckState(s, loopState),
300
+ logIterationComplete: () => debugLog("autoLoop", { phase: "iteration-complete", iteration }),
301
+ });
302
+ };
378
303
  try {
379
304
  // ── Blanket try/catch: one bad iteration must not kill the session
380
305
  const prefs = deps.loadEffectiveGSDPreferences()?.preferences;
381
306
  const uokFlags = resolveUokFlags(prefs);
382
307
  // ── 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", {
308
+ // NOTE: Sidecar dequeue MUST run before validateWorkflowSessionLock so a
309
+ // queued item is popped (and the `sidecar-dequeue` journal event emitted)
310
+ // even when the session lock invalidates this iteration. Inverting this
311
+ // order silently drops queued items on lock-loss. Refs #5308.
312
+ const sidecarItem = await dequeueSidecarItem({
313
+ queue: s.sidecarQueue,
314
+ executionGraphEnabled: uokFlags.executionGraph,
315
+ scheduleQueue: scheduleSidecarQueue,
316
+ warnSchedulingFailure: message => logWarning("dispatch", `sidecar queue scheduling failed: ${message}`),
317
+ logDequeue: payload => debugLog("autoLoop", { phase: "sidecar-dequeue", ...payload }),
318
+ emitDequeue: payload => journalReporter.emit("sidecar-dequeue", payload),
319
+ });
320
+ const sessionLockOutcome = validateWorkflowSessionLock({
321
+ active: s.active,
322
+ iteration,
323
+ maxIterations: MAX_LOOP_ITERATIONS,
324
+ deps: {
325
+ lockBase: deps.lockBase,
326
+ validateSessionLock: deps.validateSessionLock,
327
+ handleLostSessionLock: lockStatus => deps.handleLostSessionLock(ctx, lockStatus),
328
+ logInvalidSessionLock: details => debugLog("autoLoop", {
407
329
  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", {
330
+ ...details,
331
+ }),
332
+ logSessionLockExit: details => debugLog("autoLoop", {
414
333
  phase: "exit",
415
- reason: "session-lock-lost",
416
- detail: lockStatus.failureReason ?? "unknown",
417
- });
418
- break;
419
- }
334
+ ...details,
335
+ }),
336
+ },
337
+ });
338
+ if (sessionLockOutcome.action === "stop" && sessionLockOutcome.reason === "session-lock-lost") {
339
+ finishTurn("stopped", "manual-attention", sessionLockOutcome.reason);
340
+ break;
420
341
  }
421
342
  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 } });
343
+ journalReporter.emit("iteration-start", { iteration });
423
344
  let iterData;
424
345
  // ── Custom engine path ──────────────────────────────────────────────
425
346
  // When activeEngineId is a non-dev value, bypass runPreDispatch and
@@ -429,7 +350,11 @@ export async function autoLoop(ctx, pi, s, deps, options) {
429
350
  //
430
351
  // GSD_ENGINE_BYPASS=1 skips the engine layer entirely — falls through
431
352
  // to the dev path below.
432
- if (s.activeEngineId != null && s.activeEngineId !== "dev" && !sidecarItem && process.env.GSD_ENGINE_BYPASS !== "1") {
353
+ if (shouldUseCustomEnginePath({
354
+ activeEngineId: s.activeEngineId,
355
+ hasSidecarItem: Boolean(sidecarItem),
356
+ engineBypass: process.env.GSD_ENGINE_BYPASS === "1",
357
+ })) {
433
358
  debugLog("autoLoop", { phase: "custom-engine-derive", iteration, engineId: s.activeEngineId });
434
359
  const { engine, policy } = resolveEngine({
435
360
  activeEngineId: s.activeEngineId,
@@ -447,48 +372,53 @@ export async function autoLoop(ctx, pi, s, deps, options) {
447
372
  isComplete: engineState.isComplete,
448
373
  });
449
374
  if (engineState.isComplete) {
375
+ finishTurn("completed");
450
376
  await deps.stopAuto(ctx, pi, "Workflow complete");
451
377
  break;
452
378
  }
453
379
  debugLog("autoLoop", { phase: "custom-engine-dispatch", iteration });
454
380
  const dispatch = await engine.resolveDispatch(engineState, { basePath: s.basePath });
455
- if (dispatch.action === "stop") {
456
- await deps.stopAuto(ctx, pi, dispatch.reason ?? "Engine stopped");
381
+ const engineDispatchDecision = decideEngineDispatch(dispatch.action === "stop"
382
+ ? { action: "stop", reason: dispatch.reason }
383
+ : { action: dispatch.action });
384
+ const dispatchFlow = await handleCustomEngineDispatchOutcome({
385
+ decision: engineDispatchDecision,
386
+ deps: {
387
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
388
+ },
389
+ });
390
+ if (dispatchFlow.action === "break") {
391
+ finishTurn("stopped", "manual-attention", "custom-engine-dispatch-stop");
457
392
  break;
458
393
  }
459
- if (dispatch.action === "skip") {
394
+ if (dispatchFlow.action === "continue") {
395
+ finishTurn("skipped");
460
396
  continue;
461
397
  }
462
398
  // dispatch.action === "dispatch"
399
+ if (dispatch.action !== "dispatch") {
400
+ finishTurn("skipped");
401
+ continue;
402
+ }
463
403
  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",
404
+ iterData = await buildCustomEngineIterationData({
405
+ step,
468
406
  basePath: s.basePath,
469
407
  canonicalProjectRoot: s.canonicalProjectRoot,
470
- derivedPhase: gsdState.phase,
471
- activeUnit: gsdState.activeTask?.id ?? gsdState.activeSlice?.id ?? gsdState.activeMilestone?.id,
408
+ currentMilestoneId: s.currentMilestoneId,
409
+ deriveState: deps.deriveState,
410
+ logPostDerive: details => debugLog("autoLoop", {
411
+ phase: "post-derive",
412
+ ...details,
413
+ }),
472
414
  });
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
415
  observedUnitType = iterData.unitType;
486
416
  observedUnitId = iterData.unitId;
487
417
  // ── Progress widget (mirrors dev path in runDispatch) ──
488
418
  deps.updateProgressWidget(ctx, iterData.unitType, iterData.unitId, iterData.state);
489
419
  // ── Guards (shared with dev path) ──
490
420
  const guardsResult = await runGuards(ic, s.currentMilestoneId ?? "workflow");
491
- deps.uokObserver?.onPhaseResult("guard", guardsResult.action, {
421
+ phaseReporter.report("guard", guardsResult.action, {
492
422
  unitType: iterData.unitType,
493
423
  unitId: iterData.unitId,
494
424
  });
@@ -498,13 +428,23 @@ export async function autoLoop(ctx, pi, s, deps, options) {
498
428
  }
499
429
  // ── Unit execution (shared with dev path) ──
500
430
  await enforceMinRequestInterval(s, prefs);
501
- const unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState);
431
+ let unitPhaseResult;
432
+ try {
433
+ unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, undefined, unitDispatchDeps);
434
+ }
435
+ catch (err) {
436
+ if (err instanceof ModelPolicyDispatchBlockedError) {
437
+ throw err;
438
+ }
439
+ closeOutCrashedUnit(s, iterData, err);
440
+ throw err;
441
+ }
502
442
  if (unitPhaseResult.action === "next") {
503
- const requestTimestamp = unitPhaseResult.data.requestDispatchedAt ?? unitPhaseResult.data.unitStartedAt;
504
- if (typeof requestTimestamp === "number")
443
+ const requestTimestamp = resolveUnitRequestTimestamp(unitPhaseResult.data);
444
+ if (requestTimestamp !== undefined)
505
445
  s.lastRequestTimestamp = requestTimestamp;
506
446
  }
507
- deps.uokObserver?.onPhaseResult("unit", unitPhaseResult.action, {
447
+ phaseReporter.report("unit", unitPhaseResult.action, {
508
448
  unitType: iterData.unitType,
509
449
  unitId: iterData.unitId,
510
450
  });
@@ -516,103 +456,94 @@ export async function autoLoop(ctx, pi, s, deps, options) {
516
456
  debugLog("autoLoop", { phase: "custom-engine-verify", iteration, unitId: iterData.unitId });
517
457
  const verifyResult = await policy.verify(iterData.unitType, iterData.unitId, { basePath: s.basePath });
518
458
  if (verifyResult === "pause") {
519
- await deps.pauseAuto(ctx, pi);
520
- deps.uokObserver?.onPhaseResult("custom-engine", "pause", {
459
+ const verifyFlow = await handleCustomEngineVerifyPause({
521
460
  unitType: iterData.unitType,
522
461
  unitId: iterData.unitId,
462
+ deps: {
463
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
464
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
465
+ reportPause: details => phaseReporter.report("custom-engine", "pause", details),
466
+ finishTurn,
467
+ },
523
468
  });
524
- finishTurn("paused", "manual-attention", "custom-engine-verify-pause");
525
- break;
469
+ if (verifyFlow.action === "break")
470
+ break;
526
471
  }
527
472
  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", {
473
+ const retryOutcome = await handleCustomEngineVerifyRetry({
474
+ session: s,
535
475
  unitType: iterData.unitType,
536
476
  unitId: iterData.unitId,
537
- attempts,
477
+ basePath: s.basePath,
478
+ iteration,
479
+ maxRetries: MAX_CUSTOM_ENGINE_VERIFY_RETRIES,
480
+ deps: {
481
+ hydrateRetryCounts: () => hydrateCustomVerifyRetryCounts(s, {
482
+ logFailure: logCustomVerifyRetryLoadFailure,
483
+ }),
484
+ saveRetryCounts: () => saveCustomVerifyRetryCounts(s, {
485
+ logFailure: logCustomVerifyRetrySaveFailure,
486
+ }),
487
+ recover: (unitType, unitId, options) => policy.recover(unitType, unitId, options),
488
+ logRetry: details => debugLog("autoLoop", {
489
+ phase: "custom-engine-verify-retry",
490
+ ...details,
491
+ }),
492
+ reportRetry: details => phaseReporter.report("custom-engine", "retry", details),
493
+ },
494
+ });
495
+ const retryFlow = await handleCustomEngineVerifyRetryOutcome({
496
+ outcome: retryOutcome,
497
+ deps: {
498
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
499
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
500
+ reportPause: details => phaseReporter.report("custom-engine", "pause", details),
501
+ finishTurn,
502
+ },
538
503
  });
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");
504
+ if (retryFlow.action === "break")
555
505
  break;
556
- }
557
- finishTurn("retry");
558
506
  continue;
559
507
  }
560
508
  // 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(),
509
+ const reconcileOutcome = await handleCustomEngineReconcile({
510
+ session: s,
511
+ engineState,
512
+ iterData,
513
+ iteration,
514
+ deps: {
515
+ saveRetryCounts: () => saveCustomVerifyRetryCounts(s, {
516
+ logFailure: logCustomVerifyRetrySaveFailure,
517
+ }),
518
+ logReconcile: details => debugLog("autoLoop", {
519
+ phase: "custom-engine-reconcile",
520
+ ...details,
521
+ }),
522
+ reconcile: (state, completedStep) => engine.reconcile(state, completedStep),
523
+ now: () => Date.now(),
524
+ clearUnitTimeout: deps.clearUnitTimeout,
525
+ completeIteration,
526
+ },
569
527
  });
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", {
528
+ const reconcileFlow = await handleCustomEngineReconcileOutcome({
529
+ outcome: reconcileOutcome,
606
530
  unitType: iterData.unitType,
607
531
  unitId: iterData.unitId,
532
+ deps: {
533
+ stopAuto: reason => deps.stopAuto(ctx, pi, reason),
534
+ pauseAuto: () => deps.pauseAuto(ctx, pi),
535
+ report: (action, details) => phaseReporter.report("custom-engine", action, details),
536
+ finishTurn,
537
+ },
608
538
  });
609
- finishTurn("completed");
539
+ if (reconcileFlow.action === "break")
540
+ break;
610
541
  continue;
611
542
  }
612
543
  if (!sidecarItem) {
613
544
  // ── Phase 1: Pre-dispatch ─────────────────────────────────────────
614
545
  const preDispatchResult = await runPreDispatch(ic, loopState);
615
- deps.uokObserver?.onPhaseResult("pre-dispatch", preDispatchResult.action);
546
+ phaseReporter.report("pre-dispatch", preDispatchResult.action);
616
547
  if (preDispatchResult.action === "break") {
617
548
  finishTurn("stopped", "manual-attention", "pre-dispatch-break");
618
549
  break;
@@ -624,14 +555,14 @@ export async function autoLoop(ctx, pi, s, deps, options) {
624
555
  const preData = preDispatchResult.data;
625
556
  // ── Phase 2: Guards ───────────────────────────────────────────────
626
557
  const guardsResult = await runGuards(ic, preData.mid);
627
- deps.uokObserver?.onPhaseResult("guard", guardsResult.action);
558
+ phaseReporter.report("guard", guardsResult.action);
628
559
  if (guardsResult.action === "break") {
629
560
  finishTurn("stopped", "manual-attention", "guard-break");
630
561
  break;
631
562
  }
632
563
  // ── Phase 3: Dispatch ─────────────────────────────────────────────
633
564
  const dispatchResult = await runDispatch(ic, preData, loopState);
634
- deps.uokObserver?.onPhaseResult("dispatch", dispatchResult.action);
565
+ phaseReporter.report("dispatch", dispatchResult.action);
635
566
  if (dispatchResult.action === "break") {
636
567
  finishTurn("stopped", "manual-attention", "dispatch-break");
637
568
  break;
@@ -645,30 +576,19 @@ export async function autoLoop(ctx, pi, s, deps, options) {
645
576
  observedUnitId = iterData.unitId;
646
577
  }
647
578
  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",
579
+ iterData = await buildSidecarIterationData({
580
+ sidecarItem,
653
581
  basePath: s.basePath,
654
582
  canonicalProjectRoot: s.canonicalProjectRoot,
655
- derivedPhase: sidecarState.phase,
656
- activeUnit: sidecarState.activeTask?.id ?? sidecarState.activeSlice?.id ?? sidecarState.activeMilestone?.id,
583
+ deriveState: deps.deriveState,
584
+ logPostDerive: details => debugLog("autoLoop", {
585
+ phase: "post-derive",
586
+ ...details,
587
+ }),
657
588
  });
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
589
  observedUnitType = iterData.unitType;
670
590
  observedUnitId = iterData.unitId;
671
- deps.uokObserver?.onPhaseResult("dispatch", "sidecar", {
591
+ phaseReporter.report("dispatch", "sidecar", {
672
592
  unitType: iterData.unitType,
673
593
  unitId: iterData.unitId,
674
594
  sidecarKind: sidecarItem.kind,
@@ -677,56 +597,103 @@ export async function autoLoop(ctx, pi, s, deps, options) {
677
597
  await enforceMinRequestInterval(s, prefs);
678
598
  // Phase B: claim a unit_dispatches row before invoking the unit. The
679
599
  // partial unique index idx_unit_dispatches_active_per_unit prevents
680
- // a second worker from claiming the same unit concurrently. Returns
681
- // null when DB unavailable, no worker registered, or no active lease
682
- // those degraded paths fall through to the existing single-worker
683
- // 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);
600
+ // a second worker from claiming the same unit concurrently. When this
601
+ // process has a worker identity, make the milestone lease explicit before
602
+ // claiming so a step-mode handoff cannot leave us running with a stale
603
+ // in-memory token and no backing lease row.
604
+ const leaseBeforeClaim = ensureDispatchLease(s, iterData.mid, {
605
+ claimMilestoneLease,
606
+ logLeaseRecovered: logDispatchLeaseRecovered,
607
+ logLeaseRecoveryFailed: logDispatchLeaseRecoveryFailed,
608
+ });
609
+ if (leaseBeforeClaim.kind === "blocked" || leaseBeforeClaim.kind === "failed") {
610
+ const msg = `Lost milestone lease for ${iterData.mid ?? "unknown"} before dispatching ${iterData.unitType} ${iterData.unitId}: ${leaseBeforeClaim.reason}`;
611
+ ctx.ui.notify(msg, "error");
612
+ finishTurn("stopped", "execution", msg);
613
+ await deps.stopAuto(ctx, pi, msg);
614
+ break;
615
+ }
616
+ let dispatchClaim = openDispatchClaim(s, flowId, turnId, iterData, {
617
+ getRecentDispatchesForUnit,
618
+ recordDispatchClaim,
619
+ markDispatchRunning,
620
+ logClaimRejected: logDispatchClaimRejected,
621
+ logClaimFailed: logDispatchClaimFailed,
622
+ });
623
+ let dispatchDecision = decideDispatchClaim(dispatchClaim.kind === "opened"
624
+ ? { kind: "opened", dispatchId: dispatchClaim.dispatchId }
625
+ : dispatchClaim.kind === "skip"
626
+ ? { kind: "skip", reason: dispatchClaim.reason }
627
+ : { kind: "degraded" });
628
+ if (dispatchDecision.action === "skip" && dispatchDecision.reason === "stale-lease") {
629
+ const leaseRecovery = ensureDispatchLease(s, iterData.mid, {
630
+ claimMilestoneLease,
631
+ logLeaseRecovered: logDispatchLeaseRecovered,
632
+ logLeaseRecoveryFailed: logDispatchLeaseRecoveryFailed,
633
+ }, { forceReclaim: true });
634
+ if (leaseRecovery.kind === "ready") {
635
+ dispatchClaim = openDispatchClaim(s, flowId, turnId, iterData, {
636
+ getRecentDispatchesForUnit,
637
+ recordDispatchClaim,
638
+ markDispatchRunning,
639
+ logClaimRejected: logDispatchClaimRejected,
640
+ logClaimFailed: logDispatchClaimFailed,
641
+ });
642
+ dispatchDecision = decideDispatchClaim(dispatchClaim.kind === "opened"
643
+ ? { kind: "opened", dispatchId: dispatchClaim.dispatchId }
644
+ : dispatchClaim.kind === "skip"
645
+ ? { kind: "skip", reason: dispatchClaim.reason }
646
+ : { kind: "degraded" });
647
+ }
648
+ else {
649
+ const msg = `Lost milestone lease for ${iterData.mid ?? "unknown"} while claiming ${iterData.unitType} ${iterData.unitId}: ${leaseRecovery.reason}`;
650
+ ctx.ui.notify(msg, "error");
651
+ finishTurn("stopped", "execution", msg);
652
+ await deps.stopAuto(ctx, pi, msg);
653
+ break;
654
+ }
655
+ }
656
+ if (dispatchDecision.action === "skip") {
657
+ if (dispatchDecision.reason === "stale-lease") {
658
+ const msg = `Lost milestone lease for ${iterData.mid ?? "unknown"} while claiming ${iterData.unitType} ${iterData.unitId}; dispatch claim still failed after recovery.`;
659
+ ctx.ui.notify(msg, "error");
660
+ finishTurn("stopped", "execution", msg);
661
+ await deps.stopAuto(ctx, pi, msg);
662
+ break;
663
+ }
664
+ finishTurn("skipped", "execution", dispatchDecision.reason);
687
665
  continue;
688
666
  }
689
- dispatchId = dispatchClaim.kind === "opened" ? dispatchClaim.dispatchId : null;
667
+ dispatchId = dispatchDecision.dispatchId;
690
668
  let unitPhaseResult;
691
669
  try {
692
- unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem);
670
+ unitPhaseResult = await runUnitPhaseViaContract(dispatchContract, ic, iterData, loopState, sidecarItem, unitDispatchDeps);
693
671
  }
694
672
  catch (err) {
695
673
  if (err instanceof ModelPolicyDispatchBlockedError) {
696
674
  throw err;
697
675
  }
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
- }
676
+ closeOutCrashedUnit(s, iterData, err);
677
+ dispatchSettled = settleDispatchFailed(dispatchId, formatDispatchExceptionSummary({ error: err }), {
678
+ markFailed: markDispatchFailed,
679
+ logWriteFailure: logDispatchLedgerWriteFailure,
680
+ }) || dispatchSettled;
709
681
  throw err;
710
682
  }
711
683
  if (unitPhaseResult.action === "next") {
712
- const requestTimestamp = unitPhaseResult.data.requestDispatchedAt ?? unitPhaseResult.data.unitStartedAt;
713
- if (typeof requestTimestamp === "number")
684
+ const requestTimestamp = resolveUnitRequestTimestamp(unitPhaseResult.data);
685
+ if (requestTimestamp !== undefined)
714
686
  s.lastRequestTimestamp = requestTimestamp;
715
687
  }
716
- deps.uokObserver?.onPhaseResult("unit", unitPhaseResult.action, {
688
+ phaseReporter.report("unit", unitPhaseResult.action, {
717
689
  unitType: iterData.unitType,
718
690
  unitId: iterData.unitId,
719
691
  });
720
692
  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
- }
693
+ dispatchSettled = settleDispatchFailed(dispatchId, "unit-break", {
694
+ markFailed: markDispatchFailed,
695
+ logWriteFailure: logDispatchLedgerWriteFailure,
696
+ }) || dispatchSettled;
730
697
  finishTurn("stopped", "execution", "unit-break");
731
698
  break;
732
699
  }
@@ -736,88 +703,57 @@ export async function autoLoop(ctx, pi, s, deps, options) {
736
703
  finalizeResult = await runFinalize(ic, iterData, loopState, sidecarItem);
737
704
  }
738
705
  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
- }
706
+ dispatchSettled = settleDispatchFailed(dispatchId, formatDispatchExceptionSummary({ error: err }), {
707
+ markFailed: markDispatchFailed,
708
+ logWriteFailure: logDispatchLedgerWriteFailure,
709
+ }) || dispatchSettled;
750
710
  throw err;
751
711
  }
752
- deps.uokObserver?.onPhaseResult("finalize", finalizeResult.action, {
712
+ phaseReporter.report("finalize", finalizeResult.action, {
753
713
  unitType: iterData.unitType,
754
714
  unitId: iterData.unitId,
755
715
  });
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");
716
+ const finalizeDecision = decideFinalizeResult(finalizeResult.action === "break"
717
+ ? { action: "break", reason: finalizeResult.reason }
718
+ : finalizeResult.action === "continue"
719
+ ? { action: "continue" }
720
+ : { action: "next" });
721
+ if (finalizeDecision.action === "stop") {
722
+ dispatchSettled = settleDispatchFailed(dispatchId, finalizeDecision.ledgerErrorSummary, {
723
+ markFailed: markDispatchFailed,
724
+ logWriteFailure: logDispatchLedgerWriteFailure,
725
+ }) || dispatchSettled;
726
+ finishTurn("stopped", finalizeDecision.failureClass, finalizeDecision.turnError);
770
727
  break;
771
728
  }
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
- }
729
+ if (finalizeDecision.action === "retry") {
730
+ dispatchSettled = settleDispatchFailed(dispatchId, finalizeDecision.ledgerErrorSummary, {
731
+ markFailed: markDispatchFailed,
732
+ logWriteFailure: logDispatchLedgerWriteFailure,
733
+ }) || dispatchSettled;
782
734
  finishTurn("retry");
783
735
  continue;
784
736
  }
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 });
737
+ dispatchSettled = settleDispatchCompleted(dispatchId, {
738
+ markCompleted: markDispatchCompleted,
739
+ logWriteFailure: logDispatchLedgerWriteFailure,
740
+ }) || dispatchSettled;
741
+ completeIteration();
800
742
  finishTurn("completed");
801
743
  }
802
744
  catch (loopErr) {
803
745
  // ── Blanket catch: absorb unexpected exceptions, apply graduated recovery ──
804
746
  const msg = loopErr instanceof Error ? loopErr.message : String(loopErr);
805
747
  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
- }
748
+ dispatchSettled = settleDispatchFailed(dispatchId, formatUnhandledDispatchErrorSummary({ error: loopErr }), {
749
+ markFailed: markDispatchFailed,
750
+ logWriteFailure: logDispatchLedgerWriteFailure,
751
+ }) || dispatchSettled;
816
752
  }
817
753
  // Always emit iteration-end on error so the journal records iteration
818
754
  // completion even on failure (#2344). Without this, errors in
819
755
  // 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 } });
756
+ journalReporter.emit("iteration-end", { iteration, error: msg });
821
757
  // ── Pre-send model-policy block: not a retryable error (#4959 / #4850) ──
822
758
  // The model-policy gate runs before the prompt is sent. When every
823
759
  // candidate model is denied (cross-provider disabled + flat-rate
@@ -827,6 +763,12 @@ export async function autoLoop(ctx, pi, s, deps, options) {
827
763
  // instead, with the per-model deny reasons surfaced from the typed
828
764
  // error.
829
765
  if (loopErr instanceof ModelPolicyDispatchBlockedError) {
766
+ const policyDecision = decideModelPolicyBlocked({
767
+ unitType: loopErr.unitType,
768
+ unitId: loopErr.unitId,
769
+ errorMessage: msg,
770
+ reasons: loopErr.reasons,
771
+ });
830
772
  debugLog("autoLoop", {
831
773
  phase: "model-policy-blocked",
832
774
  iteration,
@@ -834,20 +776,8 @@ export async function autoLoop(ctx, pi, s, deps, options) {
834
776
  unitId: loopErr.unitId,
835
777
  reasons: loopErr.reasons,
836
778
  });
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
- });
779
+ ctx.ui.notify(policyDecision.notifyMessage, "error");
780
+ journalReporter.emit("unit-end", policyDecision.journalData);
851
781
  // Carry the blocked unit identity into the turn-result observer:
852
782
  // the throw originated inside dispatch, so observedUnitType/Id were
853
783
  // not assigned by the success path at lines 453/631/647 — but the
@@ -855,7 +785,7 @@ export async function autoLoop(ctx, pi, s, deps, options) {
855
785
  observedUnitType = loopErr.unitType;
856
786
  observedUnitId = loopErr.unitId;
857
787
  await deps.pauseAuto(ctx, pi);
858
- finishTurn("paused", "manual-attention", msg);
788
+ finishTurn(policyDecision.turnStatus, policyDecision.failureClass, msg);
859
789
  // Do NOT increment consecutiveErrors — the failure is configuration,
860
790
  // not a transient runtime fault.
861
791
  break;
@@ -865,15 +795,19 @@ export async function autoLoop(ctx, pi, s, deps, options) {
865
795
  // LLM budget on guaranteed failures.
866
796
  const infraCode = isInfrastructureError(loopErr);
867
797
  if (infraCode) {
798
+ const infraDecision = decideInfrastructureError({
799
+ code: infraCode,
800
+ errorMessage: msg,
801
+ });
868
802
  debugLog("autoLoop", {
869
803
  phase: "infrastructure-error",
870
804
  iteration,
871
805
  code: infraCode,
872
806
  error: msg,
873
807
  });
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);
808
+ ctx.ui.notify(infraDecision.notifyMessage, "error");
809
+ await deps.stopAuto(ctx, pi, infraDecision.stopMessage);
810
+ finishTurn(infraDecision.turnStatus, infraDecision.failureClass, msg);
877
811
  break;
878
812
  }
879
813
  // ── Credential cooldown: wait and retry with bounded budget ──
@@ -885,6 +819,12 @@ export async function autoLoop(ctx, pi, s, deps, options) {
885
819
  if (isTransientCooldownError(loopErr)) {
886
820
  consecutiveCooldowns++;
887
821
  const retryAfterMs = getCooldownRetryAfterMs(loopErr);
822
+ const cooldownDecision = decideCooldownRecovery({
823
+ consecutiveCooldowns,
824
+ maxCooldownRetries: MAX_COOLDOWN_RETRIES,
825
+ retryAfterMs,
826
+ fallbackWaitMs: COOLDOWN_FALLBACK_WAIT_MS,
827
+ });
888
828
  debugLog("autoLoop", {
889
829
  phase: "cooldown-wait",
890
830
  iteration,
@@ -892,16 +832,14 @@ export async function autoLoop(ctx, pi, s, deps, options) {
892
832
  retryAfterMs,
893
833
  error: msg,
894
834
  });
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`);
835
+ if (cooldownDecision.action === "stop") {
836
+ ctx.ui.notify(cooldownDecision.notifyMessage, "error");
837
+ finishTurn("stopped", "timeout", msg);
838
+ await deps.stopAuto(ctx, pi, cooldownDecision.stopMessage);
898
839
  break;
899
840
  }
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));
841
+ ctx.ui.notify(cooldownDecision.notifyMessage, "warning");
842
+ await new Promise(resolve => setTimeout(resolve, cooldownDecision.waitMs));
905
843
  finishTurn("retry", "timeout", msg);
906
844
  continue; // Retry iteration without incrementing consecutiveErrors
907
845
  }
@@ -913,26 +851,25 @@ export async function autoLoop(ctx, pi, s, deps, options) {
913
851
  consecutiveErrors,
914
852
  error: msg,
915
853
  });
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);
854
+ const errorDecision = decideIterationErrorRecovery({
855
+ consecutiveErrors,
856
+ recentErrorMessages,
857
+ currentErrorMessage: msg,
858
+ });
859
+ if (errorDecision.action === "stop") {
860
+ ctx.ui.notify(errorDecision.notifyMessage, "error");
861
+ await deps.stopAuto(ctx, pi, errorDecision.stopMessage);
862
+ finishTurn(errorDecision.turnStatus, "execution", msg);
924
863
  break;
925
864
  }
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");
865
+ if (errorDecision.action === "invalidate-and-retry") {
866
+ ctx.ui.notify(errorDecision.notifyMessage, "warning");
929
867
  deps.invalidateAllCaches();
930
868
  }
931
869
  else {
932
- // 1st error: log and retry — transient failures happen
933
- ctx.ui.notify(`Iteration error: ${msg}. Retrying.`, "warning");
870
+ ctx.ui.notify(errorDecision.notifyMessage, "warning");
934
871
  }
935
- finishTurn("retry", "execution", msg);
872
+ finishTurn(errorDecision.turnStatus, "execution", msg);
936
873
  }
937
874
  }
938
875
  _clearCurrentResolve();