gsd-pi 2.77.0-dev.eaa4973bc → 2.78.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 (655) hide show
  1. package/README.md +1 -1
  2. package/dist/claude-cli-check.js +5 -1
  3. package/dist/headless.js +49 -4
  4. package/dist/loader.js +0 -0
  5. package/dist/resource-loader.d.ts +40 -0
  6. package/dist/resource-loader.js +32 -13
  7. package/dist/resources/extensions/browser-tools/capture.js +9 -0
  8. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  9. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  10. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  11. package/dist/resources/extensions/browser-tools/tools/forms.js +5 -1
  12. package/dist/resources/extensions/browser-tools/tools/intent.js +5 -1
  13. package/dist/resources/extensions/claude-code-cli/readiness.js +5 -1
  14. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +481 -17
  15. package/dist/resources/extensions/github-sync/templates.js +103 -0
  16. package/dist/resources/extensions/google-search/index.js +3 -2
  17. package/dist/resources/extensions/gsd/auto/loop.js +124 -2
  18. package/dist/resources/extensions/gsd/auto/phases.js +57 -39
  19. package/dist/resources/extensions/gsd/auto/session.js +6 -2
  20. package/dist/resources/extensions/gsd/auto-dispatch.js +142 -29
  21. package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
  22. package/dist/resources/extensions/gsd/auto-post-unit.js +150 -64
  23. package/dist/resources/extensions/gsd/auto-prompts.js +372 -104
  24. package/dist/resources/extensions/gsd/auto-recovery.js +197 -48
  25. package/dist/resources/extensions/gsd/auto-start.js +107 -29
  26. package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
  27. package/dist/resources/extensions/gsd/auto-worktree.js +122 -26
  28. package/dist/resources/extensions/gsd/auto.js +76 -21
  29. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +19 -1
  30. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
  31. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +3 -6
  32. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
  33. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
  34. package/dist/resources/extensions/gsd/component-loader.js +447 -0
  35. package/dist/resources/extensions/gsd/component-types.js +69 -0
  36. package/dist/resources/extensions/gsd/context-store.js +23 -7
  37. package/dist/resources/extensions/gsd/detection.js +49 -1
  38. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -17
  39. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  40. package/dist/resources/extensions/gsd/forensics.js +106 -0
  41. package/dist/resources/extensions/gsd/gate-registry.js +2 -2
  42. package/dist/resources/extensions/gsd/git-constants.js +28 -1
  43. package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
  44. package/dist/resources/extensions/gsd/git-service.js +126 -2
  45. package/dist/resources/extensions/gsd/gsd-db.js +6 -3
  46. package/dist/resources/extensions/gsd/guided-flow.js +39 -13
  47. package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
  48. package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
  49. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  50. package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
  51. package/dist/resources/extensions/gsd/model-router.js +6 -0
  52. package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
  53. package/dist/resources/extensions/gsd/preferences-validation.js +23 -0
  54. package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +4 -0
  55. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  56. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  57. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  58. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  59. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
  60. package/dist/resources/extensions/gsd/service-tier.js +5 -2
  61. package/dist/resources/extensions/gsd/session-lock.js +19 -10
  62. package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
  63. package/dist/resources/extensions/gsd/slice-cadence.js +238 -0
  64. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
  65. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  66. package/dist/resources/extensions/gsd/state.js +69 -58
  67. package/dist/resources/extensions/gsd/sync-lock.js +98 -42
  68. package/dist/resources/extensions/gsd/tools/validate-milestone.js +7 -2
  69. package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
  70. package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
  71. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  72. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  73. package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
  74. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  75. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  76. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  77. package/dist/resources/extensions/gsd/uok/plan-v2.js +10 -4
  78. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  79. package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
  80. package/dist/resources/extensions/gsd/worktree-manager.js +85 -8
  81. package/dist/resources/extensions/gsd/worktree-resolver.js +86 -7
  82. package/dist/resources/extensions/gsd/worktree-telemetry.js +198 -0
  83. package/dist/resources/extensions/mcp-client/index.js +3 -1
  84. package/dist/resources/extensions/ollama/index.js +5 -1
  85. package/dist/resources/extensions/remote-questions/manager.js +11 -5
  86. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  87. package/dist/web/standalone/.next/BUILD_ID +1 -1
  88. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  89. package/dist/web/standalone/.next/build-manifest.json +3 -3
  90. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  91. package/dist/web/standalone/.next/required-server-files.json +3 -3
  92. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  93. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  103. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  107. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  109. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  113. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  115. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  119. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  131. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  133. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  135. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  151. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  155. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  157. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  161. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  165. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  167. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  171. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  172. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  175. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  177. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  179. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  181. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  183. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  185. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  187. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  190. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  196. package/dist/web/standalone/.next/server/app/index.html +1 -1
  197. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  198. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  199. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  200. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  201. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  202. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  203. package/dist/web/standalone/.next/server/app/page.js +2 -2
  204. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  205. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  206. package/dist/web/standalone/.next/server/chunks/1926.js +1 -1
  207. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  208. package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
  209. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  210. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  211. package/dist/web/standalone/.next/server/middleware.js +2 -2
  212. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  213. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  214. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  215. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  216. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  217. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-f2a7482d42a5614b.js → page-2f24283c162b6ab3.js} +1 -1
  218. package/dist/web/standalone/.next/static/chunks/app/{layout-a16c7a7ecdf0c2cf.js → layout-9ecfd95f343793f0.js} +1 -1
  219. package/dist/web/standalone/.next/static/chunks/app/page-151349214571e2b6.js +1 -0
  220. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +1 -0
  221. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +1 -0
  222. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  223. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  224. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  225. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  226. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  227. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  228. package/dist/web/standalone/server.js +1 -1
  229. package/package.json +2 -3
  230. package/packages/daemon/package.json +2 -2
  231. package/packages/daemon/src/logger.ts +4 -3
  232. package/packages/mcp-server/dist/server.d.ts +24 -0
  233. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  234. package/packages/mcp-server/dist/server.js +88 -87
  235. package/packages/mcp-server/dist/server.js.map +1 -1
  236. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  237. package/packages/mcp-server/dist/workflow-tools.js +15 -6
  238. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  239. package/packages/mcp-server/package.json +2 -2
  240. package/packages/mcp-server/src/mcp-server.test.ts +25 -3
  241. package/packages/mcp-server/src/readers/graph.test.ts +87 -15
  242. package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
  243. package/packages/mcp-server/src/server.ts +131 -105
  244. package/packages/mcp-server/src/workflow-tools.test.ts +85 -0
  245. package/packages/mcp-server/src/workflow-tools.ts +19 -6
  246. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  247. package/packages/native/package.json +2 -2
  248. package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
  249. package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
  250. package/packages/native/src/__tests__/ps.test.mjs +14 -8
  251. package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
  252. package/packages/native/src/__tests__/truncate.test.mjs +17 -2
  253. package/packages/pi-agent-core/package.json +1 -1
  254. package/packages/pi-agent-core/src/agent-loop.test.ts +5 -15
  255. package/packages/pi-agent-core/src/agent.test.ts +96 -102
  256. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  257. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
  258. package/packages/pi-ai/dist/models/capability-patches.js +9 -2
  259. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
  260. package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
  261. package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
  262. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
  263. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
  264. package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
  265. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
  266. package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
  267. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
  268. package/packages/pi-ai/dist/models/generated/openai.js +17 -0
  269. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
  270. package/packages/pi-ai/dist/models.generated.test.js +43 -70
  271. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  272. package/packages/pi-ai/dist/models.test.js +36 -11
  273. package/packages/pi-ai/dist/models.test.js.map +1 -1
  274. package/packages/pi-ai/package.json +1 -1
  275. package/packages/pi-ai/scripts/generate-models.ts +44 -0
  276. package/packages/pi-ai/src/models/capability-patches.ts +10 -2
  277. package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
  278. package/packages/pi-ai/src/models/generated/openai.ts +17 -0
  279. package/packages/pi-ai/src/models.generated.test.ts +46 -73
  280. package/packages/pi-ai/src/models.test.ts +48 -11
  281. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  282. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
  283. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  284. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
  285. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
  286. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
  287. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  288. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +5 -0
  289. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  290. package/packages/pi-coding-agent/dist/core/extensions/loader.js +61 -0
  291. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  292. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +30 -4
  293. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  294. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +17 -0
  295. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  296. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
  297. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
  298. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  299. package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
  300. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  301. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
  302. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
  304. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
  305. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
  306. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
  307. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
  308. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  309. package/packages/pi-coding-agent/dist/core/system-prompt.js +16 -2
  310. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  311. package/packages/pi-coding-agent/dist/index.d.ts +1 -0
  312. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  313. package/packages/pi-coding-agent/dist/index.js +1 -0
  314. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  315. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +36 -5
  316. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  317. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
  318. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
  319. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  320. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +30 -12
  321. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  322. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  323. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
  324. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  325. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +125 -0
  326. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
  328. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  329. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  330. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
  331. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  332. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
  333. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  334. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
  335. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
  336. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
  337. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
  338. package/packages/pi-coding-agent/package.json +1 -1
  339. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
  340. package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
  341. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
  342. package/packages/pi-coding-agent/src/core/extensions/loader.ts +58 -0
  343. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +35 -4
  344. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +20 -0
  345. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
  346. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
  347. package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
  348. package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
  349. package/packages/pi-coding-agent/src/core/system-prompt.ts +35 -1
  350. package/packages/pi-coding-agent/src/index.ts +1 -0
  351. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +49 -3
  352. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
  353. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +48 -9
  354. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +146 -1
  355. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
  356. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
  357. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
  358. package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
  359. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  360. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +18 -8
  361. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  362. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
  363. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
  364. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +37 -11
  365. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  366. package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
  367. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  368. package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
  369. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  370. package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
  371. package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
  372. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
  373. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
  374. package/packages/pi-tui/dist/components/editor.d.ts +14 -0
  375. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  376. package/packages/pi-tui/dist/components/editor.js +19 -0
  377. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  378. package/packages/pi-tui/dist/components/image.test.js +6 -5
  379. package/packages/pi-tui/dist/components/image.test.js.map +1 -1
  380. package/packages/pi-tui/dist/editor-component.d.ts +2 -0
  381. package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
  382. package/packages/pi-tui/dist/editor-component.js.map +1 -1
  383. package/packages/pi-tui/package.json +1 -1
  384. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +24 -8
  385. package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
  386. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +42 -11
  387. package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
  388. package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
  389. package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
  390. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
  391. package/packages/pi-tui/src/components/editor.ts +22 -0
  392. package/packages/pi-tui/src/components/image.test.ts +10 -5
  393. package/packages/pi-tui/src/editor-component.ts +3 -0
  394. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  395. package/packages/rpc-client/dist/rpc-client.test.js +101 -51
  396. package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
  397. package/packages/rpc-client/package.json +1 -1
  398. package/packages/rpc-client/src/rpc-client.test.ts +109 -52
  399. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  400. package/pkg/package.json +1 -1
  401. package/scripts/install.js +15 -1
  402. package/src/resources/extensions/browser-tools/capture.ts +12 -0
  403. package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  404. package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  405. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  406. package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
  407. package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
  408. package/src/resources/extensions/claude-code-cli/readiness.ts +5 -1
  409. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +518 -19
  410. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +919 -75
  411. package/src/resources/extensions/github-sync/templates.ts +151 -0
  412. package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
  413. package/src/resources/extensions/github-sync/tests/templates.test.ts +92 -1
  414. package/src/resources/extensions/google-search/index.ts +3 -2
  415. package/src/resources/extensions/gsd/auto/loop.ts +142 -2
  416. package/src/resources/extensions/gsd/auto/phases.ts +62 -38
  417. package/src/resources/extensions/gsd/auto/session.ts +7 -2
  418. package/src/resources/extensions/gsd/auto-dispatch.ts +156 -29
  419. package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
  420. package/src/resources/extensions/gsd/auto-post-unit.ts +163 -73
  421. package/src/resources/extensions/gsd/auto-prompts.ts +385 -93
  422. package/src/resources/extensions/gsd/auto-recovery.ts +230 -51
  423. package/src/resources/extensions/gsd/auto-start.ts +127 -9
  424. package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
  425. package/src/resources/extensions/gsd/auto-worktree.ts +130 -26
  426. package/src/resources/extensions/gsd/auto.ts +90 -23
  427. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +20 -1
  428. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
  429. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +3 -7
  430. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
  431. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
  432. package/src/resources/extensions/gsd/component-loader.ts +598 -0
  433. package/src/resources/extensions/gsd/component-types.ts +362 -0
  434. package/src/resources/extensions/gsd/context-store.ts +25 -8
  435. package/src/resources/extensions/gsd/detection.ts +58 -1
  436. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -20
  437. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  438. package/src/resources/extensions/gsd/forensics.ts +118 -1
  439. package/src/resources/extensions/gsd/gate-registry.ts +2 -2
  440. package/src/resources/extensions/gsd/git-constants.ts +30 -1
  441. package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
  442. package/src/resources/extensions/gsd/git-service.ts +149 -2
  443. package/src/resources/extensions/gsd/gsd-db.ts +6 -3
  444. package/src/resources/extensions/gsd/guided-flow.ts +57 -14
  445. package/src/resources/extensions/gsd/journal.ts +11 -1
  446. package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
  447. package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
  448. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  449. package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
  450. package/src/resources/extensions/gsd/model-router.ts +6 -0
  451. package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
  452. package/src/resources/extensions/gsd/preferences-validation.ts +21 -0
  453. package/src/resources/extensions/gsd/prompt-cache-optimizer.ts +4 -0
  454. package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  455. package/src/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  456. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  457. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  458. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
  459. package/src/resources/extensions/gsd/service-tier.ts +5 -2
  460. package/src/resources/extensions/gsd/session-lock.ts +20 -10
  461. package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
  462. package/src/resources/extensions/gsd/slice-cadence.ts +299 -0
  463. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
  464. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  465. package/src/resources/extensions/gsd/state.ts +76 -66
  466. package/src/resources/extensions/gsd/sync-lock.ts +97 -39
  467. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
  468. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +2 -1
  469. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
  470. package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
  471. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +133 -292
  472. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
  473. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
  474. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
  475. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +93 -0
  476. package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +4 -1
  477. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -194
  478. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +3 -2
  479. package/src/resources/extensions/gsd/tests/auto-start-cold-db-bootstrap.test.ts +2 -2
  480. package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
  481. package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +2 -2
  482. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +3 -2
  483. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +3 -2
  484. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +2 -1
  485. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
  486. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +108 -0
  487. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
  488. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
  489. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
  490. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +2 -1
  491. package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
  492. package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
  493. package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
  494. package/src/resources/extensions/gsd/tests/context-store.test.ts +79 -0
  495. package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +2 -1
  496. package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
  497. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +159 -0
  498. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  499. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
  500. package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
  501. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
  502. package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -4
  503. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +2 -1
  504. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +2 -1
  505. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +5 -0
  506. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  507. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  508. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
  509. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +4 -3
  510. package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +4 -3
  511. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  512. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
  513. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +8 -104
  514. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
  515. package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
  516. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +14 -4
  517. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  518. package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
  519. package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -56
  520. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
  521. package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
  522. package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
  523. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
  524. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
  525. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
  526. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +9 -3
  527. package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
  528. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +64 -0
  529. package/src/resources/extensions/gsd/tests/knowledge.test.ts +93 -1
  530. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +8 -37
  531. package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
  532. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -55
  533. package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
  534. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  535. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
  536. package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
  537. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -48
  538. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
  539. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +59 -2
  540. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -130
  541. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
  542. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
  543. package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +2 -1
  544. package/src/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +12 -0
  545. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +15 -4
  546. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +23 -24
  547. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +32 -0
  548. package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +3 -2
  549. package/src/resources/extensions/gsd/tests/queued-discuss-fast-path.test.ts +4 -5
  550. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
  551. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
  552. package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
  553. package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
  554. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
  555. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +6 -3
  556. package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
  557. package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
  558. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
  559. package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +3 -2
  560. package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
  561. package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
  562. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
  563. package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
  564. package/src/resources/extensions/gsd/tests/slice-cadence.test.ts +242 -0
  565. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +3 -2
  566. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
  567. package/src/resources/extensions/gsd/tests/smart-entry-draft.test.ts +2 -1
  568. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  569. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +29 -5
  570. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  571. package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +3 -3
  572. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
  573. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +7 -6
  574. package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
  575. package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
  576. package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +4 -3
  577. package/src/resources/extensions/gsd/tests/test-helpers.test.ts +98 -0
  578. package/src/resources/extensions/gsd/tests/test-helpers.ts +153 -0
  579. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -1
  580. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
  581. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
  582. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
  583. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
  584. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  585. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  586. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
  587. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
  588. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  589. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  590. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +19 -2
  591. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  592. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +12 -0
  593. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -80
  594. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
  595. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
  596. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
  597. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
  598. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
  599. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
  600. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
  601. package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
  602. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +210 -0
  603. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
  604. package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
  605. package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
  606. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
  607. package/src/resources/extensions/gsd/tools/validate-milestone.ts +8 -2
  608. package/src/resources/extensions/gsd/types.ts +3 -3
  609. package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
  610. package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
  611. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  612. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  613. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  614. package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
  615. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  616. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  617. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  618. package/src/resources/extensions/gsd/uok/plan-v2.ts +13 -5
  619. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  620. package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
  621. package/src/resources/extensions/gsd/worktree-manager.ts +108 -7
  622. package/src/resources/extensions/gsd/worktree-resolver.ts +96 -9
  623. package/src/resources/extensions/gsd/worktree-telemetry.ts +322 -0
  624. package/src/resources/extensions/mcp-client/index.ts +3 -1
  625. package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
  626. package/src/resources/extensions/ollama/index.ts +5 -1
  627. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
  628. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
  629. package/src/resources/extensions/remote-questions/manager.ts +36 -4
  630. package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
  631. package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
  632. package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
  633. package/dist/web/standalone/.next/static/chunks/app/page-5b113fd32bc2a1c3.js +0 -1
  634. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +0 -1
  635. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +0 -1
  636. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
  637. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
  638. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
  639. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
  640. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
  641. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -143
  642. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -157
  643. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
  644. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
  645. package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
  646. package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
  647. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
  648. package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -74
  649. package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
  650. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
  651. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
  652. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -125
  653. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -42
  654. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → C1zT2kEfoLhDdbWPWKrXd}/_buildManifest.js +0 -0
  655. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → C1zT2kEfoLhDdbWPWKrXd}/_ssgManifest.js +0 -0
@@ -13,6 +13,75 @@ import { logWarning } from "./workflow-logger.js";
13
13
  import { resolveUokFlags } from "./uok/flags.js";
14
14
  import { applyModelPolicyFilter } from "./uok/model-policy.js";
15
15
  import { isModelBlocked } from "./blocked-models.js";
16
+ import { getRequiredWorkflowToolsForAutoUnit } from "./workflow-mcp.js";
17
+ /**
18
+ * Thrown when the model-policy gate rejects every candidate model for a unit
19
+ * dispatch (#4959 / #4681 / #4850). The auto-loop catches this specifically
20
+ * to classify the unit as `blocked` rather than counting it as a retryable
21
+ * iteration error — pre-send policy denial is a configuration problem, not a
22
+ * transient runtime failure, so retrying just burns the consecutive-error
23
+ * budget toward a hard stop.
24
+ */
25
+ export class ModelPolicyDispatchBlockedError extends Error {
26
+ unitType;
27
+ unitId;
28
+ reasons;
29
+ constructor(unitType, unitId, reasons) {
30
+ const summary = reasons.length === 0
31
+ ? "no candidate models"
32
+ : reasons
33
+ .slice(0, 4)
34
+ .map((r) => `${r.provider}/${r.modelId} (${r.reason})`)
35
+ .join("; ");
36
+ super(`Model policy denied dispatch for ${unitType}/${unitId} before prompt send. Rejected: ${summary}`);
37
+ this.name = "ModelPolicyDispatchBlockedError";
38
+ this.unitType = unitType;
39
+ this.unitId = unitId;
40
+ this.reasons = reasons;
41
+ }
42
+ }
43
+ // Baseline active-tool set per-`pi` instance, captured the first time
44
+ // `selectAndApplyModel` runs against that instance during an auto session
45
+ // and re-applied before each subsequent dispatch. WeakMap so that test
46
+ // fakes / disposed sessions are garbage-collected normally. See
47
+ // #4959 / #4681 cross-unit poisoning notes at the call site below.
48
+ //
49
+ // LIFECYCLE: the baseline is tied to a single auto session, NOT to the
50
+ // lifetime of the `pi` instance (which can outlive many auto runs and have
51
+ // the user mutate tools between them). `clearToolBaseline` MUST be called
52
+ // at auto start AND auto stop so that a second `/gsd auto` run on the same
53
+ // `pi` does not silently restore a stale snapshot from the prior run and
54
+ // undo any tool changes the user made between sessions.
55
+ const TOOL_BASELINE = new WeakMap();
56
+ /**
57
+ * Drop the captured tool baseline for `pi` so the next `selectAndApplyModel`
58
+ * call re-captures from the live active set. Wired into `startAuto` and
59
+ * `stopAuto` in `auto.ts` to bound the baseline to a single auto session.
60
+ *
61
+ * Safe to call when no baseline is recorded (no-op).
62
+ */
63
+ export function clearToolBaseline(pi) {
64
+ TOOL_BASELINE.delete(pi);
65
+ }
66
+ function restoreToolBaseline(pi) {
67
+ const key = pi;
68
+ const baseline = TOOL_BASELINE.get(key);
69
+ if (baseline === undefined) {
70
+ // First call: capture the canonical pre-dispatch tool set. At auto-mode
71
+ // start the active set has not yet been narrowed for any provider.
72
+ // Guarded against test fakes that omit getActiveTools — record an empty
73
+ // baseline so subsequent calls don't keep re-probing.
74
+ const initial = typeof pi.getActiveTools === "function" ? pi.getActiveTools() : [];
75
+ TOOL_BASELINE.set(key, [...initial]);
76
+ return;
77
+ }
78
+ // Restore baseline before the next unit reads getActiveTools / applies
79
+ // post-selection adjustToolSet. Older fakes that omit setActiveTools are
80
+ // tolerated — the test asserts call order on real fakes.
81
+ if (typeof pi.setActiveTools === "function") {
82
+ pi.setActiveTools([...baseline]);
83
+ }
84
+ }
16
85
  function reapplyThinkingLevel(pi, level) {
17
86
  if (!level)
18
87
  return;
@@ -86,6 +155,29 @@ autoModeStartThinkingLevel) {
86
155
  : resolvePreferredModelConfig(unitType, autoModeStartModel, isAutoMode);
87
156
  let routing = null;
88
157
  let appliedModel = null;
158
+ // ── Restore active-tool baseline before policy evaluation (#4959, #4681, #4850) ──
159
+ // Per-unit narrowing at the bottom of this function (line ~417) calls
160
+ // `pi.setActiveTools(finalToolNames)` and monotonically narrows the active
161
+ // set across units. Without restoration, a previously-dispatched unit on a
162
+ // narrow-API provider (e.g. openai-completions) leaves the active set
163
+ // missing tools that the next unit's selected model fully supports, but
164
+ // `pi.getActiveTools()` snapshot-as-hard-gate (the old behaviour) blocked
165
+ // dispatch with "tool policy denied" anyway.
166
+ //
167
+ // The baseline is captured once per `pi` instance via a WeakMap and
168
+ // re-applied here so each unit starts from a clean slate. Soft adaptation
169
+ // (adjustToolSet at the bottom of this function) still trims for the
170
+ // selected model.
171
+ //
172
+ // Auto-mode only (#4965): `guided-flow.ts:dispatchWorkflow` also calls
173
+ // `selectAndApplyModel` with `isAutoMode=false`. Guided-flow has its own
174
+ // narrow/restore via discuss-tool-scoping (guided-flow.ts:587-622) and no
175
+ // baseline-clear hook of its own, so an unconditional restore here would
176
+ // resurrect an auto-era baseline on guided-flow dispatches — silently
177
+ // overwriting any tool changes made interactively between auto sessions.
178
+ // The baseline is structurally an auto-mode concept; gate it accordingly.
179
+ if (isAutoMode)
180
+ restoreToolBaseline(pi);
89
181
  if (modelConfig) {
90
182
  const availableModels = ctx.modelRegistry.getAvailable();
91
183
  const modelPolicyTraceId = `model:${ctx.sessionManager.getSessionId()}:${Date.now()}`;
@@ -114,7 +206,16 @@ autoModeStartThinkingLevel) {
114
206
  const taskMetadataForPolicy = unitType === "execute-task"
115
207
  ? extractTaskMetadata(unitId, basePath)
116
208
  : undefined;
209
+ let policyDenyReasons = [];
117
210
  if (uokFlags.modelPolicy) {
211
+ // Use the workflow-spec required-tool subset for the unit type rather
212
+ // than the live `pi.getActiveTools()` snapshot (#4959). The active set
213
+ // is poisoned by per-unit narrowing for narrow-API providers — using it
214
+ // as a hard gate promotes soft adaptation (adjustToolSet at line ~417)
215
+ // into a layering violation that throws before dispatch. The smaller
216
+ // workflow-required subset reflects what the unit actually needs; soft
217
+ // adaptation post-selection still trims provider-incompatible tools.
218
+ const requiredTools = getRequiredWorkflowToolsForAutoUnit(unitType);
118
219
  const policy = applyModelPolicyFilter(availableModels, {
119
220
  basePath,
120
221
  traceId: modelPolicyTraceId,
@@ -123,12 +224,15 @@ autoModeStartThinkingLevel) {
123
224
  taskMetadata: taskMetadataForPolicy,
124
225
  currentProvider: ctx.model?.provider,
125
226
  allowCrossProvider: routingConfig.cross_provider !== false,
126
- requiredTools: pi.getActiveTools(),
227
+ requiredTools,
127
228
  });
128
229
  routingEligibleModels = policy.eligible;
129
230
  policyAllowedModelKeys = new Set(policy.eligible.map((m) => `${m.provider.toLowerCase()}/${m.id.toLowerCase()}`));
231
+ policyDenyReasons = policy.decisions
232
+ .filter((d) => !d.allowed)
233
+ .map((d) => ({ provider: d.provider, modelId: d.modelId, reason: d.reason }));
130
234
  if (routingEligibleModels.length === 0) {
131
- throw new Error(`Model policy denied all candidate models for ${unitType}/${unitId}`);
235
+ throw new ModelPolicyDispatchBlockedError(unitType, unitId, policyDenyReasons);
132
236
  }
133
237
  }
134
238
  // Disable routing for flat-rate providers like GitHub Copilot (#3453).
@@ -169,7 +273,11 @@ autoModeStartThinkingLevel) {
169
273
  if (shouldClassify) {
170
274
  let classification = classifyUnitComplexity(unitType, unitId, basePath, budgetPct, taskMetadataForPolicy);
171
275
  const availableModelIds = routingEligibleModels.map(m => `${m.provider}/${m.id}`);
172
- // Escalate tier on retry when escalate_on_failure is enabled (default: true)
276
+ // Escalate tier on retry when escalate_on_failure is enabled (default: true).
277
+ // #4973: Deterministic policy errors are short-circuited at the postUnit
278
+ // level (auto-post-unit.ts writes a placeholder and returns "continue"),
279
+ // so this code path only runs for legitimate model-quality retries where
280
+ // tier escalation is the right response.
173
281
  if (retryContext?.isRetry &&
174
282
  retryContext.previousTier &&
175
283
  routingConfig.escalate_on_failure !== false) {
@@ -179,6 +287,18 @@ autoModeStartThinkingLevel) {
179
287
  // Always notify on tier escalation — model changes should be visible (#3962)
180
288
  ctx.ui.notify(`Tier escalation: ${retryContext.previousTier} → ${escalated} (retry after failure)`, "info");
181
289
  }
290
+ else {
291
+ // #4973: Already at max tier — keep previousTier rather than letting
292
+ // fresh classification silently downgrade the model back to a lower tier.
293
+ // Without this, a light-start unit on retry 3 would revert to the light
294
+ // model after escalating to heavy on retries 1 and 2.
295
+ const tierOrder = { light: 0, standard: 1, heavy: 2 };
296
+ const prevOrder = tierOrder[retryContext.previousTier] ?? 0;
297
+ const freshOrder = tierOrder[classification.tier] ?? 0;
298
+ if (prevOrder > freshOrder) {
299
+ classification = { ...classification, tier: retryContext.previousTier, reason: "retained escalated tier from retry" };
300
+ }
301
+ }
182
302
  }
183
303
  // Load user capability overrides from preferences (D-17: deep-merged with built-in profiles)
184
304
  const capabilityOverrides = loadCapabilityOverrides(prefs ?? {});
@@ -335,7 +455,7 @@ autoModeStartThinkingLevel) {
335
455
  }
336
456
  }
337
457
  if (uokFlags.modelPolicy && policyAllowedModelKeys && !attemptedPolicyEligible) {
338
- throw new Error(`Model policy denied dispatch for ${unitType}/${unitId} before prompt send`);
458
+ throw new ModelPolicyDispatchBlockedError(unitType, unitId, policyDenyReasons);
339
459
  }
340
460
  }
341
461
  else if (autoModeStartModel) {
@@ -47,8 +47,18 @@ import { UokGateRunner } from "./uok/gate-runner.js";
47
47
  import { writeTurnGitTransaction } from "./uok/gitops.js";
48
48
  import { isClosedStatus } from "./status-guards.js";
49
49
  import { detectAbandonMilestone } from "./abandon-detect.js";
50
+ import { isDeterministicPolicyError } from "./auto-tool-tracking.js";
50
51
  /** Maximum verification retry attempts before escalating to blocker placeholder (#2653). */
51
52
  const MAX_VERIFICATION_RETRIES = 3;
53
+ /** Keep failure toasts short while still showing concrete examples. */
54
+ const MAX_NOTIFICATION_DETAILS = 3;
55
+ const NOTIFICATION_BULLET = "•";
56
+ function formatPreExecutionCheckDetail(check) {
57
+ const category = check.category?.trim() || "unknown category";
58
+ const target = check.target?.trim() || "unknown target";
59
+ const message = check.message.split(/\r?\n/, 1)[0]?.trim() || "No details provided";
60
+ return ` ${NOTIFICATION_BULLET} [${category}] ${target}: ${message}`;
61
+ }
52
62
  const COMPLETE_MILESTONE_DB_SETTLE_MS = 1500;
53
63
  const COMPLETE_MILESTONE_DB_SETTLE_POLL_MS = 100;
54
64
  async function waitForMilestoneDbClose(mid) {
@@ -88,7 +98,7 @@ const LIFECYCLE_ONLY_UNITS = new Set([
88
98
  ]);
89
99
  import { describeNextUnit, } from "./auto-dashboard.js";
90
100
  import { existsSync, unlinkSync } from "node:fs";
91
- import { join } from "node:path";
101
+ import { join, relative } from "node:path";
92
102
  import { _resetHasChangesCache } from "./native-git-bridge.js";
93
103
  import { autoCommitCurrentBranch } from "./worktree.js";
94
104
  /**
@@ -200,6 +210,13 @@ export function detectRogueFileWrites(unitType, unitId, basePath) {
200
210
  }
201
211
  return rogues;
202
212
  }
213
+ /**
214
+ * Maximum number of times to retry a unit whose expected artifact is missing
215
+ * after execution. Matches the bounded pattern used by runPostUnitVerification
216
+ * in auto-verification.ts. Exceeding this limit pauses auto-mode instead of
217
+ * looping indefinitely (#2007).
218
+ */
219
+ const MAX_ARTIFACT_VERIFICATION_RETRIES = 3;
203
220
  export const STEP_COMPLETE_FALLBACK_MESSAGE = "Step complete. Run /clear, then /gsd to continue (or /gsd auto to run continuously).";
204
221
  export function buildStepCompleteMessage(nextState) {
205
222
  if (nextState.phase === "complete") {
@@ -521,6 +538,85 @@ export async function postUnitPreVerification(pctx, opts) {
521
538
  clearReactiveState(s.basePath, mid, sid);
522
539
  }
523
540
  });
541
+ // #4765 — slice-cadence collapse. When `git.collapse_cadence: "slice"`
542
+ // is set, squash-merge the slice's commits from the milestone branch
543
+ // onto main right here, so orphan risk shrinks from milestone-size to
544
+ // slice-size. Only runs in worktree isolation mode — the feature needs
545
+ // a milestone branch to squash from.
546
+ let sliceMergeStopped = false;
547
+ await runSafely("postUnit", "slice-cadence-merge", async () => {
548
+ const prefsResult = loadEffectiveGSDPreferences(s.basePath);
549
+ const prefs = prefsResult?.preferences;
550
+ const { getCollapseCadence, mergeSliceToMain } = await import("./slice-cadence.js");
551
+ if (getCollapseCadence(prefs) !== "slice")
552
+ return;
553
+ if (prefs?.git?.isolation !== "worktree")
554
+ return;
555
+ if (s.isolationDegraded)
556
+ return;
557
+ const projectRoot = s.originalBasePath || s.basePath;
558
+ const { milestone: mid, slice: sid } = parseUnitId(unit.id);
559
+ if (!mid || !sid)
560
+ return;
561
+ // Record the milestone start SHA before the first slice merge, so
562
+ // resquashMilestoneOnMain has a target at milestone completion.
563
+ // Resolve main branch dynamically — hard-coding "main" breaks repos
564
+ // that use "master" or a custom default branch.
565
+ if (!s.milestoneStartShas.has(mid)) {
566
+ try {
567
+ const { nativeDetectMainBranch } = await import("./native-git-bridge.js");
568
+ const mainBranch = nativeDetectMainBranch(projectRoot);
569
+ const { execFileSync } = await import("node:child_process");
570
+ const sha = execFileSync("git", ["rev-parse", mainBranch], {
571
+ cwd: projectRoot, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8",
572
+ }).trim();
573
+ if (sha)
574
+ s.milestoneStartShas.set(mid, sha);
575
+ }
576
+ catch (err) {
577
+ logWarning("engine", `slice-cadence: failed to record milestone start SHA: ${err instanceof Error ? err.message : String(err)}`);
578
+ }
579
+ }
580
+ try {
581
+ const result = mergeSliceToMain(projectRoot, mid, sid);
582
+ if (result.skipped) {
583
+ logWarning("engine", `slice-cadence: merge skipped for ${sid} — ${result.skippedReason}`);
584
+ return;
585
+ }
586
+ ctx.ui.notify(`slice-cadence: ${sid} merged to main (${result.durationMs}ms).`, "info");
587
+ }
588
+ catch (err) {
589
+ const { MergeConflictError } = await import("./git-service.js");
590
+ if (err instanceof MergeConflictError) {
591
+ ctx.ui.notify(`slice-cadence merge conflict in ${sid}: ${err.conflictedFiles.join(", ")}. ` +
592
+ `Resolve manually on main and run \`/gsd auto\` to resume.`, "error");
593
+ // Stop auto AND signal the outer postUnit flow to exit early.
594
+ // Without the flag, subsequent hooks (triage, rogue detection,
595
+ // DB writes) would keep running against a conflicted main
596
+ // checkout after the loop was already told to stop.
597
+ const { stopAuto } = await import("./auto.js");
598
+ await stopAuto(ctx, undefined, `slice-merge-conflict on ${sid}`);
599
+ sliceMergeStopped = true;
600
+ return;
601
+ }
602
+ logError("engine", `slice-cadence merge failed for ${sid}`, {
603
+ error: err instanceof Error ? err.message : String(err),
604
+ });
605
+ // Non-conflict failures (dirty main, rev-walk error, etc.) can
606
+ // leave the checkout in an unexpected state. Stop auto-mode so
607
+ // the next slice doesn't dispatch on top of it.
608
+ const { stopAuto } = await import("./auto.js");
609
+ await stopAuto(ctx, undefined, `slice-merge-error on ${sid}`);
610
+ sliceMergeStopped = true;
611
+ }
612
+ });
613
+ // Exit early after stopAuto so the rest of post-unit processing
614
+ // (triage, rogue detection, hook dispatch, DB writes) doesn't run
615
+ // against a conflicted main checkout. Return "dispatched" to match
616
+ // the convention used by other stop/pauseAuto paths in this function
617
+ // (see signal handling earlier: stop/pause also return "dispatched").
618
+ if (sliceMergeStopped)
619
+ return "dispatched";
524
620
  }
525
621
  // Post-triage: execute actionable resolutions
526
622
  if (s.currentUnit.type === "triage-captures") {
@@ -703,24 +799,38 @@ export async function postUnitPreVerification(pctx, opts) {
703
799
  // When artifact verification fails for a unit type that has a known expected
704
800
  // artifact, return "retry" so the caller re-dispatches with failure context
705
801
  // instead of blindly re-dispatching the same unit (#1571).
706
- // After MAX_VERIFICATION_RETRIES, escalate to writeBlockerPlaceholder so the
707
- // pipeline can advance instead of looping forever (#2653).
802
+ // Retries are capped at MAX_ARTIFACT_VERIFICATION_RETRIES to prevent
803
+ // unbounded loops (#2007).
804
+ //
805
+ // Pre-checks short-circuit retry for known-unrecoverable failures:
806
+ // - Deterministic policy rejection (#4973): structural write-gate failure
807
+ // that will recur on every retry, so write a blocker placeholder.
808
+ // - DB infra failure (#2517): completion tool returned db_unavailable, so
809
+ // the artifact was never written. Retrying can never succeed.
810
+ // - Tool invocation error (#2883/#3595): malformed JSON args or queued
811
+ // user message — retry will produce the same failure.
708
812
  //
709
- // HOWEVER, if the DB is unavailable (db_unavailable), the artifact was never
710
- // written because the completion tool failed at the infra level. Retrying
711
- // can never succeed and produces a costly re-dispatch loop (#2517).
712
- if (!triggerArtifactVerified && !isDbAvailable()) {
713
- // DB infra failure do NOT retry; the completion tool returned
714
- // db_unavailable so the artifact was never written. Retrying would
715
- // produce an infinite re-dispatch loop (#2517).
813
+ // #4973: Deterministic policy rejections (e.g. context_write_blocked from the
814
+ // write-gate) are checked FIRST before the DB-availability check — because
815
+ // they are structural gates that will fire on every retry regardless of DB or
816
+ // model tier. Short-circuit immediately by writing a blocker placeholder.
817
+ if (!triggerArtifactVerified && s.lastToolInvocationError && isDeterministicPolicyError(s.lastToolInvocationError)) {
818
+ const retryKey = `${s.currentUnit.type}:${s.currentUnit.id}`;
819
+ debugLog("postUnit", { phase: "deterministic-policy-error-placeholder", unitType: s.currentUnit.type, unitId: s.currentUnit.id, error: s.lastToolInvocationError });
820
+ const reason = `Deterministic policy rejection for ${s.currentUnit.type} "${s.currentUnit.id}": ${s.lastToolInvocationError}. Retrying cannot resolve this gate — writing blocker placeholder to advance pipeline.`;
821
+ s.lastToolInvocationError = null;
822
+ s.pendingVerificationRetry = null;
823
+ s.verificationRetryCount.delete(retryKey);
824
+ writeBlockerPlaceholder(s.currentUnit.type, s.currentUnit.id, s.basePath, reason);
825
+ ctx.ui.notify(`${s.currentUnit.type} ${s.currentUnit.id} — deterministic policy rejection, wrote blocker placeholder (no retries) (#4973)`, "warning");
826
+ // Fall through to "continue" — do NOT enter the retry or db-unavailable paths.
827
+ }
828
+ else if (!triggerArtifactVerified && !isDbAvailable()) {
716
829
  debugLog("postUnit", { phase: "artifact-verify-skip-db-unavailable", unitType: s.currentUnit.type, unitId: s.currentUnit.id });
717
830
  const dbSkipDiag = diagnoseExpectedArtifact(s.currentUnit.type, s.currentUnit.id, s.basePath);
718
831
  ctx.ui.notify(`Artifact missing for ${s.currentUnit.type} ${s.currentUnit.id} — DB unavailable, skipping retry.${dbSkipDiag ? ` Expected: ${dbSkipDiag}` : ""}`, "error");
719
832
  }
720
833
  else if (!triggerArtifactVerified) {
721
- // #2883/#3595: If the artifact is missing because the tool invocation
722
- // failed (malformed JSON) or was skipped (queued user message), retrying
723
- // will produce the same failure. Pause auto-mode instead of looping.
724
834
  if (s.lastToolInvocationError) {
725
835
  const isUserSkip = /queued user message/i.test(s.lastToolInvocationError);
726
836
  const errMsg = isUserSkip
@@ -736,58 +846,29 @@ export async function postUnitPreVerification(pctx, opts) {
736
846
  if (hasExpectedArtifact) {
737
847
  const retryKey = `${s.currentUnit.type}:${s.currentUnit.id}`;
738
848
  const attempt = (s.verificationRetryCount.get(retryKey) ?? 0) + 1;
739
- s.verificationRetryCount.set(retryKey, attempt);
740
- if (attempt > MAX_VERIFICATION_RETRIES) {
741
- // #4175: For complete-milestone, a blocker placeholder is harmful —
742
- // the stub SUMMARY has no recovery value (milestone is terminal),
743
- // it does not update DB status (so deriveState never advances),
744
- // and it fools stopAuto's presence check into merging a milestone
745
- // that was never legitimately completed. Pause auto-mode with a
746
- // clear single failure signal and preserve the worktree branch.
747
- if (s.currentUnit.type === "complete-milestone") {
748
- debugLog("postUnit", {
749
- phase: "artifact-verify-pause-complete-milestone",
750
- unitType: s.currentUnit.type,
751
- unitId: s.currentUnit.id,
752
- attempt,
753
- maxRetries: MAX_VERIFICATION_RETRIES,
754
- });
755
- s.verificationRetryCount.delete(retryKey);
756
- s.pendingVerificationRetry = null;
757
- ctx.ui.notify(`Milestone ${s.currentUnit.id} verification failed after ${MAX_VERIFICATION_RETRIES} retries — worktree branch preserved. Re-run /gsd auto once blockers are resolved.`, "error");
758
- await pauseAuto(ctx, pi);
759
- return "dispatched";
760
- }
761
- // Retries exhausted — write a blocker placeholder so the pipeline
762
- // can advance past this stuck unit (#2653).
763
- debugLog("postUnit", {
764
- phase: "artifact-verify-escalate",
765
- unitType: s.currentUnit.type,
766
- unitId: s.currentUnit.id,
767
- attempt,
768
- maxRetries: MAX_VERIFICATION_RETRIES,
769
- });
770
- const reason = `Artifact verification failed after ${MAX_VERIFICATION_RETRIES} retries for ${s.currentUnit.type} "${s.currentUnit.id}".`;
771
- writeBlockerPlaceholder(s.currentUnit.type, s.currentUnit.id, s.basePath, reason);
772
- ctx.ui.notify(`${s.currentUnit.type} ${s.currentUnit.id} — verification retries exhausted (${MAX_VERIFICATION_RETRIES}), wrote blocker placeholder to advance pipeline`, "warning");
773
- // Reset retry count and fall through to "continue" so the loop
774
- // re-derives state with the placeholder in place.
849
+ if (attempt > MAX_ARTIFACT_VERIFICATION_RETRIES) {
775
850
  s.verificationRetryCount.delete(retryKey);
776
- s.pendingVerificationRetry = null;
777
- // Do NOT return "retry"fall through to "continue" below.
778
- }
779
- else {
780
- s.pendingVerificationRetry = {
781
- unitId: s.currentUnit.id,
782
- failureContext: `Artifact verification failed: expected artifact for ${s.currentUnit.type} "${s.currentUnit.id}" was not found on disk after unit execution (attempt ${attempt}).`,
783
- attempt,
784
- };
785
- debugLog("postUnit", { phase: "artifact-verify-retry", unitType: s.currentUnit.type, unitId: s.currentUnit.id, attempt });
786
- ctx.ui.notify(`Artifact missing for ${s.currentUnit.type} ${s.currentUnit.id} — retrying (attempt ${attempt})`, "warning");
787
- return "retry";
851
+ debugLog("postUnit", { phase: "artifact-verify-exhausted", unitType: s.currentUnit.type, unitId: s.currentUnit.id, attempt });
852
+ ctx.ui.notify(`Artifact still missing for ${s.currentUnit.type} ${s.currentUnit.id} after ${MAX_ARTIFACT_VERIFICATION_RETRIES} retries pausing auto-mode`, "error");
853
+ await pauseAuto(ctx, pi);
854
+ return "dispatched";
788
855
  }
856
+ s.verificationRetryCount.set(retryKey, attempt);
857
+ s.pendingVerificationRetry = {
858
+ unitId: s.currentUnit.id,
859
+ failureContext: `Artifact verification failed: expected artifact for ${s.currentUnit.type} "${s.currentUnit.id}" was not found on disk after unit execution (attempt ${attempt}/${MAX_ARTIFACT_VERIFICATION_RETRIES}).`,
860
+ attempt,
861
+ };
862
+ debugLog("postUnit", { phase: "artifact-verify-retry", unitType: s.currentUnit.type, unitId: s.currentUnit.id, attempt });
863
+ ctx.ui.notify(`Artifact missing for ${s.currentUnit.type} ${s.currentUnit.id} — retrying (attempt ${attempt}/${MAX_ARTIFACT_VERIFICATION_RETRIES})`, "warning");
864
+ return "retry";
789
865
  }
790
866
  }
867
+ // Verification succeeded — clear the retry counter so a future failure
868
+ // of the same unit gets a full retry budget instead of the stale count.
869
+ if (triggerArtifactVerified) {
870
+ s.verificationRetryCount.delete(`${s.currentUnit.type}:${s.currentUnit.id}`);
871
+ }
791
872
  }
792
873
  else {
793
874
  // Hook unit completed — no additional processing needed
@@ -985,8 +1066,11 @@ export async function postUnitPostVerification(pctx) {
985
1066
  }
986
1067
  // Write evidence JSON to slice artifacts directory
987
1068
  const slicePath = resolveSlicePath(s.basePath, mid, sid);
1069
+ const evidenceFileName = `${sid}-PRE-EXEC-VERIFY.json`;
1070
+ let evidencePath = join(".gsd", "milestones", mid, "slices", sid, evidenceFileName);
988
1071
  if (slicePath) {
989
1072
  writePreExecutionEvidence(result, slicePath, mid, sid);
1073
+ evidencePath = relative(s.basePath, join(slicePath, evidenceFileName)) || evidenceFileName;
990
1074
  }
991
1075
  if (uokFlags.gates) {
992
1076
  const failedChecks = result.checks
@@ -1021,9 +1105,11 @@ export async function postUnitPostVerification(pctx) {
1021
1105
  if (result.status === "fail") {
1022
1106
  const blockingChecks = result.checks.filter(c => !c.passed && c.blocking);
1023
1107
  const blockingCount = blockingChecks.length;
1024
- const details = blockingChecks.slice(0, 3).map(c => ` \u2022 ${c.message}`).join("\n");
1025
- const suffix = blockingChecks.length > 3 ? `\n \u2022 ...and ${blockingChecks.length - 3} more` : "";
1026
- const evidenceNote = `\nSee ${sid}-PRE-EXEC-VERIFY.json for full details.`;
1108
+ const details = blockingChecks.slice(0, MAX_NOTIFICATION_DETAILS).map(formatPreExecutionCheckDetail).join("\n");
1109
+ const suffix = blockingChecks.length > MAX_NOTIFICATION_DETAILS
1110
+ ? `\n ${NOTIFICATION_BULLET} ...and ${blockingChecks.length - MAX_NOTIFICATION_DETAILS} more`
1111
+ : "";
1112
+ const evidenceNote = `\nSee ${evidencePath} for full details.`;
1027
1113
  ctx.ui.notify(`Pre-execution checks failed: ${blockingCount} blocking issue${blockingCount === 1 ? "" : "s"} found\n${details}${suffix}${evidenceNote}`, "error");
1028
1114
  // Persist failure context so the next plan-slice re-dispatch can inject
1029
1115
  // it into the prompt and break the infinite loop (#4551).