gsd-pi 2.74.0-dev.b741afb → 2.74.0-dev.ffbcc03

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 (403) hide show
  1. package/dist/resources/extensions/gsd/activity-log.js +16 -0
  2. package/dist/resources/extensions/gsd/auto/loop.js +147 -10
  3. package/dist/resources/extensions/gsd/auto/phases.js +113 -3
  4. package/dist/resources/extensions/gsd/auto/session.js +10 -0
  5. package/dist/resources/extensions/gsd/auto-dispatch.js +11 -1
  6. package/dist/resources/extensions/gsd/auto-model-selection.js +51 -5
  7. package/dist/resources/extensions/gsd/auto-post-unit.js +215 -8
  8. package/dist/resources/extensions/gsd/auto-recovery.js +24 -10
  9. package/dist/resources/extensions/gsd/auto-unit-closeout.js +18 -0
  10. package/dist/resources/extensions/gsd/auto-verification.js +100 -2
  11. package/dist/resources/extensions/gsd/auto.js +28 -2
  12. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +10 -1
  13. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +61 -9
  14. package/dist/resources/extensions/gsd/cache.js +16 -5
  15. package/dist/resources/extensions/gsd/commands/handlers/ops.js +5 -0
  16. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  17. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +1 -1
  18. package/dist/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  19. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  20. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  21. package/dist/resources/extensions/gsd/git-service.js +49 -1
  22. package/dist/resources/extensions/gsd/graph-context.js +98 -7
  23. package/dist/resources/extensions/gsd/gsd-db.js +260 -2
  24. package/dist/resources/extensions/gsd/guided-flow.js +24 -1
  25. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  26. package/dist/resources/extensions/gsd/journal.js +27 -0
  27. package/dist/resources/extensions/gsd/metrics.js +19 -0
  28. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  29. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  30. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  31. package/dist/resources/extensions/gsd/preferences-validation.js +108 -2
  32. package/dist/resources/extensions/gsd/preferences.js +26 -0
  33. package/dist/resources/extensions/gsd/safety/evidence-collector.js +15 -30
  34. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  35. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  36. package/dist/resources/extensions/gsd/tools/complete-slice.js +5 -0
  37. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  38. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  39. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  40. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  41. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  42. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  43. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  44. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  45. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  46. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  47. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  48. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  49. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  50. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  51. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  52. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  53. package/dist/web/standalone/.next/BUILD_ID +1 -1
  54. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  55. package/dist/web/standalone/.next/build-manifest.json +2 -2
  56. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  57. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  58. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  59. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  60. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  61. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  62. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  63. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  64. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  66. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  74. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  75. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  76. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  77. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  78. package/dist/web/standalone/.next/server/app/index.html +1 -1
  79. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  86. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  87. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  88. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  89. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  90. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  91. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  92. package/package.json +1 -1
  93. package/packages/mcp-server/dist/readers/graph.d.ts +1 -1
  94. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -1
  95. package/packages/mcp-server/dist/readers/graph.js +107 -0
  96. package/packages/mcp-server/dist/readers/graph.js.map +1 -1
  97. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  98. package/packages/mcp-server/dist/workflow-tools.js +88 -6
  99. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  100. package/packages/mcp-server/src/readers/graph.test.ts +178 -0
  101. package/packages/mcp-server/src/readers/graph.ts +148 -1
  102. package/packages/mcp-server/src/workflow-tools.ts +95 -10
  103. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  104. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  105. package/packages/pi-ai/dist/index.d.ts +1 -9
  106. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  107. package/packages/pi-ai/dist/index.js +1 -9
  108. package/packages/pi-ai/dist/index.js.map +1 -1
  109. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  110. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  111. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  112. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  113. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  114. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  115. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  116. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  117. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  118. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  119. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  120. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  121. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  122. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  123. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  124. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  125. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  126. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  127. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  128. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  129. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  130. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  131. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  132. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  133. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  134. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  135. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  136. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  137. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  138. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  139. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  140. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  141. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  142. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  143. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  144. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  145. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  146. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  147. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  148. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  149. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  150. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  151. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  152. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  153. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  154. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  155. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  156. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  157. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  158. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  159. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  160. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  161. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  162. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  163. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  164. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  165. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  166. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  167. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  168. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  169. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  170. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  171. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  172. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  174. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  176. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  178. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  180. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  182. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  184. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  186. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  188. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  190. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  192. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  194. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  196. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  198. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  200. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  202. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  203. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  204. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  205. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  206. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  208. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  210. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  212. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  213. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  214. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  215. package/packages/pi-ai/dist/models/index.js +80 -0
  216. package/packages/pi-ai/dist/models/index.js.map +1 -0
  217. package/packages/pi-ai/dist/models.d.ts +1 -36
  218. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  219. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  220. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  221. package/packages/pi-ai/dist/models.js +3 -112
  222. package/packages/pi-ai/dist/models.js.map +1 -1
  223. package/packages/pi-ai/dist/models.test.js +6 -5
  224. package/packages/pi-ai/dist/models.test.js.map +1 -1
  225. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  226. package/packages/pi-ai/src/index.ts +1 -9
  227. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  228. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  229. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  230. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  231. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  232. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  233. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  234. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  235. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  236. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  237. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  238. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  239. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  240. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  241. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  242. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  243. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  244. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  245. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  246. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  247. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  248. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  249. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  250. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  251. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  252. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  253. package/packages/pi-ai/src/models/index.ts +106 -0
  254. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  255. package/packages/pi-ai/src/models.test.ts +6 -5
  256. package/packages/pi-ai/src/models.ts +3 -153
  257. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  258. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  259. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
  260. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  261. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
  262. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  263. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  264. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  265. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  266. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  267. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  268. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  269. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  270. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
  271. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  272. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
  273. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  274. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
  275. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  276. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
  277. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
  278. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +47 -0
  279. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
  280. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +8 -0
  281. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  282. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +68 -8
  283. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  284. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
  286. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  288. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
  289. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  290. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  291. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  292. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  293. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  294. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +14 -0
  295. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  296. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +70 -6
  297. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  298. package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
  299. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
  300. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  301. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  302. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
  303. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
  304. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +67 -0
  305. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +83 -7
  306. package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
  307. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
  308. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  309. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +92 -6
  310. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  311. package/packages/pi-coding-agent/tsconfig.json +2 -2
  312. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  313. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  314. package/src/resources/extensions/gsd/auto/loop-deps.ts +4 -0
  315. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  316. package/src/resources/extensions/gsd/auto/phases.ts +123 -3
  317. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  318. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  319. package/src/resources/extensions/gsd/auto-model-selection.ts +66 -5
  320. package/src/resources/extensions/gsd/auto-post-unit.ts +226 -9
  321. package/src/resources/extensions/gsd/auto-recovery.ts +29 -9
  322. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  323. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  324. package/src/resources/extensions/gsd/auto.ts +34 -2
  325. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +15 -1
  326. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +72 -8
  327. package/src/resources/extensions/gsd/cache.ts +16 -5
  328. package/src/resources/extensions/gsd/commands/handlers/ops.ts +5 -0
  329. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  330. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +1 -1
  331. package/src/resources/extensions/gsd/docs/preferences-reference.md +14 -1
  332. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  333. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  334. package/src/resources/extensions/gsd/git-service.ts +68 -0
  335. package/src/resources/extensions/gsd/graph-context.ts +139 -12
  336. package/src/resources/extensions/gsd/gsd-db.ts +321 -3
  337. package/src/resources/extensions/gsd/guided-flow.ts +33 -1
  338. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  339. package/src/resources/extensions/gsd/journal.ts +30 -0
  340. package/src/resources/extensions/gsd/metrics.ts +26 -0
  341. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  342. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  343. package/src/resources/extensions/gsd/preferences-types.ts +32 -0
  344. package/src/resources/extensions/gsd/preferences-validation.ts +107 -2
  345. package/src/resources/extensions/gsd/preferences.ts +28 -0
  346. package/src/resources/extensions/gsd/safety/evidence-collector.ts +15 -31
  347. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  348. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  349. package/src/resources/extensions/gsd/templates/PREFERENCES.md +18 -0
  350. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +177 -0
  351. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  352. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  353. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  354. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +272 -0
  355. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  356. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  357. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  358. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  359. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  360. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  361. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  362. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  363. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  364. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  365. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  366. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  367. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  368. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  369. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  370. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  371. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  372. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  373. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  374. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  375. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  376. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  377. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  378. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  379. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  380. package/src/resources/extensions/gsd/tools/complete-slice.ts +9 -2
  381. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  382. package/src/resources/extensions/gsd/types.ts +14 -1
  383. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  384. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  385. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  386. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  387. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  388. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  389. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  390. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  391. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  392. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  393. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  394. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  395. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  396. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  397. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  398. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  399. package/packages/pi-ai/dist/models.generated.js +0 -14343
  400. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  401. package/packages/pi-ai/src/models.generated.ts +0 -14345
  402. /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → kn6xzWKYnogsxp2b6RpDD}/_buildManifest.js +0 -0
  403. /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → kn6xzWKYnogsxp2b6RpDD}/_ssgManifest.js +0 -0
@@ -10,12 +10,14 @@ import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
10
10
  import type { GSDPreferences } from "./preferences.js";
11
11
  import { resolveModelWithFallbacksForUnit, resolveDynamicRoutingConfig } from "./preferences.js";
12
12
  import type { ComplexityTier } from "./complexity-classifier.js";
13
- import { classifyUnitComplexity, tierLabel } from "./complexity-classifier.js";
13
+ import { classifyUnitComplexity, extractTaskMetadata, tierLabel } from "./complexity-classifier.js";
14
14
  import { resolveModelForComplexity, escalateTier, getEligibleModels, loadCapabilityOverrides, adjustToolSet, filterToolsForProvider } from "./model-router.js";
15
15
  import { getLedger, getProjectTotals } from "./metrics.js";
16
16
  import { unitPhaseLabel } from "./auto-dashboard.js";
17
17
  import { getSessionModelOverride } from "./session-model-override.js";
18
18
  import { logWarning } from "./workflow-logger.js";
19
+ import { resolveUokFlags } from "./uok/flags.js";
20
+ import { applyModelPolicyFilter } from "./uok/model-policy.js";
19
21
 
20
22
  export interface ModelSelectionResult {
21
23
  /** Routing metadata for metrics recording */
@@ -75,6 +77,7 @@ export async function selectAndApplyModel(
75
77
  /** Explicit /gsd model pin captured at bootstrap for long-running auto loops. */
76
78
  sessionModelOverride?: { provider: string; id: string } | null,
77
79
  ): Promise<ModelSelectionResult> {
80
+ const uokFlags = resolveUokFlags(prefs);
78
81
  const effectiveSessionModelOverride = sessionModelOverride === undefined
79
82
  ? getSessionModelOverride(ctx.sessionManager.getSessionId())
80
83
  : (sessionModelOverride ?? undefined);
@@ -97,6 +100,9 @@ export async function selectAndApplyModel(
97
100
 
98
101
  if (modelConfig) {
99
102
  const availableModels = ctx.modelRegistry.getAvailable();
103
+ const modelPolicyTraceId = `model:${ctx.sessionManager.getSessionId()}:${Date.now()}`;
104
+ const modelPolicyTurnId = `${unitType}:${unitId}`;
105
+ let policyAllowedModelKeys: Set<string> | null = null;
100
106
 
101
107
  // ─── Dynamic Model Routing ─────────────────────────────────────────
102
108
  // Dynamic routing (complexity-based downgrading) only applies in auto-mode.
@@ -106,8 +112,40 @@ export async function selectAndApplyModel(
106
112
  if (!isAutoMode) {
107
113
  routingConfig.enabled = false;
108
114
  }
115
+ // burn-max defaults to quality-first dispatch (no downgrade routing).
116
+ if (prefs?.token_profile === "burn-max") {
117
+ routingConfig.enabled = false;
118
+ }
109
119
  let effectiveModelConfig = modelConfig;
110
120
  let routingTierLabel = "";
121
+ let routingEligibleModels = availableModels;
122
+
123
+ const taskMetadataForPolicy = unitType === "execute-task"
124
+ ? extractTaskMetadata(unitId, basePath)
125
+ : undefined;
126
+
127
+ if (uokFlags.modelPolicy) {
128
+ const policy = applyModelPolicyFilter(
129
+ availableModels,
130
+ {
131
+ basePath,
132
+ traceId: modelPolicyTraceId,
133
+ turnId: modelPolicyTurnId,
134
+ unitType,
135
+ taskMetadata: taskMetadataForPolicy,
136
+ currentProvider: ctx.model?.provider,
137
+ allowCrossProvider: routingConfig.cross_provider !== false,
138
+ requiredTools: pi.getActiveTools(),
139
+ },
140
+ );
141
+ routingEligibleModels = policy.eligible;
142
+ policyAllowedModelKeys = new Set(
143
+ policy.eligible.map((m) => `${m.provider.toLowerCase()}/${m.id.toLowerCase()}`),
144
+ );
145
+ if (routingEligibleModels.length === 0) {
146
+ throw new Error(`Model policy denied all candidate models for ${unitType}/${unitId}`);
147
+ }
148
+ }
111
149
 
112
150
  // Disable routing for flat-rate providers like GitHub Copilot (#3453).
113
151
  // All models cost the same per request, so downgrading to a cheaper
@@ -115,7 +153,7 @@ export async function selectAndApplyModel(
115
153
  // Fail-closed: if primary model can't be resolved, fall back to
116
154
  // provider-level signals rather than allowing unwanted downgrades.
117
155
  if (routingConfig.enabled) {
118
- const primaryModel = resolveModelId(modelConfig.primary, availableModels, ctx.model?.provider);
156
+ const primaryModel = resolveModelId(modelConfig.primary, routingEligibleModels, ctx.model?.provider);
119
157
  if (primaryModel) {
120
158
  const primaryFlatRateCtx = buildFlatRateContext(primaryModel.provider, ctx, prefs);
121
159
  if (isFlatRateProvider(primaryModel.provider, primaryFlatRateCtx)) {
@@ -149,8 +187,14 @@ export async function selectAndApplyModel(
149
187
  const shouldClassify = !isHook || routingConfig.hooks !== false;
150
188
 
151
189
  if (shouldClassify) {
152
- let classification = classifyUnitComplexity(unitType, unitId, basePath, budgetPct);
153
- const availableModelIds = availableModels.map(m => m.id);
190
+ let classification = classifyUnitComplexity(
191
+ unitType,
192
+ unitId,
193
+ basePath,
194
+ budgetPct,
195
+ taskMetadataForPolicy,
196
+ );
197
+ const availableModelIds = routingEligibleModels.map(m => m.id);
154
198
 
155
199
  // Escalate tier on retry when escalate_on_failure is enabled (default: true)
156
200
  if (
@@ -257,15 +301,28 @@ export async function selectAndApplyModel(
257
301
  }
258
302
 
259
303
  const modelsToTry = [effectiveModelConfig.primary, ...effectiveModelConfig.fallbacks];
304
+ let attemptedPolicyEligible = false;
260
305
 
261
306
  for (const modelId of modelsToTry) {
262
- const model = resolveModelId(modelId, availableModels, ctx.model?.provider);
307
+ const resolutionPool = uokFlags.modelPolicy ? routingEligibleModels : availableModels;
308
+ const model = resolveModelId(modelId, resolutionPool, ctx.model?.provider);
263
309
 
264
310
  if (!model) {
265
311
  if (verbose) ctx.ui.notify(`Model ${modelId} not found, trying fallback.`, "info");
266
312
  continue;
267
313
  }
268
314
 
315
+ if (policyAllowedModelKeys) {
316
+ const key = `${model.provider.toLowerCase()}/${model.id.toLowerCase()}`;
317
+ if (!policyAllowedModelKeys.has(key)) {
318
+ if (verbose) {
319
+ ctx.ui.notify(`Model policy denied ${model.provider}/${model.id}; trying fallback.`, "warning");
320
+ }
321
+ continue;
322
+ }
323
+ attemptedPolicyEligible = true;
324
+ }
325
+
269
326
  // Warn if the ID is ambiguous across providers
270
327
  if (!modelId.includes("/")) {
271
328
  const providers = availableModels.filter(m => m.id === modelId).map(m => m.provider);
@@ -331,6 +388,10 @@ export async function selectAndApplyModel(
331
388
  }
332
389
  }
333
390
  }
391
+
392
+ if (uokFlags.modelPolicy && policyAllowedModelKeys && !attemptedPolicyEligible) {
393
+ throw new Error(`Model policy denied dispatch for ${unitType}/${unitId} before prompt send`);
394
+ }
334
395
  } else if (autoModeStartModel) {
335
396
  // No model preference for this unit type — re-apply the model captured
336
397
  // at auto-mode start to prevent bleed from shared global settings.json (#650).
@@ -29,9 +29,10 @@ import { rebuildState } from "./doctor.js";
29
29
  import { parseUnitId } from "./unit-id.js";
30
30
  import { closeoutUnit, type CloseoutOptions } from "./auto-unit-closeout.js";
31
31
  import {
32
- autoCommitCurrentBranch,
32
+ runTurnGitAction,
33
33
  type TaskCommitContext,
34
- } from "./worktree.js";
34
+ type TurnGitActionMode,
35
+ } from "./git-service.js";
35
36
  import {
36
37
  verifyExpectedArtifact,
37
38
  resolveExpectedArtifactPath,
@@ -66,6 +67,9 @@ import { getSliceTasks } from "./gsd-db.js";
66
67
  import { runPreExecutionChecks, type PreExecutionResult } from "./pre-execution-checks.js";
67
68
  import { writePreExecutionEvidence } from "./verification-evidence.js";
68
69
  import { ensureCodebaseMapFresh } from "./codebase-generator.js";
70
+ import { resolveUokFlags } from "./uok/flags.js";
71
+ import { UokGateRunner } from "./uok/gate-runner.js";
72
+ import { writeTurnGitTransaction } from "./uok/gitops.js";
69
73
 
70
74
  /** Maximum verification retry attempts before escalating to blocker placeholder (#2653). */
71
75
  const MAX_VERIFICATION_RETRIES = 3;
@@ -109,6 +113,7 @@ import {
109
113
  import { existsSync, unlinkSync } from "node:fs";
110
114
  import { join } from "node:path";
111
115
  import { _resetHasChangesCache } from "./native-git-bridge.js";
116
+ import { autoCommitCurrentBranch } from "./worktree.js";
112
117
 
113
118
  // ─── Rogue File Detection ──────────────────────────────────────────────────
114
119
 
@@ -357,10 +362,161 @@ export async function postUnitPreVerification(pctx: PostUnitContext, opts?: PreV
357
362
  await new Promise(r => setTimeout(r, 100));
358
363
  }
359
364
 
360
- // Auto-commit
365
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
366
+ const uokFlags = resolveUokFlags(prefs);
367
+
368
+ // Turn-level git action (commit | snapshot | status-only)
361
369
  if (s.currentUnit) {
362
370
  const unit = s.currentUnit;
363
- await autoCommitUnit(s.basePath, unit.type, unit.id, ctx);
371
+ const turnAction: TurnGitActionMode = uokFlags.gitops ? uokFlags.gitopsTurnAction : "commit";
372
+ const traceId = s.currentTraceId ?? `turn:${unit.startedAt}`;
373
+ const turnId = s.currentTurnId ?? `${unit.type}/${unit.id}/${unit.startedAt}`;
374
+ s.lastGitActionFailure = null;
375
+ s.lastGitActionStatus = null;
376
+ try {
377
+ let taskContext: TaskCommitContext | undefined;
378
+
379
+ if (turnAction === "commit" && s.currentUnit.type === "execute-task") {
380
+ const { milestone: mid, slice: sid, task: tid } = parseUnitId(s.currentUnit.id);
381
+ if (mid && sid && tid) {
382
+ const summaryPath = resolveTaskFile(s.basePath, mid, sid, tid, "SUMMARY");
383
+ if (summaryPath) {
384
+ try {
385
+ const summaryContent = await loadFile(summaryPath);
386
+ if (summaryContent) {
387
+ const summary = parseSummary(summaryContent);
388
+ // Look up GitHub issue number for commit linking
389
+ let ghIssueNumber: number | undefined;
390
+ try {
391
+ const { getTaskIssueNumberForCommit } = await import("../github-sync/sync.js");
392
+ ghIssueNumber = getTaskIssueNumberForCommit(s.basePath, mid, sid, tid) ?? undefined;
393
+ } catch (err) {
394
+ // GitHub sync not available — skip
395
+ logWarning("engine", `GitHub issue lookup failed: ${err instanceof Error ? err.message : String(err)}`);
396
+ }
397
+
398
+ taskContext = {
399
+ taskId: `${sid}/${tid}`,
400
+ taskTitle: summary.title?.replace(/^T\d+:\s*/, "") || tid,
401
+ oneLiner: summary.oneLiner || undefined,
402
+ keyFiles: summary.frontmatter.key_files?.filter(f => !f.includes("{{")) || undefined,
403
+ issueNumber: ghIssueNumber,
404
+ };
405
+ }
406
+ } catch (e) {
407
+ debugLog("postUnit", { phase: "task-summary-parse", error: String(e) });
408
+ }
409
+ }
410
+ }
411
+ }
412
+
413
+ // Invalidate the nativeHasChanges cache before auto-commit (#1853).
414
+ // The cache has a 10-second TTL and is keyed by basePath. A stale
415
+ // `false` result causes autoCommit to skip staging entirely, leaving
416
+ // code files only in the working tree where they are destroyed by
417
+ // `git worktree remove --force` during teardown.
418
+ _resetHasChangesCache();
419
+
420
+ const skipLifecycleCommit =
421
+ turnAction === "commit" && LIFECYCLE_ONLY_UNITS.has(s.currentUnit.type);
422
+
423
+ if (skipLifecycleCommit) {
424
+ debugLog("postUnit", {
425
+ phase: "git-action-skipped",
426
+ reason: "lifecycle-only-unit",
427
+ unitType: s.currentUnit.type,
428
+ unitId: s.currentUnit.id,
429
+ });
430
+ } else {
431
+ const gitResult = runTurnGitAction({
432
+ basePath: s.basePath,
433
+ action: turnAction,
434
+ unitType: s.currentUnit.type,
435
+ unitId: s.currentUnit.id,
436
+ taskContext,
437
+ });
438
+
439
+ if (uokFlags.gitops) {
440
+ writeTurnGitTransaction({
441
+ basePath: s.basePath,
442
+ traceId,
443
+ turnId,
444
+ unitType: unit.type,
445
+ unitId: unit.id,
446
+ stage: "publish",
447
+ action: turnAction,
448
+ push: uokFlags.gitopsTurnPush,
449
+ status: gitResult.status,
450
+ error: gitResult.error,
451
+ metadata: {
452
+ dirty: gitResult.dirty,
453
+ commitMessage: gitResult.commitMessage,
454
+ snapshotLabel: gitResult.snapshotLabel,
455
+ },
456
+ });
457
+ }
458
+
459
+ if (gitResult.status === "failed") {
460
+ s.lastGitActionFailure = gitResult.error ?? `git ${turnAction} failed`;
461
+ s.lastGitActionStatus = "failed";
462
+ if (uokFlags.gitops && uokFlags.gates) {
463
+ const parsed = parseUnitId(unit.id);
464
+ const gateRunner = new UokGateRunner();
465
+ gateRunner.register({
466
+ id: "closeout-git-action",
467
+ type: "closeout",
468
+ execute: async () => ({
469
+ outcome: "fail",
470
+ failureClass: "git",
471
+ rationale: `turn git action "${turnAction}" failed`,
472
+ findings: gitResult.error ?? "unknown git failure",
473
+ }),
474
+ });
475
+ await gateRunner.run("closeout-git-action", {
476
+ basePath: s.basePath,
477
+ traceId,
478
+ turnId,
479
+ milestoneId: parsed.milestone ?? undefined,
480
+ sliceId: parsed.slice ?? undefined,
481
+ taskId: parsed.task ?? undefined,
482
+ unitType: unit.type,
483
+ unitId: unit.id,
484
+ });
485
+ }
486
+
487
+ const failureMsg = `Git ${turnAction} failed: ${(gitResult.error ?? "unknown error").split("\n")[0]}`;
488
+ if (uokFlags.gitops) {
489
+ ctx.ui.notify(failureMsg, "error");
490
+ await pauseAuto(ctx, pi);
491
+ return "dispatched";
492
+ }
493
+ ctx.ui.notify(failureMsg, "warning");
494
+ debugLog("postUnit", {
495
+ phase: "git-action-failed-nonblocking",
496
+ action: turnAction,
497
+ error: gitResult.error ?? "unknown error",
498
+ });
499
+ }
500
+
501
+ s.lastGitActionStatus = "ok";
502
+
503
+ if (turnAction === "commit" && gitResult.commitMessage) {
504
+ ctx.ui.notify(`Committed: ${gitResult.commitMessage.split("\n")[0]}`, "info");
505
+ } else if (turnAction === "snapshot" && gitResult.snapshotLabel) {
506
+ ctx.ui.notify(`Snapshot recorded: ${gitResult.snapshotLabel}`, "info");
507
+ }
508
+ }
509
+ } catch (e) {
510
+ const message = e instanceof Error ? e.message : String(e);
511
+ s.lastGitActionFailure = message;
512
+ s.lastGitActionStatus = "failed";
513
+ debugLog("postUnit", { phase: "git-action", error: message, action: turnAction });
514
+ ctx.ui.notify(`Git ${turnAction} failed: ${message.split("\n")[0]}`, uokFlags.gitops ? "error" : "warning");
515
+ if (uokFlags.gitops) {
516
+ await pauseAuto(ctx, pi);
517
+ return "dispatched";
518
+ }
519
+ }
364
520
 
365
521
  // GitHub sync (non-blocking, opt-in)
366
522
  await runSafely("postUnit", "github-sync", async () => {
@@ -869,11 +1025,13 @@ export async function postUnitPostVerification(pctx: PostUnitContext): Promise<"
869
1025
  s.currentUnit &&
870
1026
  s.currentUnit.type === "plan-slice"
871
1027
  ) {
1028
+ const currentUnit = s.currentUnit;
872
1029
  let preExecPauseNeeded = false;
873
1030
  await runSafely("postUnitPostVerification", "pre-execution-checks", async () => {
1031
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
1032
+ const uokFlags = resolveUokFlags(prefs);
874
1033
  try {
875
1034
  // Check preferences — respect enhanced_verification and enhanced_verification_pre
876
- const prefs = loadEffectiveGSDPreferences()?.preferences;
877
1035
  const enhancedEnabled = prefs?.enhanced_verification !== false; // default true
878
1036
  const preEnabled = prefs?.enhanced_verification_pre !== false; // default true
879
1037
 
@@ -887,7 +1045,7 @@ export async function postUnitPostVerification(pctx: PostUnitContext): Promise<"
887
1045
  }
888
1046
 
889
1047
  // Parse the unit ID to get milestone/slice IDs
890
- const { milestone: mid, slice: sid } = parseUnitId(s.currentUnit!.id);
1048
+ const { milestone: mid, slice: sid } = parseUnitId(currentUnit.id);
891
1049
  if (!mid || !sid) {
892
1050
  debugLog("postUnitPostVerification", {
893
1051
  phase: "pre-execution-checks",
@@ -908,6 +1066,8 @@ export async function postUnitPostVerification(pctx: PostUnitContext): Promise<"
908
1066
  return;
909
1067
  }
910
1068
 
1069
+ const strictMode = prefs?.enhanced_verification_strict === true;
1070
+
911
1071
  // Run pre-execution checks
912
1072
  const result: PreExecutionResult = await runPreExecutionChecks(tasks, s.basePath);
913
1073
 
@@ -931,11 +1091,45 @@ export async function postUnitPostVerification(pctx: PostUnitContext): Promise<"
931
1091
  writePreExecutionEvidence(result, slicePath, mid, sid);
932
1092
  }
933
1093
 
934
- // Notify UI
1094
+ if (uokFlags.gates) {
1095
+ const failedChecks = result.checks
1096
+ .filter((check) => !check.passed)
1097
+ .map((check) => `[${check.category}] ${check.target}: ${check.message}`);
1098
+ const warnEscalated = result.status === "warn" && strictMode;
1099
+ const blockingFailure = result.status === "fail" || warnEscalated;
1100
+ const gateRunner = new UokGateRunner();
1101
+ gateRunner.register({
1102
+ id: "pre-execution-checks",
1103
+ type: "input",
1104
+ execute: async () => ({
1105
+ outcome: blockingFailure ? "fail" : "pass",
1106
+ failureClass: result.status === "fail" ? "input" : warnEscalated ? "policy" : "none",
1107
+ rationale: blockingFailure
1108
+ ? `pre-execution checks ${result.status}${warnEscalated ? " (strict)" : ""}`
1109
+ : "pre-execution checks passed",
1110
+ findings: failedChecks.join("\n"),
1111
+ }),
1112
+ });
1113
+ await gateRunner.run("pre-execution-checks", {
1114
+ basePath: s.basePath,
1115
+ traceId: `pre-execution:${currentUnit.id}`,
1116
+ turnId: currentUnit.id,
1117
+ milestoneId: mid,
1118
+ sliceId: sid,
1119
+ unitType: currentUnit.type,
1120
+ unitId: currentUnit.id,
1121
+ });
1122
+ }
1123
+
1124
+ // Notify UI — surface actionable details (#4259)
935
1125
  if (result.status === "fail") {
936
- const blockingCount = result.checks.filter(c => !c.passed && c.blocking).length;
1126
+ const blockingChecks = result.checks.filter(c => !c.passed && c.blocking);
1127
+ const blockingCount = blockingChecks.length;
1128
+ const details = blockingChecks.slice(0, 3).map(c => ` \u2022 ${c.message}`).join("\n");
1129
+ const suffix = blockingChecks.length > 3 ? `\n \u2022 ...and ${blockingChecks.length - 3} more` : "";
1130
+ const evidenceNote = `\nSee ${sid}-PRE-EXEC-VERIFY.json for full details.`;
937
1131
  ctx.ui.notify(
938
- `Pre-execution checks failed: ${blockingCount} blocking issue${blockingCount === 1 ? "" : "s"} found`,
1132
+ `Pre-execution checks failed: ${blockingCount} blocking issue${blockingCount === 1 ? "" : "s"} found\n${details}${suffix}${evidenceNote}`,
939
1133
  "error",
940
1134
  );
941
1135
  preExecPauseNeeded = true;
@@ -969,6 +1163,29 @@ export async function postUnitPostVerification(pctx: PostUnitContext): Promise<"
969
1163
  `Pre-execution checks error: ${errorMessage} — pausing for human review`,
970
1164
  "error",
971
1165
  );
1166
+ if (uokFlags.gates && s.currentUnit) {
1167
+ const { milestone: mid, slice: sid } = parseUnitId(s.currentUnit.id);
1168
+ const gateRunner = new UokGateRunner();
1169
+ gateRunner.register({
1170
+ id: "pre-execution-checks",
1171
+ type: "input",
1172
+ execute: async () => ({
1173
+ outcome: "manual-attention",
1174
+ failureClass: "manual-attention",
1175
+ rationale: "pre-execution checks threw before completion",
1176
+ findings: errorMessage,
1177
+ }),
1178
+ });
1179
+ await gateRunner.run("pre-execution-checks", {
1180
+ basePath: s.basePath,
1181
+ traceId: `pre-execution:${s.currentUnit.id}`,
1182
+ turnId: s.currentUnit.id,
1183
+ milestoneId: mid ?? undefined,
1184
+ sliceId: sid ?? undefined,
1185
+ unitType: s.currentUnit.type,
1186
+ unitId: s.currentUnit.id,
1187
+ });
1188
+ }
972
1189
  preExecPauseNeeded = true;
973
1190
  }
974
1191
  });
@@ -264,18 +264,30 @@ export function verifyExpectedArtifact(
264
264
  const absPath = resolveExpectedArtifactPath(unitType, unitId, base);
265
265
  // For unit types with no verifiable artifact (null path), the parent directory
266
266
  // is missing on disk — treat as stale completion state so the key gets evicted (#313).
267
- if (!absPath) return false;
268
- if (!existsSync(absPath)) return false;
267
+ if (!absPath) {
268
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: resolveExpectedArtifactPath returned null (parent dir missing)`);
269
+ return false;
270
+ }
271
+ if (!existsSync(absPath)) {
272
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: existsSync false for ${absPath}`);
273
+ return false;
274
+ }
269
275
 
270
276
  if (unitType === "validate-milestone") {
271
277
  const validationContent = readFileSync(absPath, "utf-8");
272
- if (!isValidationTerminal(validationContent)) return false;
278
+ if (!isValidationTerminal(validationContent)) {
279
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: validation not terminal (len=${validationContent.length}) at ${absPath}`);
280
+ return false;
281
+ }
273
282
  }
274
283
 
275
284
  if (unitType === "plan-milestone") {
276
285
  try {
277
286
  const roadmap = parseLegacyRoadmap(readFileSync(absPath, "utf-8"));
278
- if (roadmap.slices.length === 0) return false;
287
+ if (roadmap.slices.length === 0) {
288
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: roadmap has zero slices at ${absPath}`);
289
+ return false;
290
+ }
279
291
  } catch (err) {
280
292
  logWarning("recovery", `plan-milestone roadmap verification failed: ${err instanceof Error ? err.message : String(err)}`);
281
293
  return false;
@@ -292,7 +304,10 @@ export function verifyExpectedArtifact(
292
304
  // Accept checkbox-style (- [x] **T01: ...) or heading-style (### T01 -- / ### T01: / ### T01 —)
293
305
  const hasCheckboxTask = /^- \[[xX ]\] \*\*T\d+:/m.test(planContent);
294
306
  const hasHeadingTask = /^#{2,4}\s+T\d+\s*(?:--|—|:)/m.test(planContent);
295
- if (!hasCheckboxTask && !hasHeadingTask) return false;
307
+ if (!hasCheckboxTask && !hasHeadingTask) {
308
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: plan has no task checkbox/heading (len=${planContent.length}) at ${absPath}`);
309
+ return false;
310
+ }
296
311
  }
297
312
 
298
313
  // execute-task: DB status is authoritative. Fall back to checked-checkbox
@@ -349,10 +364,15 @@ export function verifyExpectedArtifact(
349
364
 
350
365
  if (taskIds && taskIds.length > 0) {
351
366
  const tasksDir = resolveTasksDir(base, mid, sid);
352
- if (tasksDir) {
353
- for (const tid of taskIds) {
354
- const taskPlanFile = join(tasksDir, `${tid}-PLAN.md`);
355
- if (!existsSync(taskPlanFile)) return false;
367
+ if (!tasksDir) {
368
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: resolveTasksDir returned null for ${mid}/${sid}`);
369
+ return false;
370
+ }
371
+ for (const tid of taskIds) {
372
+ const taskPlanFile = join(tasksDir, `${tid}-PLAN.md`);
373
+ if (!existsSync(taskPlanFile)) {
374
+ logWarning("recovery", `verify-fail ${unitType} ${unitId}: task plan missing ${taskPlanFile}`);
375
+ return false;
356
376
  }
357
377
  }
358
378
  }
@@ -8,6 +8,7 @@ import type { ExtensionContext } from "@gsd/pi-coding-agent";
8
8
  import { snapshotUnitMetrics } from "./metrics.js";
9
9
  import { saveActivityLog } from "./activity-log.js";
10
10
  import { logWarning } from "./workflow-logger.js";
11
+ import { writeTurnGitTransaction } from "./uok/gitops.js";
11
12
 
12
13
  export interface CloseoutOptions {
13
14
  promptCharCount?: number;
@@ -15,6 +16,12 @@ export interface CloseoutOptions {
15
16
  tier?: string;
16
17
  modelDowngraded?: boolean;
17
18
  continueHereFired?: boolean;
19
+ traceId?: string;
20
+ turnId?: string;
21
+ gitAction?: "commit" | "snapshot" | "status-only";
22
+ gitPush?: boolean;
23
+ gitStatus?: "ok" | "failed";
24
+ gitError?: string;
18
25
  }
19
26
 
20
27
  /**
@@ -47,6 +54,23 @@ export async function closeoutUnit(
47
54
  }
48
55
  }
49
56
 
57
+ if (opts?.traceId && opts.turnId && opts.gitAction && opts.gitStatus) {
58
+ writeTurnGitTransaction({
59
+ basePath,
60
+ traceId: opts.traceId,
61
+ turnId: opts.turnId,
62
+ unitType,
63
+ unitId,
64
+ stage: "record",
65
+ action: opts.gitAction,
66
+ push: opts.gitPush === true,
67
+ status: opts.gitStatus,
68
+ error: opts.gitError,
69
+ metadata: {
70
+ activityFile,
71
+ },
72
+ });
73
+ }
74
+
50
75
  return activityFile ?? undefined;
51
76
  }
52
-