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
@@ -33,6 +33,8 @@ import { runPostExecutionChecks, type PostExecutionResult } from "./post-executi
33
33
  import type { AutoSession } from "./auto/session.js";
34
34
  import type { VerificationResult as VerificationGateResult } from "./types.js";
35
35
  import { join } from "node:path";
36
+ import { resolveUokFlags } from "./uok/flags.js";
37
+ import { UokGateRunner } from "./uok/gate-runner.js";
36
38
 
37
39
  export interface VerificationContext {
38
40
  s: AutoSession;
@@ -67,6 +69,37 @@ async function runValidateMilestonePostCheck(
67
69
  pauseAuto: (ctx?: ExtensionContext, pi?: ExtensionAPI) => Promise<void>,
68
70
  ): Promise<VerificationResult> {
69
71
  const { s, ctx, pi } = vctx;
72
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
73
+ const uokFlags = resolveUokFlags(prefs);
74
+ const persistMilestoneValidationGate = async (
75
+ outcome: "pass" | "fail" | "retry" | "manual-attention",
76
+ failureClass: "none" | "verification" | "manual-attention",
77
+ rationale: string,
78
+ findings = "",
79
+ milestoneId?: string,
80
+ ): Promise<void> => {
81
+ if (!uokFlags.gates || !s.currentUnit) return;
82
+ const gateRunner = new UokGateRunner();
83
+ gateRunner.register({
84
+ id: "milestone-validation-post-check",
85
+ type: "verification",
86
+ execute: async () => ({
87
+ outcome,
88
+ failureClass,
89
+ rationale,
90
+ findings,
91
+ }),
92
+ });
93
+ await gateRunner.run("milestone-validation-post-check", {
94
+ basePath: s.basePath,
95
+ traceId: `validation-post-check:${s.currentUnit.id}`,
96
+ turnId: s.currentUnit.id,
97
+ milestoneId,
98
+ unitType: s.currentUnit.type,
99
+ unitId: s.currentUnit.id,
100
+ });
101
+ };
102
+
70
103
  if (!s.currentUnit) return "continue";
71
104
 
72
105
  const { milestone: mid } = parseUnitId(s.currentUnit.id);
@@ -79,14 +112,32 @@ async function runValidateMilestonePostCheck(
79
112
  if (!validationContent) return "continue";
80
113
 
81
114
  const verdict = extractVerdict(validationContent);
82
- if (verdict !== "needs-remediation") return "continue";
115
+ if (verdict !== "needs-remediation") {
116
+ await persistMilestoneValidationGate(
117
+ "pass",
118
+ "none",
119
+ `milestone validation verdict is ${verdict}; no remediation loop risk`,
120
+ "",
121
+ mid,
122
+ );
123
+ return "continue";
124
+ }
83
125
 
84
126
  const incompleteSliceCount = await countIncompleteSlices(s.basePath, mid);
85
127
 
86
128
  // If any non-closed slices exist, the agent successfully queued remediation
87
129
  // work — proceed normally. The state machine will execute those slices and
88
130
  // re-validate per the #3596/#3670 fix.
89
- if (incompleteSliceCount > 0) return "continue";
131
+ if (incompleteSliceCount > 0) {
132
+ await persistMilestoneValidationGate(
133
+ "pass",
134
+ "none",
135
+ `remediation slices present (${incompleteSliceCount}); validation can continue`,
136
+ "",
137
+ mid,
138
+ );
139
+ return "continue";
140
+ }
90
141
 
91
142
  ctx.ui.notify(
92
143
  `Milestone ${mid} validation returned verdict=needs-remediation but no remediation slices were added. Pausing for human review.`,
@@ -96,6 +147,13 @@ async function runValidateMilestonePostCheck(
96
147
  `validate-milestone: pausing — verdict=needs-remediation with no incomplete slices for ${mid}. ` +
97
148
  `The agent must call gsd_reassess_roadmap to add remediation slices before re-validation.\n`,
98
149
  );
150
+ await persistMilestoneValidationGate(
151
+ "manual-attention",
152
+ "manual-attention",
153
+ "needs-remediation verdict without queued remediation slices",
154
+ `No incomplete slices found for ${mid} while verdict=needs-remediation`,
155
+ mid,
156
+ );
99
157
  await pauseAuto(ctx, pi);
100
158
  return "pause";
101
159
  }
@@ -158,6 +216,7 @@ export async function runPostUnitVerification(
158
216
  try {
159
217
  const effectivePrefs = loadEffectiveGSDPreferences();
160
218
  const prefs = effectivePrefs?.preferences;
219
+ const uokFlags = resolveUokFlags(prefs);
161
220
 
162
221
  // Read task plan verify field
163
222
  const { milestone: mid, slice: sid, task: tid } = parseUnitId(s.currentUnit.id);
@@ -196,6 +255,37 @@ export async function runPostUnitVerification(
196
255
  }
197
256
  }
198
257
 
258
+ if (uokFlags.gates) {
259
+ const gateRunner = new UokGateRunner();
260
+ gateRunner.register({
261
+ id: "verification-gate",
262
+ type: "verification",
263
+ execute: async () => ({
264
+ outcome: result.passed ? "pass" : "fail",
265
+ failureClass: result.runtimeErrors?.some((e) => e.blocking)
266
+ ? "execution"
267
+ : "verification",
268
+ rationale: result.passed
269
+ ? "verification checks passed"
270
+ : "verification checks failed",
271
+ findings: result.passed
272
+ ? ""
273
+ : formatFailureContext(result),
274
+ }),
275
+ });
276
+
277
+ await gateRunner.run("verification-gate", {
278
+ basePath: s.basePath,
279
+ traceId: `verification:${s.currentUnit.id}`,
280
+ turnId: s.currentUnit.id,
281
+ milestoneId: mid ?? undefined,
282
+ sliceId: sid ?? undefined,
283
+ taskId: tid ?? undefined,
284
+ unitType: s.currentUnit.type,
285
+ unitId: s.currentUnit.id,
286
+ });
287
+ }
288
+
199
289
  // Auto-fix retry preferences
200
290
  const autoFixEnabled = prefs?.verification_auto_fix !== false;
201
291
  const maxRetries =
@@ -338,6 +428,43 @@ export async function runPostUnitVerification(
338
428
  );
339
429
  }
340
430
 
431
+ if (uokFlags.gates) {
432
+ const strictMode = prefs?.enhanced_verification_strict === true;
433
+ const warnEscalated = postExecResult.status === "warn" && strictMode;
434
+ const blockingFailure = postExecResult.status === "fail" || warnEscalated;
435
+ const findings = postExecResult.checks
436
+ .filter((check) => !check.passed)
437
+ .map((check) => `[${check.category}] ${check.target}: ${check.message}`)
438
+ .join("\n");
439
+ const gateRunner = new UokGateRunner();
440
+ gateRunner.register({
441
+ id: "post-execution-checks",
442
+ type: "artifact",
443
+ execute: async () => ({
444
+ outcome: blockingFailure ? "fail" : "pass",
445
+ failureClass: postExecResult.status === "fail"
446
+ ? "artifact"
447
+ : warnEscalated
448
+ ? "policy"
449
+ : "none",
450
+ rationale: blockingFailure
451
+ ? `post-execution checks ${postExecResult.status}${warnEscalated ? " (strict)" : ""}`
452
+ : "post-execution checks passed",
453
+ findings,
454
+ }),
455
+ });
456
+ await gateRunner.run("post-execution-checks", {
457
+ basePath: s.basePath,
458
+ traceId: `verification:${s.currentUnit.id}`,
459
+ turnId: s.currentUnit.id,
460
+ milestoneId: mid,
461
+ sliceId: sid,
462
+ taskId: tid,
463
+ unitType: s.currentUnit.type,
464
+ unitId: s.currentUnit.id,
465
+ });
466
+ }
467
+
341
468
  // Check for blocking failures
342
469
  if (postExecResult.status === "fail") {
343
470
  postExecBlockingFailure = true;
@@ -202,6 +202,8 @@ import {
202
202
  import { bootstrapAutoSession, openProjectDbIfPresent, type BootstrapDeps } from "./auto-start.js";
203
203
  import { initHealthWidget } from "./health-widget.js";
204
204
  import { autoLoop, resolveAgentEnd, resolveAgentEndCancelled, _resetPendingResolve, isSessionSwitchInFlight, type LoopDeps, type ErrorContext } from "./auto-loop.js";
205
+ import { runAutoLoopWithUok } from "./uok/kernel.js";
206
+ import { resolveUokFlags } from "./uok/flags.js";
205
207
  // Slice-level parallelism (#2340)
206
208
  import { getEligibleSlices } from "./slice-parallel-eligibility.js";
207
209
  import { startSliceParallel } from "./slice-parallel-orchestrator.js";
@@ -605,11 +607,29 @@ function buildSnapshotOpts(
605
607
  continueHereFired?: boolean;
606
608
  promptCharCount?: number;
607
609
  baselineCharCount?: number;
610
+ traceId?: string;
611
+ turnId?: string;
612
+ gitAction?: "commit" | "snapshot" | "status-only";
613
+ gitPush?: boolean;
614
+ gitStatus?: "ok" | "failed";
615
+ gitError?: string;
608
616
  } & Record<string, unknown> {
617
+ const prefs = loadEffectiveGSDPreferences()?.preferences;
618
+ const uokFlags = resolveUokFlags(prefs);
609
619
  return {
610
620
  ...(s.autoStartTime > 0 ? { autoSessionKey: String(s.autoStartTime) } : {}),
611
621
  promptCharCount: s.lastPromptCharCount,
612
622
  baselineCharCount: s.lastBaselineCharCount,
623
+ traceId: s.currentTraceId ?? undefined,
624
+ turnId: s.currentTurnId ?? undefined,
625
+ ...(uokFlags.gitops
626
+ ? {
627
+ gitAction: uokFlags.gitopsTurnAction,
628
+ gitPush: uokFlags.gitopsTurnPush,
629
+ gitStatus: s.lastGitActionStatus ?? undefined,
630
+ gitError: s.lastGitActionFailure ?? undefined,
631
+ }
632
+ : {}),
613
633
  ...(s.currentUnitRouting ?? {}),
614
634
  };
615
635
  }
@@ -1513,7 +1533,13 @@ export async function startAuto(
1513
1533
  logCmuxEvent(loadEffectiveGSDPreferences()?.preferences, s.stepMode ? "Step-mode resumed." : "Auto-mode resumed.", "progress");
1514
1534
 
1515
1535
  captureProjectRootEnv(s.originalBasePath || s.basePath);
1516
- await autoLoop(ctx, pi, s, buildLoopDeps());
1536
+ await runAutoLoopWithUok({
1537
+ ctx,
1538
+ pi,
1539
+ s,
1540
+ deps: buildLoopDeps(),
1541
+ runLegacyLoop: autoLoop,
1542
+ });
1517
1543
  cleanupAfterLoopExit(ctx);
1518
1544
  return;
1519
1545
  }
@@ -1548,7 +1574,13 @@ export async function startAuto(
1548
1574
  logCmuxEvent(loadEffectiveGSDPreferences()?.preferences, requestedStepMode ? "Step-mode started." : "Auto-mode started.", "progress");
1549
1575
 
1550
1576
  // Dispatch the first unit
1551
- await autoLoop(ctx, pi, s, buildLoopDeps());
1577
+ await runAutoLoopWithUok({
1578
+ ctx,
1579
+ pi,
1580
+ s,
1581
+ deps: buildLoopDeps(),
1582
+ runLegacyLoop: autoLoop,
1583
+ });
1552
1584
  cleanupAfterLoopExit(ctx);
1553
1585
  }
1554
1586
 
@@ -4,6 +4,8 @@ import type { ExtensionAPI, ExtensionCommandContext } from "@gsd/pi-coding-agent
4
4
 
5
5
  import { registerExitCommand } from "../exit-command.js";
6
6
  import { registerWorktreeCommand } from "../worktree-command.js";
7
+ import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
8
+ import { loadEcosystemExtensions } from "../ecosystem/loader.js";
7
9
  import { registerDbTools } from "./db-tools.js";
8
10
  import { registerDynamicTools } from "./dynamic-tools.js";
9
11
  import { registerJournalTools } from "./journal-tools.js";
@@ -65,6 +67,10 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
65
67
 
66
68
  installEpipeGuard();
67
69
 
70
+ // Ecosystem handlers captured by the GSDExtensionAPI wrapper for the
71
+ // GSD-owned `before_agent_start` dispatch step (#3338).
72
+ const ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[] = [];
73
+
68
74
  pi.registerCommand("kill", {
69
75
  description: "Exit GSD immediately (no cleanup)",
70
76
  handler: async (_args: string, _ctx: ExtensionCommandContext) => {
@@ -80,7 +86,15 @@ export function registerGsdExtension(pi: ExtensionAPI): void {
80
86
  ["journal-tools", () => registerJournalTools(pi)],
81
87
  ["query-tools", () => registerQueryTools(pi)],
82
88
  ["shortcuts", () => registerShortcuts(pi)],
83
- ["hooks", () => registerHooks(pi)],
89
+ ["hooks", () => registerHooks(pi, ecosystemHandlers)],
90
+ ["ecosystem", () => {
91
+ void loadEcosystemExtensions(pi, ecosystemHandlers).catch((err) => {
92
+ logWarning(
93
+ "ecosystem",
94
+ `loader failed: ${err instanceof Error ? err.message : String(err)}`,
95
+ );
96
+ });
97
+ }],
84
98
  ];
85
99
 
86
100
  for (const [name, register] of nonCriticalRegistrations) {
@@ -3,6 +3,10 @@ import { join } from "node:path";
3
3
  import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
4
4
  import { isToolCallEventType } from "@gsd/pi-coding-agent";
5
5
 
6
+ import type { GSDEcosystemBeforeAgentStartHandler } from "../ecosystem/gsd-extension-api.js";
7
+ import { updateSnapshot } from "../ecosystem/gsd-extension-api.js";
8
+ import { getEcosystemReadyPromise } from "../ecosystem/loader.js";
9
+
6
10
  import { buildMilestoneFileName, resolveMilestonePath, resolveSliceFile, resolveSlicePath } from "../paths.js";
7
11
  import { buildBeforeAgentStartResult } from "./system-context.js";
8
12
  import { handleAgentEnd } from "./agent-end-recovery.js";
@@ -35,7 +39,10 @@ async function syncServiceTierStatus(ctx: ExtensionContext): Promise<void> {
35
39
  ctx.ui.setStatus("gsd-fast", formatServiceTierFooterStatus(getEffectiveServiceTier(), ctx.model?.id));
36
40
  }
37
41
 
38
- export function registerHooks(pi: ExtensionAPI): void {
42
+ export function registerHooks(
43
+ pi: ExtensionAPI,
44
+ ecosystemHandlers: GSDEcosystemBeforeAgentStartHandler[],
45
+ ): void {
39
46
  pi.on("session_start", async (_event, ctx) => {
40
47
  initNotificationStore(process.cwd());
41
48
  installNotifyInterceptor(ctx);
@@ -45,8 +52,12 @@ export function registerHooks(pi: ExtensionAPI): void {
45
52
  resetToolCallLoopGuard();
46
53
  resetAskUserQuestionsCache();
47
54
  await syncServiceTierStatus(ctx);
48
- const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
49
- prepareWorkflowMcpForProject(ctx, process.cwd());
55
+ // Skip MCP auto-prep when running inside an auto-worktree (see session_switch below).
56
+ const { isInAutoWorktree } = await import("../auto-worktree.js");
57
+ if (!isInAutoWorktree(process.cwd())) {
58
+ const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
59
+ prepareWorkflowMcpForProject(ctx, process.cwd());
60
+ }
50
61
 
51
62
  // Apply show_token_cost preference (#1515)
52
63
  try {
@@ -87,13 +98,63 @@ export function registerHooks(pi: ExtensionAPI): void {
87
98
  resetAskUserQuestionsCache();
88
99
  clearDiscussionFlowState();
89
100
  await syncServiceTierStatus(ctx);
90
- const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
91
- prepareWorkflowMcpForProject(ctx, process.cwd());
101
+ // Skip MCP auto-prep when running inside an auto-worktree. The worktree
102
+ // already has .mcp.json from createAutoWorktree, and re-running the writer
103
+ // post-chdir rewrites the file mid-run (non-idempotent due to cwd-relative
104
+ // CLI path resolution), dirtying the tree and breaking the milestone merge.
105
+ const { isInAutoWorktree } = await import("../auto-worktree.js");
106
+ if (!isInAutoWorktree(process.cwd())) {
107
+ const { prepareWorkflowMcpForProject } = await import("../workflow-mcp-auto-prep.js");
108
+ prepareWorkflowMcpForProject(ctx, process.cwd());
109
+ }
92
110
  loadToolApiKeys();
93
111
  });
94
112
 
95
113
  pi.on("before_agent_start", async (event, ctx: ExtensionContext) => {
96
- return buildBeforeAgentStartResult(event, ctx);
114
+ // Wait for ecosystem loader to finish (no-op after first turn).
115
+ await getEcosystemReadyPromise();
116
+
117
+ // GSD's own context injection (existing behavior — unchanged).
118
+ const gsdResult = await buildBeforeAgentStartResult(event, ctx);
119
+
120
+ // Refresh the snapshot used by ecosystem getPhase()/getActiveUnit().
121
+ // deriveState has its own ~100ms cache so this is cheap on repeat calls.
122
+ try {
123
+ const state = await deriveState(process.cwd());
124
+ updateSnapshot(state);
125
+ } catch {
126
+ updateSnapshot(null);
127
+ }
128
+
129
+ // Chain ecosystem handlers using pi's runner.ts chaining protocol:
130
+ // each handler sees the systemPrompt mutated by prior handlers.
131
+ let currentSystemPrompt = gsdResult?.systemPrompt ?? event.systemPrompt;
132
+ // `any` because pi's BeforeAgentStartEventResult.message uses an internal
133
+ // CustomMessage type that's not re-exported (see ecosystem/gsd-extension-api.ts).
134
+ let lastMessage: any = gsdResult?.message;
135
+
136
+ for (const handler of ecosystemHandlers) {
137
+ try {
138
+ const r = await handler(
139
+ { ...event, systemPrompt: currentSystemPrompt },
140
+ ctx,
141
+ );
142
+ if (r?.systemPrompt !== undefined) currentSystemPrompt = r.systemPrompt;
143
+ if (r?.message) lastMessage = r.message;
144
+ } catch (err) {
145
+ safetyLogWarning(
146
+ "ecosystem",
147
+ `before_agent_start handler failed: ${err instanceof Error ? err.message : String(err)}`,
148
+ );
149
+ }
150
+ }
151
+
152
+ // Compose result. Return undefined if nothing changed (preserves runner contract).
153
+ if (currentSystemPrompt === event.systemPrompt && !lastMessage) return undefined;
154
+ return {
155
+ systemPrompt: currentSystemPrompt !== event.systemPrompt ? currentSystemPrompt : undefined,
156
+ message: lastMessage,
157
+ };
97
158
  });
98
159
 
99
160
  pi.on("agent_end", async (event, ctx: ExtensionContext) => {
@@ -125,7 +186,10 @@ export function registerHooks(pi: ExtensionAPI): void {
125
186
  await ensureDbOpen();
126
187
  const state = await deriveState(basePath);
127
188
  if (!state.activeMilestone || !state.activeSlice || !state.activeTask) return;
128
- if (state.phase !== "executing") return;
189
+ // Write checkpoint for ALL phases, not just "executing" — discuss, research,
190
+ // and planning also carry in-memory state (user answers, gate verification)
191
+ // that would be lost on compaction (#4258).
192
+ // if (state.phase !== "executing") return;
129
193
 
130
194
  const sliceDir = resolveSlicePath(basePath, state.activeMilestone.id, state.activeSlice.id);
131
195
  if (!sliceDir) return;
@@ -261,7 +325,7 @@ export function registerHooks(pi: ExtensionAPI): void {
261
325
  // ── Safety harness: evidence collection + destructive command warnings ──
262
326
  pi.on("tool_call", async (event, ctx) => {
263
327
  if (!isAutoActive()) return;
264
- safetyRecordToolCall(event.toolName, event.input as Record<string, unknown>);
328
+ safetyRecordToolCall(event.toolCallId, event.toolName, event.input as Record<string, unknown>);
265
329
 
266
330
  // Destructive command classification (warn only, never block)
267
331
  if (isToolCallEventType("bash", event)) {
@@ -1,6 +1,6 @@
1
1
  // GSD Extension — Unified Cache Invalidation
2
2
  //
3
- // Three module-scoped caches exist across the GSD extension:
3
+ // Three module-scoped read caches exist across the GSD extension:
4
4
  // 1. State cache (state.ts) — memoized deriveState() result
5
5
  // 2. Path cache (paths.ts) — directory listing results (readdirSync)
6
6
  // 3. Parse cache (files.ts) — parsed markdown file results
@@ -8,22 +8,33 @@
8
8
  // After any file write that changes .gsd/ contents, all three must be
9
9
  // invalidated together to prevent stale reads. This module provides a
10
10
  // single function that clears all three atomically.
11
+ //
12
+ // NOTE: The DB `artifacts` table is NOT included here. Earlier versions
13
+ // called clearArtifacts() as part of this bundle (#793), intending to
14
+ // force deriveState() to re-parse from disk when files were edited
15
+ // out-of-band. But invalidateAllCaches() fires on every post-unit pass,
16
+ // so bundling a DESTRUCTIVE `DELETE FROM artifacts` with routine cache
17
+ // invalidation meant every row written by saveArtifactToDb / writeAndStore
18
+ // was wiped within seconds — leaving the milestone completed on disk but
19
+ // the `artifacts` table empty and the agent looping on "file exists but
20
+ // DB record missing" recovery calls. If a call site genuinely needs the
21
+ // artifact table cleared after an out-of-band file mutation, it should
22
+ // invoke clearArtifacts() from gsd-db.js explicitly — do not add it back
23
+ // here.
11
24
 
12
25
  import { invalidateStateCache } from './state.js';
13
26
  import { clearPathCache } from './paths.js';
14
27
  import { clearParseCache } from './files.js';
15
- import { clearArtifacts } from './gsd-db.js';
16
28
 
17
29
  /**
18
- * Invalidate all GSD runtime caches in one call.
30
+ * Invalidate all GSD runtime read caches in one call.
19
31
  *
20
32
  * Call this after file writes, milestone transitions, merge reconciliation,
21
33
  * or any operation that changes .gsd/ contents on disk. Forgetting to clear
22
- * any single cache causes stale reads (see #431, #793).
34
+ * any single cache causes stale reads (see #431).
23
35
  */
24
36
  export function invalidateAllCaches(): void {
25
37
  invalidateStateCache();
26
38
  clearPathCache();
27
39
  clearParseCache();
28
- clearArtifacts();
29
40
  }
@@ -236,5 +236,10 @@ Examples:
236
236
  await handleAddTests(trimmed.replace(/^add-tests\s*/, "").trim(), ctx, pi);
237
237
  return true;
238
238
  }
239
+ if (trimmed === "extract-learnings" || trimmed.startsWith("extract-learnings ")) {
240
+ const { handleExtractLearnings } = await import("../../commands-extract-learnings.js");
241
+ await handleExtractLearnings(trimmed.replace(/^extract-learnings\s*/, "").trim(), ctx, pi);
242
+ return true;
243
+ }
239
244
  return false;
240
245
  }