gsd-pi 2.77.0-dev.58d3d4d6c → 2.77.0-dev.cfd69e714

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 (429) 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/resource-loader.d.ts +40 -0
  5. package/dist/resource-loader.js +32 -13
  6. package/dist/resources/extensions/browser-tools/capture.js +9 -0
  7. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  8. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  9. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  10. package/dist/resources/extensions/browser-tools/tools/forms.js +5 -1
  11. package/dist/resources/extensions/browser-tools/tools/intent.js +5 -1
  12. package/dist/resources/extensions/claude-code-cli/readiness.js +5 -1
  13. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +481 -17
  14. package/dist/resources/extensions/gsd/auto/loop.js +43 -0
  15. package/dist/resources/extensions/gsd/auto/phases.js +15 -21
  16. package/dist/resources/extensions/gsd/auto/session.js +0 -2
  17. package/dist/resources/extensions/gsd/auto-dispatch.js +102 -24
  18. package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
  19. package/dist/resources/extensions/gsd/auto-post-unit.js +71 -64
  20. package/dist/resources/extensions/gsd/auto-prompts.js +329 -102
  21. package/dist/resources/extensions/gsd/auto-recovery.js +195 -23
  22. package/dist/resources/extensions/gsd/auto-start.js +34 -24
  23. package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
  24. package/dist/resources/extensions/gsd/auto-worktree.js +122 -26
  25. package/dist/resources/extensions/gsd/auto.js +31 -20
  26. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +9 -1
  27. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
  28. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +3 -6
  29. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
  30. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
  31. package/dist/resources/extensions/gsd/component-loader.js +447 -0
  32. package/dist/resources/extensions/gsd/component-types.js +69 -0
  33. package/dist/resources/extensions/gsd/detection.js +49 -1
  34. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  35. package/dist/resources/extensions/gsd/gate-registry.js +2 -2
  36. package/dist/resources/extensions/gsd/git-constants.js +28 -1
  37. package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
  38. package/dist/resources/extensions/gsd/git-service.js +126 -2
  39. package/dist/resources/extensions/gsd/gsd-db.js +6 -3
  40. package/dist/resources/extensions/gsd/guided-flow.js +17 -5
  41. package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
  42. package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
  43. package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
  44. package/dist/resources/extensions/gsd/model-router.js +6 -0
  45. package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
  46. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  47. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  48. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
  49. package/dist/resources/extensions/gsd/service-tier.js +5 -2
  50. package/dist/resources/extensions/gsd/session-lock.js +19 -10
  51. package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
  52. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
  53. package/dist/resources/extensions/gsd/state.js +44 -33
  54. package/dist/resources/extensions/gsd/sync-lock.js +98 -42
  55. package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
  56. package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
  57. package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
  58. package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
  59. package/dist/resources/extensions/gsd/worktree-manager.js +34 -8
  60. package/dist/resources/extensions/mcp-client/index.js +3 -1
  61. package/dist/resources/extensions/ollama/index.js +5 -1
  62. package/dist/resources/extensions/remote-questions/manager.js +11 -5
  63. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  64. package/dist/web/standalone/.next/BUILD_ID +1 -1
  65. package/dist/web/standalone/.next/app-path-routes-manifest.json +5 -5
  66. package/dist/web/standalone/.next/build-manifest.json +2 -2
  67. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  68. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  85. package/dist/web/standalone/.next/server/app/index.html +1 -1
  86. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app-paths-manifest.json +5 -5
  93. package/dist/web/standalone/.next/server/chunks/1926.js +1 -1
  94. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  95. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  96. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  97. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  98. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  99. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  100. package/package.json +2 -3
  101. package/packages/daemon/src/logger.ts +4 -3
  102. package/packages/mcp-server/dist/server.d.ts +24 -0
  103. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  104. package/packages/mcp-server/dist/server.js +88 -87
  105. package/packages/mcp-server/dist/server.js.map +1 -1
  106. package/packages/mcp-server/src/mcp-server.test.ts +25 -3
  107. package/packages/mcp-server/src/readers/graph.test.ts +87 -15
  108. package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
  109. package/packages/mcp-server/src/server.ts +131 -105
  110. package/packages/mcp-server/src/workflow-tools.test.ts +80 -39
  111. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  112. package/packages/native/package.json +1 -1
  113. package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
  114. package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
  115. package/packages/native/src/__tests__/ps.test.mjs +14 -8
  116. package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
  117. package/packages/native/src/__tests__/truncate.test.mjs +17 -2
  118. package/packages/pi-agent-core/src/agent-loop.test.ts +5 -15
  119. package/packages/pi-agent-core/src/agent.test.ts +96 -102
  120. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  121. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
  122. package/packages/pi-ai/dist/models/capability-patches.js +9 -2
  123. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
  124. package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
  125. package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
  126. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
  127. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
  128. package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
  129. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
  130. package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
  131. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
  132. package/packages/pi-ai/dist/models/generated/openai.js +17 -0
  133. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
  134. package/packages/pi-ai/dist/models.generated.test.js +43 -70
  135. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  136. package/packages/pi-ai/dist/models.test.js +36 -11
  137. package/packages/pi-ai/dist/models.test.js.map +1 -1
  138. package/packages/pi-ai/scripts/generate-models.ts +44 -0
  139. package/packages/pi-ai/src/models/capability-patches.ts +10 -2
  140. package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
  141. package/packages/pi-ai/src/models/generated/openai.ts +17 -0
  142. package/packages/pi-ai/src/models.generated.test.ts +46 -73
  143. package/packages/pi-ai/src/models.test.ts +48 -11
  144. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  145. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
  146. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  147. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
  148. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
  149. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
  150. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  151. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +5 -0
  152. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  153. package/packages/pi-coding-agent/dist/core/extensions/loader.js +61 -0
  154. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  155. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +30 -4
  156. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  157. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +17 -0
  158. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  159. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
  160. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
  161. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  162. package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
  163. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  164. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
  165. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  166. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
  167. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
  168. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
  169. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
  170. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
  171. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  172. package/packages/pi-coding-agent/dist/core/system-prompt.js +16 -2
  173. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  174. package/packages/pi-coding-agent/dist/index.d.ts +1 -0
  175. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  176. package/packages/pi-coding-agent/dist/index.js +1 -0
  177. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  178. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
  179. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
  180. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
  182. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +125 -0
  184. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  185. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
  186. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  187. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  188. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
  189. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  190. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
  191. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  192. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
  193. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
  194. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
  195. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
  196. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
  197. package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
  198. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
  199. package/packages/pi-coding-agent/src/core/extensions/loader.ts +58 -0
  200. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +35 -4
  201. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +20 -0
  202. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
  203. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
  204. package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
  205. package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
  206. package/packages/pi-coding-agent/src/core/system-prompt.ts +35 -1
  207. package/packages/pi-coding-agent/src/index.ts +1 -0
  208. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
  209. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +146 -1
  210. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
  211. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
  212. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
  213. package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
  214. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  215. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +18 -8
  216. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  217. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
  218. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
  219. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +36 -12
  220. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  221. package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
  222. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  223. package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
  224. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  225. package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
  226. package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
  227. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
  228. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
  229. package/packages/pi-tui/dist/components/editor.d.ts +14 -0
  230. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  231. package/packages/pi-tui/dist/components/editor.js +19 -0
  232. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  233. package/packages/pi-tui/dist/components/image.test.js +6 -5
  234. package/packages/pi-tui/dist/components/image.test.js.map +1 -1
  235. package/packages/pi-tui/dist/editor-component.d.ts +2 -0
  236. package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
  237. package/packages/pi-tui/dist/editor-component.js.map +1 -1
  238. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +24 -8
  239. package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
  240. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +41 -12
  241. package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
  242. package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
  243. package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
  244. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
  245. package/packages/pi-tui/src/components/editor.ts +22 -0
  246. package/packages/pi-tui/src/components/image.test.ts +10 -5
  247. package/packages/pi-tui/src/editor-component.ts +3 -0
  248. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  249. package/packages/rpc-client/dist/rpc-client.test.js +101 -51
  250. package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
  251. package/packages/rpc-client/src/rpc-client.test.ts +109 -52
  252. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  253. package/scripts/install.js +15 -1
  254. package/src/resources/extensions/browser-tools/capture.ts +12 -0
  255. package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  256. package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  257. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  258. package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
  259. package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
  260. package/src/resources/extensions/claude-code-cli/readiness.ts +5 -1
  261. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +518 -19
  262. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +919 -75
  263. package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
  264. package/src/resources/extensions/github-sync/tests/templates.test.ts +33 -1
  265. package/src/resources/extensions/gsd/auto/loop.ts +47 -0
  266. package/src/resources/extensions/gsd/auto/phases.ts +16 -20
  267. package/src/resources/extensions/gsd/auto/session.ts +0 -2
  268. package/src/resources/extensions/gsd/auto-dispatch.ts +113 -24
  269. package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
  270. package/src/resources/extensions/gsd/auto-post-unit.ts +82 -73
  271. package/src/resources/extensions/gsd/auto-prompts.ts +330 -90
  272. package/src/resources/extensions/gsd/auto-recovery.ts +225 -24
  273. package/src/resources/extensions/gsd/auto-start.ts +54 -6
  274. package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
  275. package/src/resources/extensions/gsd/auto-worktree.ts +130 -26
  276. package/src/resources/extensions/gsd/auto.ts +43 -22
  277. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +9 -1
  278. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
  279. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +3 -7
  280. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
  281. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
  282. package/src/resources/extensions/gsd/component-loader.ts +598 -0
  283. package/src/resources/extensions/gsd/component-types.ts +362 -0
  284. package/src/resources/extensions/gsd/detection.ts +58 -1
  285. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  286. package/src/resources/extensions/gsd/gate-registry.ts +2 -2
  287. package/src/resources/extensions/gsd/git-constants.ts +30 -1
  288. package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
  289. package/src/resources/extensions/gsd/git-service.ts +133 -2
  290. package/src/resources/extensions/gsd/gsd-db.ts +6 -3
  291. package/src/resources/extensions/gsd/guided-flow.ts +20 -5
  292. package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
  293. package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
  294. package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
  295. package/src/resources/extensions/gsd/model-router.ts +6 -0
  296. package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
  297. package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  298. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  299. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
  300. package/src/resources/extensions/gsd/service-tier.ts +5 -2
  301. package/src/resources/extensions/gsd/session-lock.ts +20 -10
  302. package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
  303. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
  304. package/src/resources/extensions/gsd/state.ts +49 -44
  305. package/src/resources/extensions/gsd/sync-lock.ts +97 -39
  306. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
  307. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
  308. package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
  309. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +94 -289
  310. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
  311. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
  312. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
  313. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +93 -0
  314. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -197
  315. package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
  316. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
  317. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
  318. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
  319. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
  320. package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
  321. package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
  322. package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
  323. package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
  324. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
  325. package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
  326. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
  327. package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -3
  328. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
  329. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
  330. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +9 -105
  331. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
  332. package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
  333. package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
  334. package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -57
  335. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
  336. package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
  337. package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
  338. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
  339. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
  340. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
  341. package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
  342. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +8 -37
  343. package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
  344. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -62
  345. package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
  346. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
  347. package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
  348. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -49
  349. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
  350. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -133
  351. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
  352. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
  353. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +23 -24
  354. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +32 -0
  355. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
  356. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
  357. package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
  358. package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
  359. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
  360. package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
  361. package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
  362. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
  363. package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
  364. package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
  365. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
  366. package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
  367. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
  368. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +5 -5
  369. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
  370. package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
  371. package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
  372. package/src/resources/extensions/gsd/tests/test-helpers.test.ts +12 -61
  373. package/src/resources/extensions/gsd/tests/test-helpers.ts +21 -8
  374. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
  375. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
  376. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
  377. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
  378. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
  379. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
  380. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +1 -0
  381. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -81
  382. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
  383. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
  384. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
  385. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
  386. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
  387. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
  388. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
  389. package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
  390. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
  391. package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
  392. package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
  393. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
  394. package/src/resources/extensions/gsd/types.ts +3 -3
  395. package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
  396. package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
  397. package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
  398. package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
  399. package/src/resources/extensions/gsd/worktree-manager.ts +55 -7
  400. package/src/resources/extensions/mcp-client/index.ts +3 -1
  401. package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
  402. package/src/resources/extensions/ollama/index.ts +5 -1
  403. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
  404. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
  405. package/src/resources/extensions/remote-questions/manager.ts +36 -4
  406. package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
  407. package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
  408. package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
  409. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
  410. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
  411. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
  412. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
  413. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
  414. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -144
  415. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -157
  416. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
  417. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
  418. package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
  419. package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
  420. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
  421. package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -75
  422. package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
  423. package/src/resources/extensions/gsd/tests/forensics-worktree-telemetry.test.ts +0 -145
  424. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
  425. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
  426. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -130
  427. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -43
  428. /package/dist/web/standalone/.next/static/{Cev5xrAYA3ZGTRLyjR2fX → SvCJDZPQW104bR1KnBQg1}/_buildManifest.js +0 -0
  429. /package/dist/web/standalone/.next/static/{Cev5xrAYA3ZGTRLyjR2fX → SvCJDZPQW104bR1KnBQg1}/_ssgManifest.js +0 -0
@@ -15,8 +15,9 @@ import { isDbAvailable, getTask, getSlice, getSliceTasks, getPendingGates, updat
15
15
  import { isValidationTerminal } from "./state.js";
16
16
  import { getErrorMessage } from "./error-utils.js";
17
17
  import { logWarning, logError } from "./workflow-logger.js";
18
+ import { readIntegrationBranch } from "./git-service.js";
18
19
  import { isClosedStatus } from "./status-guards.js";
19
- import { nativeConflictFiles, nativeCommit, nativeCheckoutTheirs, nativeAddPaths, nativeMergeAbort, nativeResetHard, } from "./native-git-bridge.js";
20
+ import { nativeConflictFiles, nativeCommit, nativeCheckoutTheirs, nativeAddPaths, nativeMergeAbort, nativeRebaseAbort, nativeResetHard, } from "./native-git-bridge.js";
20
21
  import { resolveSlicePath, resolveSliceFile, resolveTasksDir, resolveTaskFiles, relMilestoneFile, relSliceFile, buildSliceFileName, resolveMilestoneFile, clearPathCache, resolveGsdRootFile, } from "./paths.js";
21
22
  import { existsSync, mkdirSync, readFileSync, writeFileSync, unlinkSync, } from "node:fs";
22
23
  import { execFileSync } from "node:child_process";
@@ -28,14 +29,16 @@ export { resolveExpectedArtifactPath, diagnoseExpectedArtifact };
28
29
  export { classifyMilestoneSummaryContent, } from "./milestone-summary-classifier.js";
29
30
  // ─── Artifact Resolution & Verification ───────────────────────────────────────
30
31
  /**
31
- * Check whether a milestone produced implementation artifacts (non-`.gsd/` files)
32
- * in the git history. Uses `git log --name-only` to inspect all commits on the
33
- * current branch that touch files outside `.gsd/`.
32
+ * Check whether a milestone produced implementation artifacts (non-`.gsd/`
33
+ * files) in git history. The primary signal is the branch diff against the
34
+ * integration branch. When a retry is already on the integration branch, that
35
+ * diff is a self-diff; if a milestone ID is available, fall back to recent
36
+ * GSD-tagged commits for that milestone.
34
37
  *
35
38
  * Returns "present" if implementation files found, "absent" if only .gsd/ files,
36
39
  * "unknown" if git is unavailable or check failed (callers decide how to handle).
37
40
  */
38
- export function hasImplementationArtifacts(basePath) {
41
+ export function hasImplementationArtifacts(basePath, milestoneId) {
39
42
  try {
40
43
  // Verify we're in a git repo
41
44
  try {
@@ -51,19 +54,31 @@ export function hasImplementationArtifacts(basePath) {
51
54
  }
52
55
  // Strategy: check `git diff --name-only` against the merge-base with the
53
56
  // main branch. This captures ALL files changed during the milestone's
54
- // lifetime. If no merge-base exists (e.g., single-branch workflow), fall
55
- // back to checking the last N commits.
56
- const mainBranch = detectMainBranch(basePath);
57
- const changedFiles = getChangedFilesSinceBranch(basePath, mainBranch);
58
- // No files changed at all — unknown (could be detached HEAD, single-
59
- // commit repo, or other edge case where git diff returns nothing).
60
- if (changedFiles.length === 0)
57
+ // lifetime while running on a milestone branch.
58
+ const integrationBranch = milestoneId
59
+ ? readIntegrationBranch(basePath, milestoneId) ?? detectMainBranch(basePath)
60
+ : detectMainBranch(basePath);
61
+ const currentBranch = getCurrentBranch(basePath);
62
+ const branchDiff = getChangedFilesSinceBranch(basePath, integrationBranch);
63
+ if (!branchDiff.ok)
61
64
  return "unknown";
62
- // Filter out .gsd/ files — only implementation files count.
63
- // If every changed file is under .gsd/, the milestone produced no
64
- // implementation code (#1703).
65
- const implFiles = changedFiles.filter(f => !f.startsWith(".gsd/") && !f.startsWith(".gsd\\"));
66
- return implFiles.length > 0 ? "present" : "absent";
65
+ const changedFiles = branchDiff.files;
66
+ // No branch-diff files can mean the unit retried on main after milestone
67
+ // commits already landed there. In that topology, inspect GSD-tagged
68
+ // milestone commits instead of treating the self-diff as proof of no work.
69
+ if (changedFiles.length === 0) {
70
+ if (milestoneId && currentBranch === integrationBranch) {
71
+ const tagged = getChangedFilesFromMilestoneTaggedCommits(basePath, milestoneId);
72
+ if (!tagged.ok)
73
+ return "unknown";
74
+ if (tagged.matched)
75
+ return classifyImplementationFiles(tagged.files);
76
+ }
77
+ if (currentBranch && currentBranch !== "HEAD")
78
+ return "absent";
79
+ return "unknown";
80
+ }
81
+ return classifyImplementationFiles(changedFiles);
67
82
  }
68
83
  catch (e) {
69
84
  // Non-fatal — if git operations fail, return unknown so callers can decide
@@ -71,6 +86,26 @@ export function hasImplementationArtifacts(basePath) {
71
86
  return "unknown";
72
87
  }
73
88
  }
89
+ function getCurrentBranch(basePath) {
90
+ try {
91
+ const branch = execFileSync("git", ["rev-parse", "--abbrev-ref", "HEAD"], {
92
+ cwd: basePath,
93
+ stdio: ["ignore", "pipe", "pipe"],
94
+ encoding: "utf-8",
95
+ }).trim();
96
+ return branch || null;
97
+ }
98
+ catch {
99
+ return null;
100
+ }
101
+ }
102
+ function classifyImplementationFiles(files) {
103
+ const implFiles = files.filter(isImplementationPath);
104
+ return implFiles.length > 0 ? "present" : "absent";
105
+ }
106
+ function isImplementationPath(file) {
107
+ return !file.startsWith(".gsd/") && !file.startsWith(".gsd\\");
108
+ }
74
109
  /**
75
110
  * Detect the main/master branch name.
76
111
  */
@@ -115,7 +150,7 @@ function getChangedFilesSinceBranch(basePath, targetBranch) {
115
150
  const mergeBase = execFileSync("git", ["merge-base", targetBranch, "HEAD"], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
116
151
  if (mergeBase) {
117
152
  const result = execFileSync("git", ["diff", "--name-only", mergeBase, "HEAD"], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
118
- return result ? result.split("\n").filter(Boolean) : [];
153
+ return { ok: true, files: result ? result.split("\n").filter(Boolean) : [] };
119
154
  }
120
155
  }
121
156
  catch (err) {
@@ -125,13 +160,75 @@ function getChangedFilesSinceBranch(basePath, targetBranch) {
125
160
  // Fallback: check last 20 commits
126
161
  try {
127
162
  const result = execFileSync("git", ["log", "--name-only", "--pretty=format:", "-20", "HEAD"], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
128
- return result ? [...new Set(result.split("\n").filter(Boolean))] : [];
163
+ return { ok: true, files: result ? [...new Set(result.split("\n").filter(Boolean))] : [] };
129
164
  }
130
165
  catch (e) {
131
166
  logWarning("recovery", `git log fallback failed: ${e.message}`);
132
- return [];
167
+ return { ok: false, files: [] };
133
168
  }
134
169
  }
170
+ function getChangedFilesFromMilestoneTaggedCommits(basePath, milestoneId) {
171
+ try {
172
+ const logOutput = execFileSync("git", ["log", "--format=%H%x1f%B%x1e", "HEAD", "--", `.gsd/milestones/${milestoneId}`], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" });
173
+ const records = logOutput
174
+ .split("\x1e")
175
+ .map((record) => record.trim())
176
+ .filter(Boolean)
177
+ .flatMap((record) => {
178
+ const sep = record.indexOf("\x1f");
179
+ if (sep === -1)
180
+ return [];
181
+ const hash = record.slice(0, sep).trim();
182
+ const message = record.slice(sep + 1);
183
+ return [{ hash, message }];
184
+ });
185
+ const files = new Set();
186
+ let matched = false;
187
+ for (const { hash, message } of records) {
188
+ if (!commitMessageHasGsdTrailer(message))
189
+ continue;
190
+ const commitFiles = getChangedFilesForCommit(basePath, hash);
191
+ if (!commitMatchesMilestone(message, milestoneId, commitFiles))
192
+ continue;
193
+ matched = true;
194
+ for (const file of commitFiles) {
195
+ files.add(file);
196
+ }
197
+ }
198
+ return { ok: true, matched, files: [...files] };
199
+ }
200
+ catch (e) {
201
+ logWarning("recovery", `milestone-tagged commit scan failed: ${e.message}`);
202
+ return { ok: false, matched: false, files: [] };
203
+ }
204
+ }
205
+ function getChangedFilesForCommit(basePath, hash) {
206
+ const fileOutput = execFileSync("git", ["diff-tree", "--root", "--no-commit-id", "-r", "--name-only", hash], { cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
207
+ return fileOutput.split("\n").map((f) => f.trim()).filter(Boolean);
208
+ }
209
+ function commitMessageHasGsdTrailer(message) {
210
+ return /^GSD-(?:Task|Unit):\s*\S+/m.test(message);
211
+ }
212
+ function commitMatchesMilestone(message, milestoneId, files) {
213
+ if (commitTrailerStartsWithMilestone(message, milestoneId))
214
+ return true;
215
+ // Meaningful execute-task commits currently store task scope as Sxx/Tyy
216
+ // rather than Mxx/Sxx/Tyy. Bind those commits back to the milestone only
217
+ // when the commit also touched this milestone's artifacts.
218
+ if (/^GSD-Task:\s*S[^/\s]+\/T\S+/m.test(message)) {
219
+ return files.some((file) => isMilestoneArtifactPath(file, milestoneId));
220
+ }
221
+ return false;
222
+ }
223
+ function commitTrailerStartsWithMilestone(message, milestoneId) {
224
+ const escapedMilestone = milestoneId.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
225
+ const trailerPattern = new RegExp(`^GSD-(?:Task|Unit):\\s*${escapedMilestone}(?:$|[\\s/])`, "m");
226
+ return trailerPattern.test(message);
227
+ }
228
+ function isMilestoneArtifactPath(file, milestoneId) {
229
+ return file.startsWith(`.gsd/milestones/${milestoneId}/`)
230
+ || file.startsWith(`.gsd\\milestones\\${milestoneId}\\`);
231
+ }
135
232
  /**
136
233
  * Check whether the expected artifact(s) for a unit exist on disk.
137
234
  * Returns true if all required artifacts exist, or if the unit type has no
@@ -447,7 +544,7 @@ export function verifyExpectedArtifact(unitType, unitId, base) {
447
544
  if (!isClosedStatus(dbMilestone.status) && summaryOutcome !== "success")
448
545
  return false;
449
546
  }
450
- if (hasImplementationArtifacts(base) === "absent")
547
+ if (hasImplementationArtifacts(base, mid) === "absent")
451
548
  return false;
452
549
  }
453
550
  return true;
@@ -577,6 +674,72 @@ function abortAndResetMerge(basePath, hasMergeHead, squashMsgPath) {
577
674
  logError("recovery", `git reset failed: ${err instanceof Error ? err.message : String(err)}`);
578
675
  }
579
676
  }
677
+ /**
678
+ * Detect and abort other in-progress git operations left behind by a SIGKILL'd
679
+ * worker (rebase, cherry-pick, revert). Without this, a killed worker mid-rebase
680
+ * leaves `.git/rebase-merge/` or `.git/CHERRY_PICK_HEAD` and the worktree is
681
+ * wedged until the user manually runs the matching `--abort`.
682
+ *
683
+ * Called before merge-state reconciliation because these states block any
684
+ * subsequent merge/commit operation. (Issue #4980 HIGH-7)
685
+ */
686
+ function reconcileOtherInProgressGitOps(basePath, ctx) {
687
+ const gitDir = join(basePath, ".git");
688
+ const states = [
689
+ {
690
+ label: "rebase",
691
+ indicators: [join(gitDir, "rebase-merge"), join(gitDir, "rebase-apply")],
692
+ abort: () => nativeRebaseAbort(basePath),
693
+ },
694
+ {
695
+ label: "cherry-pick",
696
+ indicators: [join(gitDir, "CHERRY_PICK_HEAD")],
697
+ abort: () => {
698
+ // No native helper; fall back to git CLI.
699
+ try {
700
+ execFileSync("git", ["cherry-pick", "--abort"], {
701
+ cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8",
702
+ });
703
+ }
704
+ catch (err) {
705
+ logWarning("recovery", `cherry-pick --abort failed: ${getErrorMessage(err)}`);
706
+ }
707
+ },
708
+ },
709
+ {
710
+ label: "revert",
711
+ indicators: [join(gitDir, "REVERT_HEAD")],
712
+ abort: () => {
713
+ try {
714
+ execFileSync("git", ["revert", "--abort"], {
715
+ cwd: basePath, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8",
716
+ });
717
+ }
718
+ catch (err) {
719
+ logWarning("recovery", `revert --abort failed: ${getErrorMessage(err)}`);
720
+ }
721
+ },
722
+ },
723
+ ];
724
+ let reconciled = false;
725
+ for (const s of states) {
726
+ const present = s.indicators.some((p) => existsSync(p));
727
+ if (!present)
728
+ continue;
729
+ try {
730
+ s.abort();
731
+ ctx.ui.notify(`Detected leftover ${s.label} state from prior session — aborted.`, "warning");
732
+ reconciled = true;
733
+ }
734
+ catch (err) {
735
+ logError("recovery", `${s.label} abort failed: ${getErrorMessage(err)}`);
736
+ ctx.ui.notify(`Detected leftover ${s.label} state but auto-abort failed. ` +
737
+ `Run \`git ${s.label} --abort\` manually before retrying.`, "error");
738
+ return "blocked";
739
+ }
740
+ }
741
+ return reconciled ? "reconciled" : "clean";
742
+ }
580
743
  /**
581
744
  * Detect leftover merge state from a prior session and reconcile it.
582
745
  * If MERGE_HEAD or SQUASH_MSG exists, check whether conflicts are resolved.
@@ -584,12 +747,21 @@ function abortAndResetMerge(basePath, hasMergeHead, squashMsgPath) {
584
747
  * If code conflicts remain: fail safe without modifying the worktree.
585
748
  */
586
749
  export function reconcileMergeState(basePath, ctx) {
750
+ // First, abort any rebase/cherry-pick/revert left over from a SIGKILL'd
751
+ // worker. Doing this before the merge-state check unblocks any merge that
752
+ // would otherwise refuse with "you have unfinished operation". (HIGH-7)
753
+ const otherOpsResult = reconcileOtherInProgressGitOps(basePath, ctx);
754
+ if (otherOpsResult === "blocked")
755
+ return "blocked";
587
756
  const mergeHeadPath = join(basePath, ".git", "MERGE_HEAD");
588
757
  const squashMsgPath = join(basePath, ".git", "SQUASH_MSG");
589
758
  const hasMergeHead = existsSync(mergeHeadPath);
590
759
  const hasSquashMsg = existsSync(squashMsgPath);
591
- if (!hasMergeHead && !hasSquashMsg)
592
- return "clean";
760
+ if (!hasMergeHead && !hasSquashMsg) {
761
+ // If we cleaned up another op type, return "reconciled" so the caller
762
+ // re-derives state from a known-good baseline.
763
+ return otherOpsResult === "reconciled" ? "reconciled" : "clean";
764
+ }
593
765
  const conflictedFiles = nativeConflictFiles(basePath);
594
766
  if (conflictedFiles.length === 0) {
595
767
  // All conflicts resolved — finalize the merge/squash commit
@@ -64,25 +64,15 @@ export async function openProjectDbIfPresent(basePath) {
64
64
  logWarning("engine", `gsd-db: failed to open existing database: ${err instanceof Error ? err.message : String(err)}`);
65
65
  }
66
66
  }
67
- /**
68
- * Audit for orphaned milestone branches at bootstrap.
69
- *
70
- * After a milestone completes, the teardown step (merge branch → main,
71
- * delete branch, remove worktree) runs as a post-completion engine step.
72
- * If the session ends between completion and teardown, the branch and
73
- * worktree are orphaned — the DB says "complete" so auto-mode won't
74
- * re-enter the milestone, and the teardown is never retried.
75
- *
76
- * This audit runs on every fresh bootstrap to catch that gap:
77
- * 1. Lists all local `milestone/*` branches.
78
- * 2. For each, checks if the milestone's DB status is "complete".
79
- * 3. If the branch is already merged into main → deletes the branch
80
- * and cleans up any orphaned worktree directory (safe, no data loss).
81
- * 4. If the branch is NOT merged → preserves it and warns the user
82
- * so they can merge manually (data safety first).
83
- *
84
- * Returns a summary of actions taken for the caller to surface via notify.
85
- */
67
+ export function decideSurvivorAction(hasSurvivorBranch, phase) {
68
+ if (!hasSurvivorBranch)
69
+ return "none";
70
+ if (phase === "needs-discussion")
71
+ return "discuss";
72
+ if (phase === "complete")
73
+ return "finalize";
74
+ return "none";
75
+ }
86
76
  export function auditOrphanedMilestoneBranches(basePath, isolationMode) {
87
77
  const recovered = [];
88
78
  const warnings = [];
@@ -450,7 +440,7 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
450
440
  // The worktree/branch was created but the milestone only has CONTEXT-DRAFT.md.
451
441
  // Route to the interactive discussion handler instead of falling through to
452
442
  // auto-mode, which would immediately stop with "needs discussion".
453
- if (hasSurvivorBranch && state.phase === "needs-discussion") {
443
+ if (decideSurvivorAction(hasSurvivorBranch, state.phase) === "discuss") {
454
444
  const { showSmartEntry } = await import("./guided-flow.js");
455
445
  await showSmartEntry(ctx, pi, base, { step: requestedStepMode });
456
446
  invalidateAllCaches();
@@ -470,7 +460,9 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
470
460
  // The milestone artifacts were written but finalization (merge, worktree
471
461
  // cleanup) never ran. Run mergeAndExit to finalize, then re-derive state
472
462
  // so the normal "all milestones complete" or "next milestone" path runs.
473
- if (hasSurvivorBranch && state.phase === "complete") {
463
+ // Re-evaluate via the helper the discuss branch above may have cleared
464
+ // hasSurvivorBranch after a successful promotion.
465
+ if (decideSurvivorAction(hasSurvivorBranch, state.phase) === "finalize") {
474
466
  const mid = state.activeMilestone.id;
475
467
  ctx.ui.notify(`Milestone ${mid} is complete but branch/worktree was not finalized. Running merge now.`, "info");
476
468
  const resolver = buildResolver();
@@ -776,14 +768,32 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
776
768
  catch (err) {
777
769
  ctx.ui.notify(`Secrets collection error: ${err instanceof Error ? err.message : String(err)}. Continuing with next task.`, "warning");
778
770
  }
779
- // Self-heal: remove stale .git/index.lock
771
+ // Self-heal: remove stale .git/index.lock.
772
+ //
773
+ // Threshold raised from 60s → 5min because a 60s-old lock is not
774
+ // definitively stale: `git gc --auto` triggered by a heavy commit, NFS
775
+ // delays, or concurrent worktree writes can hold .git/index.lock for
776
+ // minutes on large repos. Force-removing a live lock causes the holder
777
+ // to encounter `fatal: Unable to create '.git/index.lock'` on its next
778
+ // write, or worse, operate on a partially-written index → corruption
779
+ // requiring `git fsck`/`git reset` to recover.
780
+ // (Issue #4980 CRIT-3)
780
781
  try {
781
782
  const gitLockFile = join(base, ".git", "index.lock");
782
783
  if (existsSync(gitLockFile)) {
783
784
  const lockAge = Date.now() - statSync(gitLockFile).mtimeMs;
784
- if (lockAge > 60_000) {
785
+ const STALE_GIT_LOCK_THRESHOLD_MS = 5 * 60_000;
786
+ if (lockAge > STALE_GIT_LOCK_THRESHOLD_MS) {
785
787
  unlinkSync(gitLockFile);
786
- ctx.ui.notify("Removed stale .git/index.lock from prior crash.", "info");
788
+ ctx.ui.notify(`Removed stale .git/index.lock (age ${Math.round(lockAge / 1000)}s, > 5min threshold).`, "warning");
789
+ }
790
+ else {
791
+ // Lock present but not yet stale — surface so the user knows why
792
+ // git ops may be queueing instead of silently waiting.
793
+ debugLog("git-lock-present-not-stale", {
794
+ ageMs: lockAge,
795
+ thresholdMs: STALE_GIT_LOCK_THRESHOLD_MS,
796
+ });
787
797
  }
788
798
  }
789
799
  }
@@ -77,20 +77,21 @@ export function clearInFlightTools() {
77
77
  }
78
78
  // ─── Tool invocation error classification (#2883) ────────────────────────
79
79
  /**
80
- * Patterns that indicate a tool invocation failed due to malformed or truncated
81
- * JSON arguments — as opposed to a normal business-logic error from the tool
82
- * handler. When these errors occur, retrying the same unit will produce the same
83
- * failure, so the retry loop must be broken.
80
+ * Patterns that indicate a tool invocation failed deterministically before
81
+ * useful work could be completed — as opposed to a normal business-logic error
82
+ * from the tool handler. When these errors occur, retrying the same unit will
83
+ * produce the same failure, so the retry loop must be broken.
84
84
  */
85
85
  const TOOL_INVOCATION_ERROR_RE = /Validation failed for tool|Expected ',' or '\}'(?: after property value)?(?: in JSON)?|Unexpected end of JSON|Unexpected token.*in JSON/i;
86
+ const DETERMINISTIC_POLICY_ERROR_RE = /(?:^|\b)(?:HARD BLOCK:|Blocked: \/gsd queue is a planning tool|Direct writes to \.gsd\/STATE\.md and \.gsd\/gsd\.db are blocked|This is a mechanical gate)/i;
86
87
  /**
87
- * Returns true if the error message indicates a tool invocation failure due to
88
- * malformed/truncated arguments (as opposed to a normal tool execution error).
88
+ * Returns true if the error message indicates a deterministic invocation or
89
+ * policy failure (as opposed to a normal tool execution error).
89
90
  */
90
91
  export function isToolInvocationError(errorMsg) {
91
92
  if (!errorMsg)
92
93
  return false;
93
- return TOOL_INVOCATION_ERROR_RE.test(errorMsg);
94
+ return TOOL_INVOCATION_ERROR_RE.test(errorMsg) || isDeterministicPolicyError(errorMsg);
94
95
  }
95
96
  /**
96
97
  * Returns true if the error message indicates the tool was skipped because
@@ -102,3 +103,42 @@ export function isQueuedUserMessageSkip(errorMsg) {
102
103
  return false;
103
104
  return /^Skipped due to queued user message\.?$/i.test(errorMsg.trim());
104
105
  }
106
+ // ─── Deterministic policy error classification (#4973, #4974) ──────────────
107
+ /**
108
+ * Known deterministic policy error substrings. Each entry is a stable string
109
+ * that will appear in the tool error text content when the corresponding
110
+ * policy gate fires. Retrying these errors will always produce the same outcome.
111
+ *
112
+ * Add new entries here as new deterministic gates are introduced. Do NOT use
113
+ * regex — explicit substrings keep the list auditable.
114
+ */
115
+ export const DETERMINISTIC_POLICY_ERROR_STRINGS = [
116
+ // gsd_summary_save write-gate: CONTEXT artifact blocked pending depth verification (#4973).
117
+ // Matches the fallback text in workflow-tool-executors.ts and the verbose reason
118
+ // from shouldBlockContextArtifactSaveInSnapshot at write-gate.ts:432-442.
119
+ "context write blocked",
120
+ "CONTEXT without depth verification",
121
+ // Raw write tool gate (#4973): shouldBlockContextWrite at write-gate.ts:390-399 emits
122
+ // "Cannot write to milestone CONTEXT.md without depth verification." for direct
123
+ // write tool calls to *-CONTEXT.md paths (different code path than gsd_summary_save).
124
+ "CONTEXT.md without depth verification",
125
+ ];
126
+ /**
127
+ * Returns true if the error message indicates a deterministic policy gate
128
+ * blocked the tool call before execution. Retrying the same unit without
129
+ * changing behavior will hit the same gate, so auto-mode should pause instead
130
+ * of re-dispatching.
131
+ *
132
+ * Combines the regex-based gate set from #4974 (HARD BLOCK / queue planning /
133
+ * STATE.md / mechanical gate) and the substring-based set from #4973 (context
134
+ * write block / CONTEXT depth verification). Both branches landed on main
135
+ * independently and their parallel `isDeterministicPolicyError` declarations
136
+ * were not deduplicated at merge — this consolidated form preserves both
137
+ * matchers under a single export.
138
+ */
139
+ export function isDeterministicPolicyError(errorMsg) {
140
+ if (!errorMsg)
141
+ return false;
142
+ return (DETERMINISTIC_POLICY_ERROR_RE.test(errorMsg) ||
143
+ DETERMINISTIC_POLICY_ERROR_STRINGS.some(s => errorMsg.includes(s)));
144
+ }