gsd-pi 2.33.1-dev.ee47f1b → 2.34.0-dev.0150ae9

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 (356) hide show
  1. package/dist/bundled-resource-path.d.ts +8 -0
  2. package/dist/bundled-resource-path.js +14 -0
  3. package/dist/headless-query.js +6 -6
  4. package/dist/resources/extensions/gsd/auto/session.js +27 -32
  5. package/dist/resources/extensions/gsd/auto-dashboard.js +29 -109
  6. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +6 -1
  7. package/dist/resources/extensions/gsd/auto-dispatch.js +52 -81
  8. package/dist/resources/extensions/gsd/auto-loop.js +956 -0
  9. package/dist/resources/extensions/gsd/auto-observability.js +4 -2
  10. package/dist/resources/extensions/gsd/auto-post-unit.js +75 -185
  11. package/dist/resources/extensions/gsd/auto-prompts.js +133 -101
  12. package/dist/resources/extensions/gsd/auto-recovery.js +59 -97
  13. package/dist/resources/extensions/gsd/auto-start.js +330 -309
  14. package/dist/resources/extensions/gsd/auto-supervisor.js +5 -11
  15. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +7 -7
  16. package/dist/resources/extensions/gsd/auto-timers.js +3 -4
  17. package/dist/resources/extensions/gsd/auto-verification.js +35 -73
  18. package/dist/resources/extensions/gsd/auto-worktree-sync.js +167 -0
  19. package/dist/resources/extensions/gsd/auto-worktree.js +291 -126
  20. package/dist/resources/extensions/gsd/auto.js +283 -1013
  21. package/dist/resources/extensions/gsd/captures.js +10 -4
  22. package/dist/resources/extensions/gsd/dispatch-guard.js +7 -8
  23. package/dist/resources/extensions/gsd/docs/preferences-reference.md +25 -18
  24. package/dist/resources/extensions/gsd/doctor-checks.js +3 -4
  25. package/dist/resources/extensions/gsd/git-service.js +1 -1
  26. package/dist/resources/extensions/gsd/gsd-db.js +296 -151
  27. package/dist/resources/extensions/gsd/index.js +92 -228
  28. package/dist/resources/extensions/gsd/post-unit-hooks.js +13 -13
  29. package/dist/resources/extensions/gsd/progress-score.js +61 -156
  30. package/dist/resources/extensions/gsd/quick.js +98 -122
  31. package/dist/resources/extensions/gsd/session-lock.js +13 -0
  32. package/dist/resources/extensions/gsd/templates/preferences.md +1 -0
  33. package/dist/resources/extensions/gsd/undo.js +43 -48
  34. package/dist/resources/extensions/gsd/unit-runtime.js +16 -15
  35. package/dist/resources/extensions/gsd/verification-evidence.js +0 -1
  36. package/dist/resources/extensions/gsd/verification-gate.js +6 -35
  37. package/dist/resources/extensions/gsd/worktree-command.js +30 -24
  38. package/dist/resources/extensions/gsd/worktree-manager.js +2 -3
  39. package/dist/resources/extensions/gsd/worktree-resolver.js +344 -0
  40. package/dist/resources/extensions/gsd/worktree.js +7 -44
  41. package/dist/resources/extensions/mcp-client/index.js +2 -1
  42. package/dist/tool-bootstrap.js +59 -11
  43. package/dist/worktree-cli.js +7 -7
  44. package/package.json +1 -1
  45. package/packages/native/dist/native.d.ts +0 -2
  46. package/packages/native/dist/native.js +0 -2
  47. package/packages/native/src/native.ts +0 -3
  48. package/packages/pi-agent-core/dist/proxy.d.ts +1 -25
  49. package/packages/pi-agent-core/dist/proxy.d.ts.map +1 -1
  50. package/packages/pi-agent-core/dist/proxy.js +1 -1
  51. package/packages/pi-agent-core/dist/proxy.js.map +1 -1
  52. package/packages/pi-agent-core/src/proxy.ts +1 -1
  53. package/packages/pi-ai/dist/api-registry.d.ts +0 -2
  54. package/packages/pi-ai/dist/api-registry.d.ts.map +1 -1
  55. package/packages/pi-ai/dist/api-registry.js +0 -10
  56. package/packages/pi-ai/dist/api-registry.js.map +1 -1
  57. package/packages/pi-ai/dist/models.generated.d.ts +3630 -5483
  58. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  59. package/packages/pi-ai/dist/models.generated.js +735 -2588
  60. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  61. package/packages/pi-ai/dist/providers/anthropic.d.ts +0 -8
  62. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  63. package/packages/pi-ai/dist/providers/anthropic.js +1 -1
  64. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  65. package/packages/pi-ai/dist/providers/github-copilot-headers.d.ts +0 -1
  66. package/packages/pi-ai/dist/providers/github-copilot-headers.d.ts.map +1 -1
  67. package/packages/pi-ai/dist/providers/github-copilot-headers.js +1 -1
  68. package/packages/pi-ai/dist/providers/github-copilot-headers.js.map +1 -1
  69. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts +1 -43
  70. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
  71. package/packages/pi-ai/dist/providers/google-gemini-cli.js +2 -2
  72. package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
  73. package/packages/pi-ai/dist/providers/google-shared.d.ts +0 -4
  74. package/packages/pi-ai/dist/providers/google-shared.d.ts.map +1 -1
  75. package/packages/pi-ai/dist/providers/google-shared.js +1 -1
  76. package/packages/pi-ai/dist/providers/google-shared.js.map +1 -1
  77. package/packages/pi-ai/dist/providers/register-builtins.d.ts +0 -1
  78. package/packages/pi-ai/dist/providers/register-builtins.d.ts.map +1 -1
  79. package/packages/pi-ai/dist/providers/register-builtins.js +1 -1
  80. package/packages/pi-ai/dist/providers/register-builtins.js.map +1 -1
  81. package/packages/pi-ai/dist/utils/event-stream.d.ts +0 -2
  82. package/packages/pi-ai/dist/utils/event-stream.d.ts.map +1 -1
  83. package/packages/pi-ai/dist/utils/event-stream.js +0 -4
  84. package/packages/pi-ai/dist/utils/event-stream.js.map +1 -1
  85. package/packages/pi-ai/dist/utils/overflow.d.ts +0 -4
  86. package/packages/pi-ai/dist/utils/overflow.d.ts.map +1 -1
  87. package/packages/pi-ai/dist/utils/overflow.js +0 -6
  88. package/packages/pi-ai/dist/utils/overflow.js.map +1 -1
  89. package/packages/pi-ai/dist/utils/validation.d.ts +0 -8
  90. package/packages/pi-ai/dist/utils/validation.d.ts.map +1 -1
  91. package/packages/pi-ai/dist/utils/validation.js +0 -14
  92. package/packages/pi-ai/dist/utils/validation.js.map +1 -1
  93. package/packages/pi-ai/src/api-registry.ts +0 -12
  94. package/packages/pi-ai/src/models.generated.ts +1039 -2892
  95. package/packages/pi-ai/src/providers/anthropic.ts +1 -1
  96. package/packages/pi-ai/src/providers/github-copilot-headers.ts +1 -1
  97. package/packages/pi-ai/src/providers/google-gemini-cli.ts +2 -2
  98. package/packages/pi-ai/src/providers/google-shared.ts +1 -1
  99. package/packages/pi-ai/src/providers/register-builtins.ts +1 -1
  100. package/packages/pi-ai/src/utils/event-stream.ts +0 -5
  101. package/packages/pi-ai/src/utils/overflow.ts +1 -8
  102. package/packages/pi-ai/src/utils/validation.ts +0 -15
  103. package/packages/pi-coding-agent/dist/config.d.ts +0 -9
  104. package/packages/pi-coding-agent/dist/config.d.ts.map +1 -1
  105. package/packages/pi-coding-agent/dist/config.js +4 -8
  106. package/packages/pi-coding-agent/dist/config.js.map +1 -1
  107. package/packages/pi-coding-agent/dist/core/export-html/ansi-to-html.d.ts +0 -4
  108. package/packages/pi-coding-agent/dist/core/export-html/ansi-to-html.d.ts.map +1 -1
  109. package/packages/pi-coding-agent/dist/core/export-html/ansi-to-html.js +1 -1
  110. package/packages/pi-coding-agent/dist/core/export-html/ansi-to-html.js.map +1 -1
  111. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +0 -5
  112. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  113. package/packages/pi-coding-agent/dist/core/extensions/runner.js +0 -13
  114. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  115. package/packages/pi-coding-agent/dist/core/keybindings.d.ts +0 -8
  116. package/packages/pi-coding-agent/dist/core/keybindings.d.ts.map +1 -1
  117. package/packages/pi-coding-agent/dist/core/keybindings.js +2 -2
  118. package/packages/pi-coding-agent/dist/core/keybindings.js.map +1 -1
  119. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts +0 -17
  120. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -1
  121. package/packages/pi-coding-agent/dist/core/lsp/client.js +3 -62
  122. package/packages/pi-coding-agent/dist/core/lsp/client.js.map +1 -1
  123. package/packages/pi-coding-agent/dist/core/lsp/config.d.ts +0 -2
  124. package/packages/pi-coding-agent/dist/core/lsp/config.d.ts.map +1 -1
  125. package/packages/pi-coding-agent/dist/core/lsp/config.js +0 -7
  126. package/packages/pi-coding-agent/dist/core/lsp/config.js.map +1 -1
  127. package/packages/pi-coding-agent/dist/core/lsp/edits.d.ts +0 -5
  128. package/packages/pi-coding-agent/dist/core/lsp/edits.d.ts.map +1 -1
  129. package/packages/pi-coding-agent/dist/core/lsp/edits.js +1 -1
  130. package/packages/pi-coding-agent/dist/core/lsp/edits.js.map +1 -1
  131. package/packages/pi-coding-agent/dist/core/lsp/lspmux.d.ts +0 -1
  132. package/packages/pi-coding-agent/dist/core/lsp/lspmux.d.ts.map +1 -1
  133. package/packages/pi-coding-agent/dist/core/lsp/lspmux.js +1 -1
  134. package/packages/pi-coding-agent/dist/core/lsp/lspmux.js.map +1 -1
  135. package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts +1 -6
  136. package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts.map +1 -1
  137. package/packages/pi-coding-agent/dist/core/lsp/utils.js +1 -28
  138. package/packages/pi-coding-agent/dist/core/lsp/utils.js.map +1 -1
  139. package/packages/pi-coding-agent/dist/core/messages.d.ts +0 -8
  140. package/packages/pi-coding-agent/dist/core/messages.d.ts.map +1 -1
  141. package/packages/pi-coding-agent/dist/core/messages.js +5 -5
  142. package/packages/pi-coding-agent/dist/core/messages.js.map +1 -1
  143. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +0 -3
  144. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  145. package/packages/pi-coding-agent/dist/core/model-registry.js +1 -3
  146. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/model-resolver.d.ts +1 -26
  148. package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
  149. package/packages/pi-coding-agent/dist/core/model-resolver.js +3 -59
  150. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  151. package/packages/pi-coding-agent/dist/core/prompt-templates.d.ts +0 -17
  152. package/packages/pi-coding-agent/dist/core/prompt-templates.d.ts.map +1 -1
  153. package/packages/pi-coding-agent/dist/core/prompt-templates.js +2 -2
  154. package/packages/pi-coding-agent/dist/core/prompt-templates.js.map +1 -1
  155. package/packages/pi-coding-agent/dist/core/session-manager.d.ts +0 -4
  156. package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
  157. package/packages/pi-coding-agent/dist/core/session-manager.js +2 -4
  158. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  159. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +0 -12
  160. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  161. package/packages/pi-coding-agent/dist/core/settings-manager.js +2 -2
  162. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  163. package/packages/pi-coding-agent/dist/migrations.d.ts +0 -16
  164. package/packages/pi-coding-agent/dist/migrations.d.ts.map +1 -1
  165. package/packages/pi-coding-agent/dist/migrations.js +2 -2
  166. package/packages/pi-coding-agent/dist/migrations.js.map +1 -1
  167. package/packages/pi-coding-agent/dist/modes/interactive/components/session-selector-search.d.ts +0 -2
  168. package/packages/pi-coding-agent/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -1
  169. package/packages/pi-coding-agent/dist/modes/interactive/components/session-selector-search.js +2 -2
  170. package/packages/pi-coding-agent/dist/modes/interactive/components/session-selector-search.js.map +1 -1
  171. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +1 -24
  172. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  173. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +50 -512
  174. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  175. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts +71 -0
  176. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts.map +1 -0
  177. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +514 -0
  178. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -0
  179. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts +65 -4
  180. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js +6 -23
  182. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts +12 -0
  184. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts.map +1 -0
  185. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js +175 -0
  186. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js.map +1 -0
  187. package/packages/pi-coding-agent/dist/utils/changelog.d.ts +0 -4
  188. package/packages/pi-coding-agent/dist/utils/changelog.d.ts.map +1 -1
  189. package/packages/pi-coding-agent/dist/utils/changelog.js +1 -1
  190. package/packages/pi-coding-agent/dist/utils/changelog.js.map +1 -1
  191. package/packages/pi-coding-agent/dist/utils/clipboard-image.d.ts +0 -1
  192. package/packages/pi-coding-agent/dist/utils/clipboard-image.d.ts.map +1 -1
  193. package/packages/pi-coding-agent/dist/utils/clipboard-image.js +1 -1
  194. package/packages/pi-coding-agent/dist/utils/clipboard-image.js.map +1 -1
  195. package/packages/pi-coding-agent/dist/utils/photon.d.ts +0 -19
  196. package/packages/pi-coding-agent/dist/utils/photon.d.ts.map +1 -1
  197. package/packages/pi-coding-agent/dist/utils/photon.js +1 -120
  198. package/packages/pi-coding-agent/dist/utils/photon.js.map +1 -1
  199. package/packages/pi-coding-agent/dist/utils/tools-manager.d.ts +0 -1
  200. package/packages/pi-coding-agent/dist/utils/tools-manager.d.ts.map +1 -1
  201. package/packages/pi-coding-agent/dist/utils/tools-manager.js +1 -1
  202. package/packages/pi-coding-agent/dist/utils/tools-manager.js.map +1 -1
  203. package/packages/pi-coding-agent/package.json +1 -1
  204. package/packages/pi-coding-agent/src/config.ts +5 -10
  205. package/packages/pi-coding-agent/src/core/export-html/ansi-to-html.ts +1 -1
  206. package/packages/pi-coding-agent/src/core/extensions/runner.ts +0 -13
  207. package/packages/pi-coding-agent/src/core/keybindings.ts +2 -2
  208. package/packages/pi-coding-agent/src/core/lsp/client.ts +3 -73
  209. package/packages/pi-coding-agent/src/core/lsp/config.ts +0 -11
  210. package/packages/pi-coding-agent/src/core/lsp/edits.ts +1 -1
  211. package/packages/pi-coding-agent/src/core/lsp/lspmux.ts +1 -1
  212. package/packages/pi-coding-agent/src/core/lsp/utils.ts +1 -33
  213. package/packages/pi-coding-agent/src/core/messages.ts +5 -5
  214. package/packages/pi-coding-agent/src/core/model-registry.ts +0 -2
  215. package/packages/pi-coding-agent/src/core/model-resolver.ts +3 -77
  216. package/packages/pi-coding-agent/src/core/prompt-templates.ts +2 -2
  217. package/packages/pi-coding-agent/src/core/session-manager.ts +2 -4
  218. package/packages/pi-coding-agent/src/core/settings-manager.ts +2 -2
  219. package/packages/pi-coding-agent/src/migrations.ts +2 -2
  220. package/packages/pi-coding-agent/src/modes/interactive/components/session-selector-search.ts +2 -2
  221. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +50 -561
  222. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +653 -0
  223. package/packages/pi-coding-agent/src/modes/interactive/theme/theme.ts +7 -26
  224. package/packages/pi-coding-agent/src/modes/interactive/theme/themes.ts +196 -0
  225. package/packages/pi-coding-agent/src/utils/changelog.ts +1 -1
  226. package/packages/pi-coding-agent/src/utils/clipboard-image.ts +1 -1
  227. package/packages/pi-coding-agent/src/utils/photon.ts +0 -137
  228. package/packages/pi-coding-agent/src/utils/tools-manager.ts +1 -1
  229. package/packages/pi-tui/dist/components/editor.d.ts +0 -10
  230. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  231. package/packages/pi-tui/dist/components/editor.js +1 -1
  232. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  233. package/packages/pi-tui/dist/overlay-layout.d.ts +55 -0
  234. package/packages/pi-tui/dist/overlay-layout.d.ts.map +1 -0
  235. package/packages/pi-tui/dist/overlay-layout.js +288 -0
  236. package/packages/pi-tui/dist/overlay-layout.js.map +1 -0
  237. package/packages/pi-tui/dist/tui.d.ts +0 -22
  238. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  239. package/packages/pi-tui/dist/tui.js +6 -272
  240. package/packages/pi-tui/dist/tui.js.map +1 -1
  241. package/packages/pi-tui/dist/utils.d.ts +0 -7
  242. package/packages/pi-tui/dist/utils.d.ts.map +1 -1
  243. package/packages/pi-tui/dist/utils.js +0 -44
  244. package/packages/pi-tui/dist/utils.js.map +1 -1
  245. package/packages/pi-tui/src/components/editor.ts +1 -1
  246. package/packages/pi-tui/src/overlay-layout.ts +372 -0
  247. package/packages/pi-tui/src/tui.ts +11 -312
  248. package/packages/pi-tui/src/utils.ts +0 -43
  249. package/pkg/dist/core/export-html/ansi-to-html.d.ts +0 -4
  250. package/pkg/dist/core/export-html/ansi-to-html.d.ts.map +1 -1
  251. package/pkg/dist/core/export-html/ansi-to-html.js +1 -1
  252. package/pkg/dist/core/export-html/ansi-to-html.js.map +1 -1
  253. package/pkg/dist/modes/interactive/theme/theme.d.ts +65 -4
  254. package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  255. package/pkg/dist/modes/interactive/theme/theme.js +6 -23
  256. package/pkg/dist/modes/interactive/theme/theme.js.map +1 -1
  257. package/pkg/dist/modes/interactive/theme/themes.d.ts +12 -0
  258. package/pkg/dist/modes/interactive/theme/themes.d.ts.map +1 -0
  259. package/pkg/dist/modes/interactive/theme/themes.js +175 -0
  260. package/pkg/dist/modes/interactive/theme/themes.js.map +1 -0
  261. package/pkg/package.json +1 -1
  262. package/src/resources/extensions/gsd/auto/session.ts +47 -30
  263. package/src/resources/extensions/gsd/auto-dashboard.ts +28 -131
  264. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +6 -1
  265. package/src/resources/extensions/gsd/auto-dispatch.ts +135 -91
  266. package/src/resources/extensions/gsd/auto-loop.ts +1665 -0
  267. package/src/resources/extensions/gsd/auto-observability.ts +4 -2
  268. package/src/resources/extensions/gsd/auto-post-unit.ts +85 -228
  269. package/src/resources/extensions/gsd/auto-prompts.ts +138 -109
  270. package/src/resources/extensions/gsd/auto-recovery.ts +124 -118
  271. package/src/resources/extensions/gsd/auto-start.ts +440 -354
  272. package/src/resources/extensions/gsd/auto-supervisor.ts +5 -12
  273. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +8 -8
  274. package/src/resources/extensions/gsd/auto-timers.ts +3 -4
  275. package/src/resources/extensions/gsd/auto-verification.ts +76 -90
  276. package/src/resources/extensions/gsd/auto-worktree-sync.ts +204 -0
  277. package/src/resources/extensions/gsd/auto-worktree.ts +389 -141
  278. package/src/resources/extensions/gsd/auto.ts +515 -1199
  279. package/src/resources/extensions/gsd/captures.ts +10 -4
  280. package/src/resources/extensions/gsd/dispatch-guard.ts +13 -9
  281. package/src/resources/extensions/gsd/docs/preferences-reference.md +25 -18
  282. package/src/resources/extensions/gsd/doctor-checks.ts +3 -4
  283. package/src/resources/extensions/gsd/git-service.ts +8 -1
  284. package/src/resources/extensions/gsd/gitignore.ts +4 -2
  285. package/src/resources/extensions/gsd/gsd-db.ts +375 -180
  286. package/src/resources/extensions/gsd/index.ts +104 -263
  287. package/src/resources/extensions/gsd/post-unit-hooks.ts +13 -13
  288. package/src/resources/extensions/gsd/progress-score.ts +65 -200
  289. package/src/resources/extensions/gsd/quick.ts +121 -125
  290. package/src/resources/extensions/gsd/session-lock.ts +11 -0
  291. package/src/resources/extensions/gsd/templates/preferences.md +1 -0
  292. package/src/resources/extensions/gsd/tests/agent-end-retry.test.ts +32 -59
  293. package/src/resources/extensions/gsd/tests/all-milestones-complete-merge.test.ts +75 -27
  294. package/src/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +1 -1
  295. package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +37 -0
  296. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +1458 -0
  297. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +8 -162
  298. package/src/resources/extensions/gsd/tests/auto-secrets-gate.test.ts +2 -108
  299. package/src/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +1 -3
  300. package/src/resources/extensions/gsd/tests/auto-worktree-milestone-merge.test.ts +0 -3
  301. package/src/resources/extensions/gsd/tests/auto-worktree.test.ts +58 -0
  302. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +0 -55
  303. package/src/resources/extensions/gsd/tests/headless-query.test.ts +22 -0
  304. package/src/resources/extensions/gsd/tests/milestone-transition-worktree.test.ts +8 -11
  305. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +4 -6
  306. package/src/resources/extensions/gsd/tests/run-uat.test.ts +3 -3
  307. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +64 -0
  308. package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +181 -0
  309. package/src/resources/extensions/gsd/tests/stale-worktree-cwd.test.ts +0 -3
  310. package/src/resources/extensions/gsd/tests/token-profile.test.ts +6 -6
  311. package/src/resources/extensions/gsd/tests/triage-dispatch.test.ts +6 -6
  312. package/src/resources/extensions/gsd/tests/undo.test.ts +6 -0
  313. package/src/resources/extensions/gsd/tests/verification-evidence.test.ts +24 -26
  314. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +7 -201
  315. package/src/resources/extensions/gsd/tests/worktree-db-integration.test.ts +205 -0
  316. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +442 -0
  317. package/src/resources/extensions/gsd/tests/worktree-e2e.test.ts +0 -3
  318. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +705 -0
  319. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +57 -106
  320. package/src/resources/extensions/gsd/tests/worktree.test.ts +5 -1
  321. package/src/resources/extensions/gsd/tests/write-gate.test.ts +43 -132
  322. package/src/resources/extensions/gsd/types.ts +90 -81
  323. package/src/resources/extensions/gsd/undo.ts +42 -46
  324. package/src/resources/extensions/gsd/unit-runtime.ts +14 -18
  325. package/src/resources/extensions/gsd/verification-evidence.ts +1 -3
  326. package/src/resources/extensions/gsd/verification-gate.ts +6 -39
  327. package/src/resources/extensions/gsd/worktree-command.ts +36 -24
  328. package/src/resources/extensions/gsd/worktree-manager.ts +2 -3
  329. package/src/resources/extensions/gsd/worktree-resolver.ts +485 -0
  330. package/src/resources/extensions/gsd/worktree.ts +7 -44
  331. package/src/resources/extensions/mcp-client/index.ts +2 -1
  332. package/dist/resources/extensions/gsd/auto-constants.js +0 -5
  333. package/dist/resources/extensions/gsd/auto-idempotency.js +0 -106
  334. package/dist/resources/extensions/gsd/auto-stuck-detection.js +0 -165
  335. package/dist/resources/extensions/gsd/mechanical-completion.js +0 -351
  336. package/packages/pi-coding-agent/dist/modes/interactive/theme/dark.json +0 -85
  337. package/packages/pi-coding-agent/dist/modes/interactive/theme/light.json +0 -84
  338. package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.json +0 -335
  339. package/packages/pi-coding-agent/src/modes/interactive/theme/dark.json +0 -85
  340. package/packages/pi-coding-agent/src/modes/interactive/theme/light.json +0 -84
  341. package/packages/pi-coding-agent/src/modes/interactive/theme/theme-schema.json +0 -335
  342. package/pkg/dist/modes/interactive/theme/dark.json +0 -85
  343. package/pkg/dist/modes/interactive/theme/light.json +0 -84
  344. package/pkg/dist/modes/interactive/theme/theme-schema.json +0 -335
  345. package/src/resources/extensions/gsd/auto-constants.ts +0 -6
  346. package/src/resources/extensions/gsd/auto-idempotency.ts +0 -151
  347. package/src/resources/extensions/gsd/auto-stuck-detection.ts +0 -221
  348. package/src/resources/extensions/gsd/mechanical-completion.ts +0 -430
  349. package/src/resources/extensions/gsd/tests/auto-dispatch-loop.test.ts +0 -691
  350. package/src/resources/extensions/gsd/tests/auto-reentrancy-guard.test.ts +0 -127
  351. package/src/resources/extensions/gsd/tests/auto-skip-loop.test.ts +0 -123
  352. package/src/resources/extensions/gsd/tests/dispatch-stall-guard.test.ts +0 -126
  353. package/src/resources/extensions/gsd/tests/loop-regression.test.ts +0 -874
  354. package/src/resources/extensions/gsd/tests/mechanical-completion.test.ts +0 -356
  355. package/src/resources/extensions/gsd/tests/progress-score.test.ts +0 -206
  356. package/src/resources/extensions/gsd/tests/session-lock.test.ts +0 -434
@@ -0,0 +1,956 @@
1
+ /**
2
+ * auto-loop.ts — Linear loop execution backbone for auto-mode.
3
+ *
4
+ * Replaces the recursive dispatchNextUnit → handleAgentEnd → dispatchNextUnit
5
+ * pattern with a while loop. The agent_end event resolves a promise instead
6
+ * of recursing.
7
+ *
8
+ * MAINTENANCE RULE: The only module-level mutable state here is `_activeSession`,
9
+ * used by the agent_end bridge. Promise state itself lives on AutoSession so
10
+ * concurrent auto sessions cannot corrupt each other.
11
+ */
12
+ import { NEW_SESSION_TIMEOUT_MS } from "./auto/session.js";
13
+ import { debugLog } from "./debug-logger.js";
14
+ /**
15
+ * Maximum total loop iterations before forced stop. Prevents runaway loops
16
+ * when units alternate IDs (bypassing the same-unit stuck detector).
17
+ * A milestone with 20 slices × 5 tasks × 3 phases ≈ 300 units. 500 gives
18
+ * generous headroom including retries and sidecar work.
19
+ */
20
+ const MAX_LOOP_ITERATIONS = 500;
21
+ // ─── Session-scoped promise state ───────────────────────────────────────────
22
+ //
23
+ // pendingResolve and pendingAgentEndQueue live on AutoSession (not module-level)
24
+ // so concurrent sessions cannot corrupt each other's promises.
25
+ /**
26
+ * The singleton session reference used by resolveAgentEnd. Set by autoLoop
27
+ * on entry so that the agent_end handler in index.ts can resolve the correct
28
+ * session's promise without needing a direct reference to `s`.
29
+ */
30
+ let _activeSession = null;
31
+ // ─── resolveAgentEnd ─────────────────────────────────────────────────────────
32
+ /**
33
+ * Called from the agent_end event handler in index.ts to resolve the
34
+ * in-flight unit promise. One-shot: the resolver is nulled before calling
35
+ * to prevent double-resolution from model fallback retries.
36
+ *
37
+ * If no pendingResolve exists (event arrived between loop iterations),
38
+ * the event is queued on the session so the next runUnit can drain it.
39
+ */
40
+ export function resolveAgentEnd(event) {
41
+ const s = _activeSession;
42
+ if (!s) {
43
+ debugLog("resolveAgentEnd", {
44
+ status: "no-active-session",
45
+ warning: "agent_end with no active loop session",
46
+ });
47
+ return;
48
+ }
49
+ if (s.pendingResolve) {
50
+ debugLog("resolveAgentEnd", { status: "resolving", hasEvent: true });
51
+ const r = s.pendingResolve;
52
+ s.pendingResolve = null;
53
+ r({ status: "completed", event });
54
+ }
55
+ else {
56
+ // Queue the event so the next runUnit picks it up immediately
57
+ debugLog("resolveAgentEnd", {
58
+ status: "queued",
59
+ queueLength: s.pendingAgentEndQueue.length + 1,
60
+ warning: "agent_end arrived between loop iterations — queued for next runUnit",
61
+ });
62
+ s.pendingAgentEndQueue.push(event);
63
+ }
64
+ }
65
+ export function isSessionSwitchInFlight() {
66
+ return _activeSession?.sessionSwitchInFlight ?? false;
67
+ }
68
+ // ─── resetPendingResolve (test helper) ───────────────────────────────────────
69
+ /**
70
+ * Reset session promise state. Only exported for test cleanup — production code
71
+ * should never call this.
72
+ */
73
+ export function _resetPendingResolve() {
74
+ if (_activeSession) {
75
+ _activeSession.pendingResolve = null;
76
+ _activeSession.pendingAgentEndQueue = [];
77
+ }
78
+ _activeSession = null;
79
+ }
80
+ /**
81
+ * Set the active session for resolveAgentEnd. Only exported for test setup —
82
+ * production code sets this via autoLoop entry.
83
+ */
84
+ export function _setActiveSession(session) {
85
+ _activeSession = session;
86
+ }
87
+ // ─── runUnit ─────────────────────────────────────────────────────────────────
88
+ /**
89
+ * Execute a single unit: create a new session, send the prompt, and await
90
+ * the agent_end promise. Returns a UnitResult describing what happened.
91
+ *
92
+ * The promise is one-shot: resolveAgentEnd() is the only way to resolve it.
93
+ * On session creation failure or timeout, returns { status: 'cancelled' }
94
+ * without awaiting the promise.
95
+ */
96
+ export async function runUnit(ctx, pi, s, unitType, unitId, prompt, _prefs) {
97
+ debugLog("runUnit", { phase: "start", unitType, unitId });
98
+ // ── Drain queued events from error-recovery retries ──
99
+ // If an agent_end arrived between iterations (e.g. from a model fallback
100
+ // sendMessage retry), consume it immediately instead of creating a new promise.
101
+ // Cap queue to 3 entries to prevent unbounded growth from stale events.
102
+ if (s.pendingAgentEndQueue.length > 3) {
103
+ debugLog("runUnit", {
104
+ phase: "queue-overflow",
105
+ dropped: s.pendingAgentEndQueue.length - 1,
106
+ unitType,
107
+ unitId,
108
+ });
109
+ s.pendingAgentEndQueue = [
110
+ s.pendingAgentEndQueue[s.pendingAgentEndQueue.length - 1],
111
+ ];
112
+ }
113
+ if (s.pendingAgentEndQueue.length > 0) {
114
+ const queued = s.pendingAgentEndQueue.shift();
115
+ debugLog("runUnit", {
116
+ phase: "drained-queued-event",
117
+ unitType,
118
+ unitId,
119
+ queueRemaining: s.pendingAgentEndQueue.length,
120
+ });
121
+ return { status: "completed", event: queued };
122
+ }
123
+ // ── Session creation with timeout ──
124
+ debugLog("runUnit", { phase: "session-create", unitType, unitId });
125
+ let sessionResult;
126
+ let sessionTimeoutHandle;
127
+ s.sessionSwitchInFlight = true;
128
+ try {
129
+ const sessionPromise = s.cmdCtx.newSession().finally(() => {
130
+ s.sessionSwitchInFlight = false;
131
+ });
132
+ const timeoutPromise = new Promise((resolve) => {
133
+ sessionTimeoutHandle = setTimeout(() => resolve({ cancelled: true }), NEW_SESSION_TIMEOUT_MS);
134
+ });
135
+ sessionResult = await Promise.race([sessionPromise, timeoutPromise]);
136
+ }
137
+ catch (sessionErr) {
138
+ if (sessionTimeoutHandle)
139
+ clearTimeout(sessionTimeoutHandle);
140
+ const msg = sessionErr instanceof Error ? sessionErr.message : String(sessionErr);
141
+ debugLog("runUnit", {
142
+ phase: "session-error",
143
+ unitType,
144
+ unitId,
145
+ error: msg,
146
+ });
147
+ return { status: "cancelled" };
148
+ }
149
+ if (sessionTimeoutHandle)
150
+ clearTimeout(sessionTimeoutHandle);
151
+ if (sessionResult.cancelled) {
152
+ debugLog("runUnit-session-timeout", { unitType, unitId });
153
+ return { status: "cancelled" };
154
+ }
155
+ if (!s.active) {
156
+ return { status: "cancelled" };
157
+ }
158
+ // ── Create the agent_end promise (session-scoped) ──
159
+ // This happens after newSession completes so session-switch agent_end events
160
+ // from the previous session cannot resolve the new unit.
161
+ const unitPromise = new Promise((resolve) => {
162
+ s.pendingResolve = resolve;
163
+ });
164
+ // ── Send the prompt ──
165
+ debugLog("runUnit", { phase: "send-message", unitType, unitId });
166
+ pi.sendMessage({ customType: "gsd-auto", content: prompt, display: s.verbose }, { triggerTurn: true });
167
+ // ── Await agent_end ──
168
+ debugLog("runUnit", { phase: "awaiting-agent-end", unitType, unitId });
169
+ const result = await unitPromise;
170
+ debugLog("runUnit", {
171
+ phase: "agent-end-received",
172
+ unitType,
173
+ unitId,
174
+ status: result.status,
175
+ });
176
+ return result;
177
+ }
178
+ // ─── autoLoop ────────────────────────────────────────────────────────────────
179
+ /**
180
+ * Main auto-mode execution loop. Iterates: derive → dispatch → guards →
181
+ * runUnit → finalize → repeat. Exits when s.active becomes false or a
182
+ * terminal condition is reached.
183
+ *
184
+ * This is the linear replacement for the recursive
185
+ * dispatchNextUnit → handleAgentEnd → dispatchNextUnit chain.
186
+ */
187
+ export async function autoLoop(ctx, pi, s, deps) {
188
+ debugLog("autoLoop", { phase: "enter" });
189
+ _activeSession = s;
190
+ let iteration = 0;
191
+ let lastDerivedUnit = "";
192
+ let sameUnitCount = 0;
193
+ let consecutiveErrors = 0;
194
+ while (s.active) {
195
+ iteration++;
196
+ debugLog("autoLoop", { phase: "loop-top", iteration });
197
+ if (iteration > MAX_LOOP_ITERATIONS) {
198
+ debugLog("autoLoop", {
199
+ phase: "exit",
200
+ reason: "max-iterations",
201
+ iteration,
202
+ });
203
+ await deps.stopAuto(ctx, pi, `Safety: loop exceeded ${MAX_LOOP_ITERATIONS} iterations — possible runaway`);
204
+ break;
205
+ }
206
+ if (!s.cmdCtx) {
207
+ debugLog("autoLoop", { phase: "exit", reason: "no-cmdCtx" });
208
+ break;
209
+ }
210
+ try {
211
+ // ── Blanket try/catch: one bad iteration must not kill the session
212
+ if (deps.lockBase() && !deps.validateSessionLock(deps.lockBase())) {
213
+ deps.handleLostSessionLock(ctx);
214
+ debugLog("autoLoop", { phase: "exit", reason: "session-lock-lost" });
215
+ break;
216
+ }
217
+ // ── Phase 1: Pre-dispatch ───────────────────────────────────────────
218
+ // Resource version guard
219
+ const staleMsg = deps.checkResourcesStale(s.resourceVersionOnStart);
220
+ if (staleMsg) {
221
+ await deps.stopAuto(ctx, pi, staleMsg);
222
+ debugLog("autoLoop", { phase: "exit", reason: "resources-stale" });
223
+ break;
224
+ }
225
+ deps.invalidateAllCaches();
226
+ s.lastPromptCharCount = undefined;
227
+ s.lastBaselineCharCount = undefined;
228
+ // Pre-dispatch health gate
229
+ try {
230
+ const healthGate = await deps.preDispatchHealthGate(s.basePath);
231
+ if (healthGate.fixesApplied.length > 0) {
232
+ ctx.ui.notify(`Pre-dispatch: ${healthGate.fixesApplied.join(", ")}`, "info");
233
+ }
234
+ if (!healthGate.proceed) {
235
+ ctx.ui.notify(healthGate.reason ?? "Pre-dispatch health check failed.", "error");
236
+ await deps.pauseAuto(ctx, pi);
237
+ debugLog("autoLoop", { phase: "exit", reason: "health-gate-failed" });
238
+ break;
239
+ }
240
+ }
241
+ catch {
242
+ // Non-fatal
243
+ }
244
+ // Sync project root artifacts into worktree
245
+ if (s.originalBasePath &&
246
+ s.basePath !== s.originalBasePath &&
247
+ s.currentMilestoneId) {
248
+ deps.syncProjectRootToWorktree(s.originalBasePath, s.basePath, s.currentMilestoneId);
249
+ }
250
+ // Derive state
251
+ let state = await deps.deriveState(s.basePath);
252
+ let mid = state.activeMilestone?.id;
253
+ let midTitle = state.activeMilestone?.title;
254
+ debugLog("autoLoop", {
255
+ phase: "state-derived",
256
+ iteration,
257
+ mid,
258
+ statePhase: state.phase,
259
+ });
260
+ // ── Milestone transition ────────────────────────────────────────────
261
+ if (mid && s.currentMilestoneId && mid !== s.currentMilestoneId) {
262
+ ctx.ui.notify(`Milestone ${s.currentMilestoneId} complete. Advancing to ${mid}: ${midTitle}.`, "info");
263
+ deps.sendDesktopNotification("GSD", `Milestone ${s.currentMilestoneId} complete!`, "success", "milestone");
264
+ const vizPrefs = deps.loadEffectiveGSDPreferences()?.preferences;
265
+ if (vizPrefs?.auto_visualize) {
266
+ ctx.ui.notify("Run /gsd visualize to see progress overview.", "info");
267
+ }
268
+ if (vizPrefs?.auto_report !== false) {
269
+ try {
270
+ const { loadVisualizerData } = await import("./visualizer-data.js");
271
+ const { generateHtmlReport } = await import("./export-html.js");
272
+ const { writeReportSnapshot } = await import("./reports.js");
273
+ const { basename } = await import("node:path");
274
+ const snapData = await loadVisualizerData(s.basePath);
275
+ const completedMs = snapData.milestones.find((m) => m.id === s.currentMilestoneId);
276
+ const msTitle = completedMs?.title ?? s.currentMilestoneId;
277
+ const gsdVersion = process.env.GSD_VERSION ?? "0.0.0";
278
+ const projName = basename(s.basePath);
279
+ const doneSlices = snapData.milestones.reduce((acc, m) => acc +
280
+ m.slices.filter((sl) => sl.done).length, 0);
281
+ const totalSlices = snapData.milestones.reduce((acc, m) => acc + m.slices.length, 0);
282
+ const outPath = writeReportSnapshot({
283
+ basePath: s.basePath,
284
+ html: generateHtmlReport(snapData, {
285
+ projectName: projName,
286
+ projectPath: s.basePath,
287
+ gsdVersion,
288
+ milestoneId: s.currentMilestoneId,
289
+ indexRelPath: "index.html",
290
+ }),
291
+ milestoneId: s.currentMilestoneId,
292
+ milestoneTitle: msTitle,
293
+ kind: "milestone",
294
+ projectName: projName,
295
+ projectPath: s.basePath,
296
+ gsdVersion,
297
+ totalCost: snapData.totals?.cost ?? 0,
298
+ totalTokens: snapData.totals?.tokens.total ?? 0,
299
+ totalDuration: snapData.totals?.duration ?? 0,
300
+ doneSlices,
301
+ totalSlices,
302
+ doneMilestones: snapData.milestones.filter((m) => m.status === "complete").length,
303
+ totalMilestones: snapData.milestones.length,
304
+ phase: snapData.phase,
305
+ });
306
+ ctx.ui.notify(`Report saved: .gsd/reports/${(await import("node:path")).basename(outPath)} — open index.html to browse progression.`, "info");
307
+ }
308
+ catch (err) {
309
+ ctx.ui.notify(`Report generation failed: ${err instanceof Error ? err.message : String(err)}`, "warning");
310
+ }
311
+ }
312
+ // Reset dispatch counters for new milestone
313
+ s.unitDispatchCount.clear();
314
+ s.unitRecoveryCount.clear();
315
+ s.unitLifetimeDispatches.clear();
316
+ lastDerivedUnit = "";
317
+ sameUnitCount = 0;
318
+ // Worktree lifecycle on milestone transition — merge current, enter next
319
+ deps.resolver.mergeAndExit(s.currentMilestoneId, ctx.ui);
320
+ deps.invalidateAllCaches();
321
+ state = await deps.deriveState(s.basePath);
322
+ mid = state.activeMilestone?.id;
323
+ midTitle = state.activeMilestone?.title;
324
+ if (mid) {
325
+ if (deps.getIsolationMode() !== "none") {
326
+ deps.captureIntegrationBranch(s.basePath, mid, {
327
+ commitDocs: deps.loadEffectiveGSDPreferences()?.preferences?.git
328
+ ?.commit_docs,
329
+ });
330
+ }
331
+ deps.resolver.enterMilestone(mid, ctx.ui);
332
+ }
333
+ else {
334
+ // mid is undefined — no milestone to capture integration branch for
335
+ }
336
+ const pendingIds = state.registry
337
+ .filter((m) => m.status !== "complete" && m.status !== "parked")
338
+ .map((m) => m.id);
339
+ deps.pruneQueueOrder(s.basePath, pendingIds);
340
+ }
341
+ if (mid) {
342
+ s.currentMilestoneId = mid;
343
+ deps.setActiveMilestoneId(s.basePath, mid);
344
+ }
345
+ // ── Terminal conditions ──────────────────────────────────────────────
346
+ if (!mid) {
347
+ if (s.currentUnit) {
348
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
349
+ }
350
+ const incomplete = state.registry.filter((m) => m.status !== "complete" && m.status !== "parked");
351
+ if (incomplete.length === 0) {
352
+ // All milestones complete — merge milestone branch before stopping
353
+ if (s.currentMilestoneId) {
354
+ deps.resolver.mergeAndExit(s.currentMilestoneId, ctx.ui);
355
+ }
356
+ deps.sendDesktopNotification("GSD", "All milestones complete!", "success", "milestone");
357
+ await deps.stopAuto(ctx, pi, "All milestones complete");
358
+ }
359
+ else if (state.phase === "blocked") {
360
+ const blockerMsg = `Blocked: ${state.blockers.join(", ")}`;
361
+ await deps.stopAuto(ctx, pi, blockerMsg);
362
+ ctx.ui.notify(`${blockerMsg}. Fix and run /gsd auto.`, "warning");
363
+ deps.sendDesktopNotification("GSD", blockerMsg, "error", "attention");
364
+ }
365
+ else {
366
+ const ids = incomplete.map((m) => m.id).join(", ");
367
+ const diag = `basePath=${s.basePath}, milestones=[${state.registry.map((m) => `${m.id}:${m.status}`).join(", ")}], phase=${state.phase}`;
368
+ ctx.ui.notify(`Unexpected: ${incomplete.length} incomplete milestone(s) (${ids}) but no active milestone.\n Diagnostic: ${diag}`, "error");
369
+ await deps.stopAuto(ctx, pi, `No active milestone — ${incomplete.length} incomplete (${ids}), see diagnostic above`);
370
+ }
371
+ debugLog("autoLoop", { phase: "exit", reason: "no-active-milestone" });
372
+ break;
373
+ }
374
+ if (!midTitle) {
375
+ midTitle = mid;
376
+ ctx.ui.notify(`Milestone ${mid} has no title in roadmap — using ID as fallback.`, "warning");
377
+ }
378
+ // Mid-merge safety check
379
+ if (deps.reconcileMergeState(s.basePath, ctx)) {
380
+ deps.invalidateAllCaches();
381
+ state = await deps.deriveState(s.basePath);
382
+ mid = state.activeMilestone?.id;
383
+ midTitle = state.activeMilestone?.title;
384
+ }
385
+ if (!mid || !midTitle) {
386
+ if (s.currentUnit) {
387
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
388
+ }
389
+ const noMilestoneReason = !mid
390
+ ? "No active milestone after merge reconciliation"
391
+ : `Milestone ${mid} has no title after reconciliation`;
392
+ await deps.stopAuto(ctx, pi, noMilestoneReason);
393
+ debugLog("autoLoop", {
394
+ phase: "exit",
395
+ reason: "no-milestone-after-reconciliation",
396
+ });
397
+ break;
398
+ }
399
+ // Terminal: complete
400
+ if (state.phase === "complete") {
401
+ if (s.currentUnit) {
402
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
403
+ }
404
+ // Milestone merge on complete
405
+ if (s.currentMilestoneId) {
406
+ deps.resolver.mergeAndExit(s.currentMilestoneId, ctx.ui);
407
+ }
408
+ deps.sendDesktopNotification("GSD", `Milestone ${mid} complete!`, "success", "milestone");
409
+ await deps.stopAuto(ctx, pi, `Milestone ${mid} complete`);
410
+ debugLog("autoLoop", { phase: "exit", reason: "milestone-complete" });
411
+ break;
412
+ }
413
+ // Terminal: blocked
414
+ if (state.phase === "blocked") {
415
+ if (s.currentUnit) {
416
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
417
+ }
418
+ const blockerMsg = `Blocked: ${state.blockers.join(", ")}`;
419
+ await deps.stopAuto(ctx, pi, blockerMsg);
420
+ ctx.ui.notify(`${blockerMsg}. Fix and run /gsd auto.`, "warning");
421
+ deps.sendDesktopNotification("GSD", blockerMsg, "error", "attention");
422
+ debugLog("autoLoop", { phase: "exit", reason: "blocked" });
423
+ break;
424
+ }
425
+ // ── Phase 2: Guards ─────────────────────────────────────────────────
426
+ const prefs = deps.loadEffectiveGSDPreferences()?.preferences;
427
+ // Budget ceiling guard
428
+ const budgetCeiling = prefs?.budget_ceiling;
429
+ if (budgetCeiling !== undefined && budgetCeiling > 0) {
430
+ const currentLedger = deps.getLedger();
431
+ const totalCost = currentLedger
432
+ ? deps.getProjectTotals(currentLedger.units).cost
433
+ : 0;
434
+ const budgetPct = totalCost / budgetCeiling;
435
+ const budgetAlertLevel = deps.getBudgetAlertLevel(budgetPct);
436
+ const newBudgetAlertLevel = deps.getNewBudgetAlertLevel(s.lastBudgetAlertLevel, budgetPct);
437
+ const enforcement = prefs?.budget_enforcement ?? "pause";
438
+ const budgetEnforcementAction = deps.getBudgetEnforcementAction(enforcement, budgetPct);
439
+ if (newBudgetAlertLevel === 100 && budgetEnforcementAction !== "none") {
440
+ const msg = `Budget ceiling ${deps.formatCost(budgetCeiling)} reached (spent ${deps.formatCost(totalCost)}).`;
441
+ s.lastBudgetAlertLevel =
442
+ newBudgetAlertLevel;
443
+ if (budgetEnforcementAction === "halt") {
444
+ deps.sendDesktopNotification("GSD", msg, "error", "budget");
445
+ await deps.stopAuto(ctx, pi, "Budget ceiling reached");
446
+ debugLog("autoLoop", { phase: "exit", reason: "budget-halt" });
447
+ break;
448
+ }
449
+ if (budgetEnforcementAction === "pause") {
450
+ ctx.ui.notify(`${msg} Pausing auto-mode — /gsd auto to override and continue.`, "warning");
451
+ deps.sendDesktopNotification("GSD", msg, "warning", "budget");
452
+ await deps.pauseAuto(ctx, pi);
453
+ debugLog("autoLoop", { phase: "exit", reason: "budget-pause" });
454
+ break;
455
+ }
456
+ ctx.ui.notify(`${msg} Continuing (enforcement: warn).`, "warning");
457
+ deps.sendDesktopNotification("GSD", msg, "warning", "budget");
458
+ }
459
+ else if (newBudgetAlertLevel === 90) {
460
+ s.lastBudgetAlertLevel =
461
+ newBudgetAlertLevel;
462
+ ctx.ui.notify(`Budget 90%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "warning");
463
+ deps.sendDesktopNotification("GSD", `Budget 90%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "warning", "budget");
464
+ }
465
+ else if (newBudgetAlertLevel === 80) {
466
+ s.lastBudgetAlertLevel =
467
+ newBudgetAlertLevel;
468
+ ctx.ui.notify(`Approaching budget ceiling — 80%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "warning");
469
+ deps.sendDesktopNotification("GSD", `Approaching budget ceiling — 80%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "warning", "budget");
470
+ }
471
+ else if (newBudgetAlertLevel === 75) {
472
+ s.lastBudgetAlertLevel =
473
+ newBudgetAlertLevel;
474
+ ctx.ui.notify(`Budget 75%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "info");
475
+ deps.sendDesktopNotification("GSD", `Budget 75%: ${deps.formatCost(totalCost)} / ${deps.formatCost(budgetCeiling)}`, "info", "budget");
476
+ }
477
+ else if (budgetAlertLevel === 0) {
478
+ s.lastBudgetAlertLevel = 0;
479
+ }
480
+ }
481
+ else {
482
+ s.lastBudgetAlertLevel = 0;
483
+ }
484
+ // Context window guard
485
+ const contextThreshold = prefs?.context_pause_threshold ?? 0;
486
+ if (contextThreshold > 0 && s.cmdCtx) {
487
+ const contextUsage = s.cmdCtx.getContextUsage();
488
+ if (contextUsage &&
489
+ contextUsage.percent !== null &&
490
+ contextUsage.percent >= contextThreshold) {
491
+ const msg = `Context window at ${contextUsage.percent}% (threshold: ${contextThreshold}%). Pausing to prevent truncated output.`;
492
+ ctx.ui.notify(`${msg} Run /gsd auto to continue (will start fresh session).`, "warning");
493
+ deps.sendDesktopNotification("GSD", `Context ${contextUsage.percent}% — paused`, "warning", "attention");
494
+ await deps.pauseAuto(ctx, pi);
495
+ debugLog("autoLoop", { phase: "exit", reason: "context-window" });
496
+ break;
497
+ }
498
+ }
499
+ // Secrets re-check gate
500
+ try {
501
+ const manifestStatus = await deps.getManifestStatus(s.basePath, mid);
502
+ if (manifestStatus && manifestStatus.pending.length > 0) {
503
+ const result = await deps.collectSecretsFromManifest(s.basePath, mid, ctx);
504
+ if (result &&
505
+ result.applied &&
506
+ result.skipped &&
507
+ result.existingSkipped) {
508
+ ctx.ui.notify(`Secrets collected: ${result.applied.length} applied, ${result.skipped.length} skipped, ${result.existingSkipped.length} already set.`, "info");
509
+ }
510
+ else {
511
+ ctx.ui.notify("Secrets collection skipped.", "info");
512
+ }
513
+ }
514
+ }
515
+ catch (err) {
516
+ ctx.ui.notify(`Secrets collection error: ${err instanceof Error ? err.message : String(err)}. Continuing with next task.`, "warning");
517
+ }
518
+ // ── Phase 3: Dispatch resolution ────────────────────────────────────
519
+ debugLog("autoLoop", { phase: "dispatch-resolve", iteration });
520
+ const dispatchResult = await deps.resolveDispatch({
521
+ basePath: s.basePath,
522
+ mid,
523
+ midTitle: midTitle,
524
+ state,
525
+ prefs,
526
+ });
527
+ if (dispatchResult.action === "stop") {
528
+ if (s.currentUnit) {
529
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
530
+ }
531
+ await deps.stopAuto(ctx, pi, dispatchResult.reason);
532
+ debugLog("autoLoop", { phase: "exit", reason: "dispatch-stop" });
533
+ break;
534
+ }
535
+ if (dispatchResult.action !== "dispatch") {
536
+ // Non-dispatch action (e.g. "skip") — re-derive state
537
+ await new Promise((r) => setImmediate(r));
538
+ continue;
539
+ }
540
+ let unitType = dispatchResult.unitType;
541
+ let unitId = dispatchResult.unitId;
542
+ let prompt = dispatchResult.prompt;
543
+ const pauseAfterUatDispatch = dispatchResult.pauseAfterDispatch ?? false;
544
+ // ── Same-unit stuck counter with graduated recovery ──
545
+ const derivedKey = `${unitType}/${unitId}`;
546
+ if (derivedKey === lastDerivedUnit && !s.pendingVerificationRetry) {
547
+ sameUnitCount++;
548
+ debugLog("autoLoop", {
549
+ phase: "stuck-check",
550
+ unitType,
551
+ unitId,
552
+ sameUnitCount,
553
+ });
554
+ if (sameUnitCount === 3) {
555
+ // Level 1: try verifying the artifact — maybe it was written but not detected
556
+ const artifactExists = deps.verifyExpectedArtifact(unitType, unitId, s.basePath);
557
+ if (artifactExists) {
558
+ debugLog("autoLoop", {
559
+ phase: "stuck-recovery",
560
+ level: 1,
561
+ action: "artifact-found",
562
+ });
563
+ ctx.ui.notify(`Stuck recovery: artifact for ${unitType} ${unitId} found on disk. Invalidating caches.`, "info");
564
+ deps.invalidateAllCaches();
565
+ continue;
566
+ }
567
+ ctx.ui.notify(`Stuck on ${unitType} ${unitId} (attempt ${sameUnitCount}). Invalidating caches and retrying.`, "warning");
568
+ deps.invalidateAllCaches();
569
+ }
570
+ else if (sameUnitCount === 5) {
571
+ // Level 2: hard stop — genuinely stuck
572
+ debugLog("autoLoop", {
573
+ phase: "stuck-detected",
574
+ unitType,
575
+ unitId,
576
+ sameUnitCount,
577
+ });
578
+ await deps.stopAuto(ctx, pi, `Stuck: ${unitType} ${unitId} derived ${sameUnitCount} consecutive times without progress`);
579
+ ctx.ui.notify(`Stuck on ${unitType} ${unitId} — deriveState returns the same unit after ${sameUnitCount} attempts. The expected artifact was not written.`, "error");
580
+ break;
581
+ }
582
+ }
583
+ else {
584
+ if (derivedKey !== lastDerivedUnit) {
585
+ debugLog("autoLoop", {
586
+ phase: "stuck-counter-reset",
587
+ from: lastDerivedUnit,
588
+ to: derivedKey,
589
+ });
590
+ }
591
+ lastDerivedUnit = derivedKey;
592
+ sameUnitCount = 0;
593
+ }
594
+ // Pre-dispatch hooks
595
+ const preDispatchResult = deps.runPreDispatchHooks(unitType, unitId, prompt, s.basePath);
596
+ if (preDispatchResult.firedHooks.length > 0) {
597
+ ctx.ui.notify(`Pre-dispatch hook${preDispatchResult.firedHooks.length > 1 ? "s" : ""}: ${preDispatchResult.firedHooks.join(", ")}`, "info");
598
+ }
599
+ if (preDispatchResult.action === "skip") {
600
+ ctx.ui.notify(`Skipping ${unitType} ${unitId} (pre-dispatch hook).`, "info");
601
+ await new Promise((r) => setImmediate(r));
602
+ continue;
603
+ }
604
+ if (preDispatchResult.action === "replace") {
605
+ prompt = preDispatchResult.prompt ?? prompt;
606
+ if (preDispatchResult.unitType)
607
+ unitType = preDispatchResult.unitType;
608
+ }
609
+ else if (preDispatchResult.prompt) {
610
+ prompt = preDispatchResult.prompt;
611
+ }
612
+ const priorSliceBlocker = deps.getPriorSliceCompletionBlocker(s.basePath, deps.getMainBranch(s.basePath), unitType, unitId);
613
+ if (priorSliceBlocker) {
614
+ await deps.stopAuto(ctx, pi, priorSliceBlocker);
615
+ debugLog("autoLoop", { phase: "exit", reason: "prior-slice-blocker" });
616
+ break;
617
+ }
618
+ const observabilityIssues = await deps.collectObservabilityWarnings(ctx, s.basePath, unitType, unitId);
619
+ // ── Phase 4: Unit execution ─────────────────────────────────────────
620
+ debugLog("autoLoop", {
621
+ phase: "unit-execution",
622
+ iteration,
623
+ unitType,
624
+ unitId,
625
+ });
626
+ // Closeout previous unit
627
+ if (s.currentUnit) {
628
+ await deps.closeoutUnit(ctx, s.basePath, s.currentUnit.type, s.currentUnit.id, s.currentUnit.startedAt, deps.buildSnapshotOpts(s.currentUnit.type, s.currentUnit.id));
629
+ if (s.currentUnitRouting) {
630
+ const isRetry = s.currentUnit.type === unitType && s.currentUnit.id === unitId;
631
+ deps.recordOutcome(s.currentUnit.type, s.currentUnitRouting.tier, !isRetry);
632
+ }
633
+ const closeoutKey = `${s.currentUnit.type}/${s.currentUnit.id}`;
634
+ const incomingKey = `${unitType}/${unitId}`;
635
+ const isHookUnit = s.currentUnit.type.startsWith("hook/");
636
+ const artifactVerified = isHookUnit ||
637
+ deps.verifyExpectedArtifact(s.currentUnit.type, s.currentUnit.id, s.basePath);
638
+ if (closeoutKey !== incomingKey && artifactVerified) {
639
+ s.completedUnits.push({
640
+ type: s.currentUnit.type,
641
+ id: s.currentUnit.id,
642
+ startedAt: s.currentUnit.startedAt,
643
+ finishedAt: Date.now(),
644
+ });
645
+ if (s.completedUnits.length > 200) {
646
+ s.completedUnits = s.completedUnits.slice(-200);
647
+ }
648
+ deps.clearUnitRuntimeRecord(s.basePath, s.currentUnit.type, s.currentUnit.id);
649
+ s.unitDispatchCount.delete(`${s.currentUnit.type}/${s.currentUnit.id}`);
650
+ s.unitRecoveryCount.delete(`${s.currentUnit.type}/${s.currentUnit.id}`);
651
+ }
652
+ }
653
+ s.currentUnit = { type: unitType, id: unitId, startedAt: Date.now() };
654
+ deps.captureAvailableSkills();
655
+ deps.writeUnitRuntimeRecord(s.basePath, unitType, unitId, s.currentUnit.startedAt, {
656
+ phase: "dispatched",
657
+ wrapupWarningSent: false,
658
+ timeoutAt: null,
659
+ lastProgressAt: s.currentUnit.startedAt,
660
+ progressCount: 0,
661
+ lastProgressKind: "dispatch",
662
+ });
663
+ // Status bar + progress widget
664
+ ctx.ui.setStatus("gsd-auto", "auto");
665
+ if (mid)
666
+ deps.updateSliceProgressCache(s.basePath, mid, state.activeSlice?.id);
667
+ deps.updateProgressWidget(ctx, unitType, unitId, state);
668
+ deps.ensurePreconditions(unitType, unitId, s.basePath, state);
669
+ // Prompt injection
670
+ const MAX_RECOVERY_CHARS = 50_000;
671
+ let finalPrompt = prompt;
672
+ if (s.pendingVerificationRetry) {
673
+ const retryCtx = s.pendingVerificationRetry;
674
+ s.pendingVerificationRetry = null;
675
+ const capped = retryCtx.failureContext.length > MAX_RECOVERY_CHARS
676
+ ? retryCtx.failureContext.slice(0, MAX_RECOVERY_CHARS) +
677
+ "\n\n[...failure context truncated]"
678
+ : retryCtx.failureContext;
679
+ finalPrompt = `**VERIFICATION FAILED — AUTO-FIX ATTEMPT ${retryCtx.attempt}**\n\nThe verification gate ran after your previous attempt and found failures. Fix these issues before completing the task.\n\n${capped}\n\n---\n\n${finalPrompt}`;
680
+ }
681
+ if (s.pendingCrashRecovery) {
682
+ const capped = s.pendingCrashRecovery.length > MAX_RECOVERY_CHARS
683
+ ? s.pendingCrashRecovery.slice(0, MAX_RECOVERY_CHARS) +
684
+ "\n\n[...recovery briefing truncated to prevent memory exhaustion]"
685
+ : s.pendingCrashRecovery;
686
+ finalPrompt = `${capped}\n\n---\n\n${finalPrompt}`;
687
+ s.pendingCrashRecovery = null;
688
+ }
689
+ else if ((s.unitDispatchCount.get(`${unitType}/${unitId}`) ?? 0) > 1) {
690
+ const diagnostic = deps.getDeepDiagnostic(s.basePath);
691
+ if (diagnostic) {
692
+ const cappedDiag = diagnostic.length > MAX_RECOVERY_CHARS
693
+ ? diagnostic.slice(0, MAX_RECOVERY_CHARS) +
694
+ "\n\n[...diagnostic truncated to prevent memory exhaustion]"
695
+ : diagnostic;
696
+ finalPrompt = `**RETRY — your previous attempt did not produce the required artifact.**\n\nDiagnostic from previous attempt:\n${cappedDiag}\n\nFix whatever went wrong and make sure you write the required file this time.\n\n---\n\n${finalPrompt}`;
697
+ }
698
+ }
699
+ const repairBlock = deps.buildObservabilityRepairBlock(observabilityIssues);
700
+ if (repairBlock) {
701
+ finalPrompt = `${finalPrompt}${repairBlock}`;
702
+ }
703
+ // Prompt char measurement
704
+ s.lastPromptCharCount = finalPrompt.length;
705
+ s.lastBaselineCharCount = undefined;
706
+ if (deps.isDbAvailable()) {
707
+ try {
708
+ const { inlineGsdRootFile } = await import("./auto-prompts.js");
709
+ const [decisionsContent, requirementsContent, projectContent] = await Promise.all([
710
+ inlineGsdRootFile(s.basePath, "decisions.md", "Decisions"),
711
+ inlineGsdRootFile(s.basePath, "requirements.md", "Requirements"),
712
+ inlineGsdRootFile(s.basePath, "project.md", "Project"),
713
+ ]);
714
+ s.lastBaselineCharCount =
715
+ (decisionsContent?.length ?? 0) +
716
+ (requirementsContent?.length ?? 0) +
717
+ (projectContent?.length ?? 0);
718
+ }
719
+ catch {
720
+ // Non-fatal
721
+ }
722
+ }
723
+ // Cache-optimize prompt section ordering
724
+ try {
725
+ finalPrompt = deps.reorderForCaching(finalPrompt);
726
+ }
727
+ catch (reorderErr) {
728
+ const msg = reorderErr instanceof Error ? reorderErr.message : String(reorderErr);
729
+ process.stderr.write(`[gsd] prompt reorder failed (non-fatal): ${msg}\n`);
730
+ }
731
+ // Select and apply model
732
+ const modelResult = await deps.selectAndApplyModel(ctx, pi, unitType, unitId, s.basePath, prefs, s.verbose, s.autoModeStartModel);
733
+ s.currentUnitRouting =
734
+ modelResult.routing;
735
+ // Start unit supervision
736
+ deps.clearUnitTimeout();
737
+ deps.startUnitSupervision({
738
+ s,
739
+ ctx,
740
+ pi,
741
+ unitType,
742
+ unitId,
743
+ prefs,
744
+ buildSnapshotOpts: () => deps.buildSnapshotOpts(unitType, unitId),
745
+ buildRecoveryContext: () => ({}),
746
+ pauseAuto: deps.pauseAuto,
747
+ });
748
+ // Session + send + await
749
+ const sessionFile = deps.getSessionFile(ctx);
750
+ deps.updateSessionLock(deps.lockBase(), unitType, unitId, s.completedUnits.length, sessionFile);
751
+ deps.writeLock(deps.lockBase(), unitType, unitId, s.completedUnits.length, sessionFile);
752
+ debugLog("autoLoop", {
753
+ phase: "runUnit-start",
754
+ iteration,
755
+ unitType,
756
+ unitId,
757
+ });
758
+ const unitResult = await runUnit(ctx, pi, s, unitType, unitId, finalPrompt, prefs);
759
+ debugLog("autoLoop", {
760
+ phase: "runUnit-end",
761
+ iteration,
762
+ unitType,
763
+ unitId,
764
+ status: unitResult.status,
765
+ });
766
+ if (unitResult.status === "cancelled") {
767
+ ctx.ui.notify(`Session creation timed out or was cancelled for ${unitType} ${unitId}. Will retry.`, "warning");
768
+ await deps.stopAuto(ctx, pi, "Session creation failed");
769
+ debugLog("autoLoop", { phase: "exit", reason: "session-failed" });
770
+ break;
771
+ }
772
+ // ── Phase 5: Finalize ───────────────────────────────────────────────
773
+ debugLog("autoLoop", { phase: "finalize", iteration });
774
+ // Clear unit timeout (unit completed)
775
+ deps.clearUnitTimeout();
776
+ // Post-unit context for pre/post verification
777
+ const postUnitCtx = {
778
+ s,
779
+ ctx,
780
+ pi,
781
+ buildSnapshotOpts: deps.buildSnapshotOpts,
782
+ lockBase: deps.lockBase,
783
+ stopAuto: deps.stopAuto,
784
+ pauseAuto: deps.pauseAuto,
785
+ updateProgressWidget: deps.updateProgressWidget,
786
+ };
787
+ // Pre-verification processing (commit, doctor, state rebuild, etc.)
788
+ const preResult = await deps.postUnitPreVerification(postUnitCtx);
789
+ if (preResult === "dispatched") {
790
+ debugLog("autoLoop", {
791
+ phase: "exit",
792
+ reason: "pre-verification-dispatched",
793
+ });
794
+ break;
795
+ }
796
+ if (pauseAfterUatDispatch) {
797
+ ctx.ui.notify("UAT requires human execution. Auto-mode will pause after this unit writes the result file.", "info");
798
+ await deps.pauseAuto(ctx, pi);
799
+ debugLog("autoLoop", { phase: "exit", reason: "uat-pause" });
800
+ break;
801
+ }
802
+ // Verification gate — the loop handles retries via s.pendingVerificationRetry
803
+ const verificationResult = await deps.runPostUnitVerification({ s, ctx, pi }, deps.pauseAuto);
804
+ if (verificationResult === "pause") {
805
+ debugLog("autoLoop", { phase: "exit", reason: "verification-pause" });
806
+ break;
807
+ }
808
+ if (verificationResult === "retry") {
809
+ // s.pendingVerificationRetry was set by runPostUnitVerification.
810
+ // Continue the loop — next iteration will inject the retry context into the prompt.
811
+ debugLog("autoLoop", { phase: "verification-retry", iteration });
812
+ continue;
813
+ }
814
+ // Post-verification processing (DB dual-write, hooks, triage, quick-tasks)
815
+ const postResult = await deps.postUnitPostVerification(postUnitCtx);
816
+ if (postResult === "stopped") {
817
+ debugLog("autoLoop", {
818
+ phase: "exit",
819
+ reason: "post-verification-stopped",
820
+ });
821
+ break;
822
+ }
823
+ if (postResult === "step-wizard") {
824
+ // Step mode — exit the loop (caller handles wizard)
825
+ debugLog("autoLoop", { phase: "exit", reason: "step-wizard" });
826
+ break;
827
+ }
828
+ // ── Sidecar drain: dispatch enqueued hooks/triage/quick-tasks ──
829
+ let sidecarBroke = false;
830
+ while (s.sidecarQueue.length > 0 && s.active) {
831
+ const item = s.sidecarQueue.shift();
832
+ debugLog("autoLoop", {
833
+ phase: "sidecar-dequeue",
834
+ kind: item.kind,
835
+ unitType: item.unitType,
836
+ unitId: item.unitId,
837
+ });
838
+ // Set up as current unit
839
+ const sidecarStartedAt = Date.now();
840
+ s.currentUnit = {
841
+ type: item.unitType,
842
+ id: item.unitId,
843
+ startedAt: sidecarStartedAt,
844
+ };
845
+ deps.writeUnitRuntimeRecord(s.basePath, item.unitType, item.unitId, sidecarStartedAt, {
846
+ phase: "dispatched",
847
+ wrapupWarningSent: false,
848
+ timeoutAt: null,
849
+ lastProgressAt: sidecarStartedAt,
850
+ progressCount: 0,
851
+ lastProgressKind: "dispatch",
852
+ });
853
+ // Model selection (handles hook model override)
854
+ await deps.selectAndApplyModel(ctx, pi, item.unitType, item.unitId, s.basePath, prefs, s.verbose, s.autoModeStartModel);
855
+ // Supervision
856
+ deps.clearUnitTimeout();
857
+ deps.startUnitSupervision({
858
+ s,
859
+ ctx,
860
+ pi,
861
+ unitType: item.unitType,
862
+ unitId: item.unitId,
863
+ prefs,
864
+ buildSnapshotOpts: () => deps.buildSnapshotOpts(item.unitType, item.unitId),
865
+ buildRecoveryContext: () => ({}),
866
+ pauseAuto: deps.pauseAuto,
867
+ });
868
+ // Write lock
869
+ const sidecarSessionFile = deps.getSessionFile(ctx);
870
+ deps.writeLock(deps.lockBase(), item.unitType, item.unitId, s.completedUnits.length, sidecarSessionFile);
871
+ // Execute via standard runUnit
872
+ const sidecarResult = await runUnit(ctx, pi, s, item.unitType, item.unitId, item.prompt, prefs);
873
+ deps.clearUnitTimeout();
874
+ if (sidecarResult.status === "cancelled") {
875
+ ctx.ui.notify(`Sidecar unit ${item.unitType} ${item.unitId} session cancelled. Stopping.`, "warning");
876
+ await deps.stopAuto(ctx, pi, "Sidecar session creation failed");
877
+ sidecarBroke = true;
878
+ break;
879
+ }
880
+ // Run pre-verification for the sidecar unit
881
+ const sidecarPreResult = await deps.postUnitPreVerification(postUnitCtx);
882
+ if (sidecarPreResult === "dispatched") {
883
+ // Pre-verification caused stop/pause
884
+ debugLog("autoLoop", {
885
+ phase: "exit",
886
+ reason: "sidecar-pre-verification-stop",
887
+ });
888
+ sidecarBroke = true;
889
+ break;
890
+ }
891
+ // Verification gate for non-hook sidecar units (triage, quick-tasks)
892
+ // Hook units are lightweight and don't need verification.
893
+ if (item.kind !== "hook") {
894
+ const sidecarVerification = await deps.runPostUnitVerification({ s, ctx, pi }, deps.pauseAuto);
895
+ if (sidecarVerification === "pause") {
896
+ debugLog("autoLoop", {
897
+ phase: "exit",
898
+ reason: "sidecar-verification-pause",
899
+ });
900
+ sidecarBroke = true;
901
+ break;
902
+ }
903
+ // "retry" for sidecars — skip retry, just continue (sidecar retries are not worth the complexity)
904
+ }
905
+ // Post-verification (may enqueue more sidecar items)
906
+ const sidecarPostResult = await deps.postUnitPostVerification(postUnitCtx);
907
+ if (sidecarPostResult === "stopped") {
908
+ debugLog("autoLoop", { phase: "exit", reason: "sidecar-stopped" });
909
+ sidecarBroke = true;
910
+ break;
911
+ }
912
+ if (sidecarPostResult === "step-wizard") {
913
+ debugLog("autoLoop", {
914
+ phase: "exit",
915
+ reason: "sidecar-step-wizard",
916
+ });
917
+ sidecarBroke = true;
918
+ break;
919
+ }
920
+ // "continue" — loop checks sidecarQueue again
921
+ }
922
+ if (sidecarBroke)
923
+ break;
924
+ consecutiveErrors = 0; // Iteration completed successfully
925
+ debugLog("autoLoop", { phase: "iteration-complete", iteration });
926
+ }
927
+ catch (loopErr) {
928
+ // ── Blanket catch: absorb unexpected exceptions, apply graduated recovery ──
929
+ consecutiveErrors++;
930
+ const msg = loopErr instanceof Error ? loopErr.message : String(loopErr);
931
+ debugLog("autoLoop", {
932
+ phase: "iteration-error",
933
+ iteration,
934
+ consecutiveErrors,
935
+ error: msg,
936
+ });
937
+ if (consecutiveErrors >= 3) {
938
+ // 3+ consecutive: hard stop — something is fundamentally broken
939
+ ctx.ui.notify(`Auto-mode stopped: ${consecutiveErrors} consecutive iteration failures. Last: ${msg}`, "error");
940
+ await deps.stopAuto(ctx, pi, `${consecutiveErrors} consecutive iteration failures`);
941
+ break;
942
+ }
943
+ else if (consecutiveErrors === 2) {
944
+ // 2nd consecutive: try invalidating caches + re-deriving state
945
+ ctx.ui.notify(`Iteration error (attempt ${consecutiveErrors}): ${msg}. Invalidating caches and retrying.`, "warning");
946
+ deps.invalidateAllCaches();
947
+ }
948
+ else {
949
+ // 1st error: log and retry — transient failures happen
950
+ ctx.ui.notify(`Iteration error: ${msg}. Retrying.`, "warning");
951
+ }
952
+ }
953
+ }
954
+ _activeSession = null;
955
+ debugLog("autoLoop", { phase: "exit", totalIterations: iteration });
956
+ }