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

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 (430) 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 +164 -9
  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 +54 -8
  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-worktree.js +2 -0
  12. package/dist/resources/extensions/gsd/auto.js +28 -2
  13. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +5 -3
  14. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +10 -1
  15. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +61 -9
  16. package/dist/resources/extensions/gsd/cache.js +16 -5
  17. package/dist/resources/extensions/gsd/commands/catalog.js +6 -1
  18. package/dist/resources/extensions/gsd/commands/handlers/core.js +5 -1
  19. package/dist/resources/extensions/gsd/commands/handlers/ops.js +5 -0
  20. package/dist/resources/extensions/gsd/commands-extract-learnings.js +225 -0
  21. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +51 -4
  22. package/dist/resources/extensions/gsd/docs/preferences-reference.md +16 -1
  23. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +144 -0
  24. package/dist/resources/extensions/gsd/ecosystem/loader.js +145 -0
  25. package/dist/resources/extensions/gsd/git-service.js +49 -1
  26. package/dist/resources/extensions/gsd/graph-context.js +98 -7
  27. package/dist/resources/extensions/gsd/gsd-db.js +260 -2
  28. package/dist/resources/extensions/gsd/guided-flow.js +31 -6
  29. package/dist/resources/extensions/gsd/init-wizard.js +1 -0
  30. package/dist/resources/extensions/gsd/journal.js +27 -0
  31. package/dist/resources/extensions/gsd/metrics.js +19 -0
  32. package/dist/resources/extensions/gsd/parallel-orchestrator.js +33 -1
  33. package/dist/resources/extensions/gsd/preferences-models.js +20 -3
  34. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  35. package/dist/resources/extensions/gsd/preferences-validation.js +118 -2
  36. package/dist/resources/extensions/gsd/preferences.js +31 -0
  37. package/dist/resources/extensions/gsd/safety/evidence-collector.js +15 -30
  38. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +12 -2
  39. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
  40. package/dist/resources/extensions/gsd/tools/complete-slice.js +5 -0
  41. package/dist/resources/extensions/gsd/tools/validate-milestone.js +39 -4
  42. package/dist/resources/extensions/gsd/unit-ownership.js +1 -1
  43. package/dist/resources/extensions/gsd/uok/audit-toggle.js +7 -0
  44. package/dist/resources/extensions/gsd/uok/audit.js +40 -0
  45. package/dist/resources/extensions/gsd/uok/contracts.js +1 -0
  46. package/dist/resources/extensions/gsd/uok/execution-graph.js +179 -0
  47. package/dist/resources/extensions/gsd/uok/flags.js +29 -0
  48. package/dist/resources/extensions/gsd/uok/gate-runner.js +109 -0
  49. package/dist/resources/extensions/gsd/uok/gitops.js +53 -0
  50. package/dist/resources/extensions/gsd/uok/kernel.js +80 -0
  51. package/dist/resources/extensions/gsd/uok/loop-adapter.js +133 -0
  52. package/dist/resources/extensions/gsd/uok/model-policy.js +66 -0
  53. package/dist/resources/extensions/gsd/uok/plan-v2.js +132 -0
  54. package/dist/resources/extensions/gsd/workflow-logger.js +22 -0
  55. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  56. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  57. package/dist/web/standalone/.next/BUILD_ID +1 -1
  58. package/dist/web/standalone/.next/app-path-routes-manifest.json +8 -8
  59. package/dist/web/standalone/.next/build-manifest.json +2 -2
  60. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  61. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  62. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  63. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  64. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  65. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  70. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  78. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  79. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +1 -1
  80. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  81. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  82. package/dist/web/standalone/.next/server/app/index.html +1 -1
  83. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  87. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app-paths-manifest.json +8 -8
  90. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  91. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  92. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  93. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  94. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  95. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  96. package/package.json +1 -1
  97. package/packages/mcp-server/dist/readers/graph.d.ts +1 -1
  98. package/packages/mcp-server/dist/readers/graph.d.ts.map +1 -1
  99. package/packages/mcp-server/dist/readers/graph.js +107 -0
  100. package/packages/mcp-server/dist/readers/graph.js.map +1 -1
  101. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  102. package/packages/mcp-server/dist/workflow-tools.js +88 -6
  103. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  104. package/packages/mcp-server/src/readers/graph.test.ts +178 -0
  105. package/packages/mcp-server/src/readers/graph.ts +148 -1
  106. package/packages/mcp-server/src/workflow-tools.ts +95 -10
  107. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  108. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  109. package/packages/pi-ai/dist/index.d.ts +1 -9
  110. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  111. package/packages/pi-ai/dist/index.js +1 -9
  112. package/packages/pi-ai/dist/index.js.map +1 -1
  113. package/packages/pi-ai/dist/models/capability-patches.d.ts +19 -0
  114. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -0
  115. package/packages/pi-ai/dist/models/capability-patches.js +36 -0
  116. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -0
  117. package/packages/pi-ai/dist/{models.custom.d.ts → models/custom.d.ts} +1 -1
  118. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -0
  119. package/packages/pi-ai/dist/{models.custom.js → models/custom.js} +4 -4
  120. package/packages/pi-ai/dist/models/custom.js.map +1 -0
  121. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +1482 -0
  122. package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -0
  123. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +1484 -0
  124. package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -0
  125. package/packages/pi-ai/dist/models/generated/anthropic.d.ts +377 -0
  126. package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -0
  127. package/packages/pi-ai/dist/models/generated/anthropic.js +379 -0
  128. package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -0
  129. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts +700 -0
  130. package/packages/pi-ai/dist/models/generated/azure-openai-responses.d.ts.map +1 -0
  131. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js +702 -0
  132. package/packages/pi-ai/dist/models/generated/azure-openai-responses.js.map +1 -0
  133. package/packages/pi-ai/dist/models/generated/cerebras.d.ts +71 -0
  134. package/packages/pi-ai/dist/models/generated/cerebras.d.ts.map +1 -0
  135. package/packages/pi-ai/dist/models/generated/cerebras.js +73 -0
  136. package/packages/pi-ai/dist/models/generated/cerebras.js.map +1 -0
  137. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts +590 -0
  138. package/packages/pi-ai/dist/models/generated/github-copilot.d.ts.map +1 -0
  139. package/packages/pi-ai/dist/models/generated/github-copilot.js +444 -0
  140. package/packages/pi-ai/dist/models/generated/github-copilot.js.map +1 -0
  141. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +156 -0
  142. package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -0
  143. package/packages/pi-ai/dist/models/generated/google-antigravity.js +158 -0
  144. package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -0
  145. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts +105 -0
  146. package/packages/pi-ai/dist/models/generated/google-gemini-cli.d.ts.map +1 -0
  147. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js +107 -0
  148. package/packages/pi-ai/dist/models/generated/google-gemini-cli.js.map +1 -0
  149. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts +207 -0
  150. package/packages/pi-ai/dist/models/generated/google-vertex.d.ts.map +1 -0
  151. package/packages/pi-ai/dist/models/generated/google-vertex.js +209 -0
  152. package/packages/pi-ai/dist/models/generated/google-vertex.js.map +1 -0
  153. package/packages/pi-ai/dist/models/generated/google.d.ts +462 -0
  154. package/packages/pi-ai/dist/models/generated/google.d.ts.map +1 -0
  155. package/packages/pi-ai/dist/models/generated/google.js +464 -0
  156. package/packages/pi-ai/dist/models/generated/google.js.map +1 -0
  157. package/packages/pi-ai/dist/models/generated/groq.d.ts +309 -0
  158. package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -0
  159. package/packages/pi-ai/dist/models/generated/groq.js +311 -0
  160. package/packages/pi-ai/dist/models/generated/groq.js.map +1 -0
  161. package/packages/pi-ai/dist/models/generated/huggingface.d.ts +383 -0
  162. package/packages/pi-ai/dist/models/generated/huggingface.d.ts.map +1 -0
  163. package/packages/pi-ai/dist/models/generated/huggingface.js +347 -0
  164. package/packages/pi-ai/dist/models/generated/huggingface.js.map +1 -0
  165. package/packages/pi-ai/dist/{models.generated.d.ts → models/generated/index.d.ts} +1 -1
  166. package/packages/pi-ai/dist/{models.generated.d.ts.map → models/generated/index.d.ts.map} +1 -1
  167. package/packages/pi-ai/dist/models/generated/index.js +51 -0
  168. package/packages/pi-ai/dist/models/generated/index.js.map +1 -0
  169. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts +37 -0
  170. package/packages/pi-ai/dist/models/generated/kimi-coding.d.ts.map +1 -0
  171. package/packages/pi-ai/dist/models/generated/kimi-coding.js +39 -0
  172. package/packages/pi-ai/dist/models/generated/kimi-coding.js.map +1 -0
  173. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts +105 -0
  174. package/packages/pi-ai/dist/models/generated/minimax-cn.d.ts.map +1 -0
  175. package/packages/pi-ai/dist/models/generated/minimax-cn.js +107 -0
  176. package/packages/pi-ai/dist/models/generated/minimax-cn.js.map +1 -0
  177. package/packages/pi-ai/dist/models/generated/minimax.d.ts +105 -0
  178. package/packages/pi-ai/dist/models/generated/minimax.d.ts.map +1 -0
  179. package/packages/pi-ai/dist/models/generated/minimax.js +107 -0
  180. package/packages/pi-ai/dist/models/generated/minimax.js.map +1 -0
  181. package/packages/pi-ai/dist/models/generated/mistral.d.ts +445 -0
  182. package/packages/pi-ai/dist/models/generated/mistral.d.ts.map +1 -0
  183. package/packages/pi-ai/dist/models/generated/mistral.js +447 -0
  184. package/packages/pi-ai/dist/models/generated/mistral.js.map +1 -0
  185. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +139 -0
  186. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -0
  187. package/packages/pi-ai/dist/models/generated/openai-codex.js +141 -0
  188. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -0
  189. package/packages/pi-ai/dist/models/generated/openai.d.ts +700 -0
  190. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -0
  191. package/packages/pi-ai/dist/models/generated/openai.js +702 -0
  192. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -0
  193. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts +122 -0
  194. package/packages/pi-ai/dist/models/generated/opencode-go.d.ts.map +1 -0
  195. package/packages/pi-ai/dist/models/generated/opencode-go.js +124 -0
  196. package/packages/pi-ai/dist/models/generated/opencode-go.js.map +1 -0
  197. package/packages/pi-ai/dist/models/generated/opencode.d.ts +530 -0
  198. package/packages/pi-ai/dist/models/generated/opencode.d.ts.map +1 -0
  199. package/packages/pi-ai/dist/models/generated/opencode.js +532 -0
  200. package/packages/pi-ai/dist/models/generated/opencode.js.map +1 -0
  201. package/packages/pi-ai/dist/models/generated/openrouter.d.ts +4270 -0
  202. package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -0
  203. package/packages/pi-ai/dist/models/generated/openrouter.js +4272 -0
  204. package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -0
  205. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts +2604 -0
  206. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.d.ts.map +1 -0
  207. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js +2606 -0
  208. package/packages/pi-ai/dist/models/generated/vercel-ai-gateway.js.map +1 -0
  209. package/packages/pi-ai/dist/models/generated/xai.d.ts +411 -0
  210. package/packages/pi-ai/dist/models/generated/xai.d.ts.map +1 -0
  211. package/packages/pi-ai/dist/models/generated/xai.js +413 -0
  212. package/packages/pi-ai/dist/models/generated/xai.js.map +1 -0
  213. package/packages/pi-ai/dist/models/generated/zai.d.ts +276 -0
  214. package/packages/pi-ai/dist/models/generated/zai.d.ts.map +1 -0
  215. package/packages/pi-ai/dist/models/generated/zai.js +239 -0
  216. package/packages/pi-ai/dist/models/generated/zai.js.map +1 -0
  217. package/packages/pi-ai/dist/models/index.d.ts +27 -0
  218. package/packages/pi-ai/dist/models/index.d.ts.map +1 -0
  219. package/packages/pi-ai/dist/models/index.js +80 -0
  220. package/packages/pi-ai/dist/models/index.js.map +1 -0
  221. package/packages/pi-ai/dist/models.d.ts +1 -36
  222. package/packages/pi-ai/dist/models.d.ts.map +1 -1
  223. package/packages/pi-ai/dist/models.generated.test.js +1 -2
  224. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  225. package/packages/pi-ai/dist/models.js +3 -112
  226. package/packages/pi-ai/dist/models.js.map +1 -1
  227. package/packages/pi-ai/dist/models.test.js +6 -5
  228. package/packages/pi-ai/dist/models.test.js.map +1 -1
  229. package/packages/pi-ai/scripts/generate-models.ts +74 -40
  230. package/packages/pi-ai/src/index.ts +1 -9
  231. package/packages/pi-ai/src/models/capability-patches.ts +40 -0
  232. package/packages/pi-ai/src/{models.custom.ts → models/custom.ts} +4 -4
  233. package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +1486 -0
  234. package/packages/pi-ai/src/models/generated/anthropic.ts +381 -0
  235. package/packages/pi-ai/src/models/generated/azure-openai-responses.ts +704 -0
  236. package/packages/pi-ai/src/models/generated/cerebras.ts +75 -0
  237. package/packages/pi-ai/src/models/generated/github-copilot.ts +446 -0
  238. package/packages/pi-ai/src/models/generated/google-antigravity.ts +160 -0
  239. package/packages/pi-ai/src/models/generated/google-gemini-cli.ts +109 -0
  240. package/packages/pi-ai/src/models/generated/google-vertex.ts +211 -0
  241. package/packages/pi-ai/src/models/generated/google.ts +466 -0
  242. package/packages/pi-ai/src/models/generated/groq.ts +313 -0
  243. package/packages/pi-ai/src/models/generated/huggingface.ts +349 -0
  244. package/packages/pi-ai/src/models/generated/index.ts +52 -0
  245. package/packages/pi-ai/src/models/generated/kimi-coding.ts +41 -0
  246. package/packages/pi-ai/src/models/generated/minimax-cn.ts +109 -0
  247. package/packages/pi-ai/src/models/generated/minimax.ts +109 -0
  248. package/packages/pi-ai/src/models/generated/mistral.ts +449 -0
  249. package/packages/pi-ai/src/models/generated/openai-codex.ts +143 -0
  250. package/packages/pi-ai/src/models/generated/openai.ts +704 -0
  251. package/packages/pi-ai/src/models/generated/opencode-go.ts +126 -0
  252. package/packages/pi-ai/src/models/generated/opencode.ts +534 -0
  253. package/packages/pi-ai/src/models/generated/openrouter.ts +4274 -0
  254. package/packages/pi-ai/src/models/generated/vercel-ai-gateway.ts +2608 -0
  255. package/packages/pi-ai/src/models/generated/xai.ts +415 -0
  256. package/packages/pi-ai/src/models/generated/zai.ts +241 -0
  257. package/packages/pi-ai/src/models/index.ts +106 -0
  258. package/packages/pi-ai/src/models.generated.test.ts +1 -2
  259. package/packages/pi-ai/src/models.test.ts +6 -5
  260. package/packages/pi-ai/src/models.ts +3 -153
  261. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  262. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  263. package/packages/pi-coding-agent/dist/core/agent-session.js +8 -2
  264. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  265. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +472 -0
  266. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  267. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts +2 -0
  268. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.d.ts.map +1 -0
  269. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js +52 -0
  270. package/packages/pi-coding-agent/dist/core/model-registry-env-fallback.test.js.map +1 -0
  271. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  272. package/packages/pi-coding-agent/dist/core/model-registry.js +2 -2
  273. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  274. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.d.ts +2 -0
  275. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.d.ts.map +1 -0
  276. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js +61 -0
  277. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.js.map +1 -0
  278. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +11 -0
  279. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  280. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +1 -0
  281. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  282. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +23 -9
  283. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  284. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +11 -0
  285. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -0
  286. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +53 -0
  287. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -0
  288. package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.d.ts +8 -5
  289. package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -1
  290. package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.js +27 -13
  291. package/packages/pi-coding-agent/dist/modes/interactive/components/compaction-summary-message.js.map +1 -1
  292. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +8 -0
  293. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  294. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +68 -8
  295. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  296. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
  297. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +22 -22
  298. package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
  299. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  300. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +232 -18
  301. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  302. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts +2 -0
  303. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.d.ts.map +1 -0
  304. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +38 -0
  305. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -0
  306. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +14 -0
  307. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  308. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +70 -6
  309. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  310. package/packages/pi-coding-agent/src/core/agent-session.ts +12 -6
  311. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +612 -0
  312. package/packages/pi-coding-agent/src/core/model-registry-env-fallback.test.ts +59 -0
  313. package/packages/pi-coding-agent/src/core/model-registry.ts +2 -1
  314. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/chat-frame-compaction-tone.test.ts +92 -0
  315. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +19 -0
  316. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +25 -10
  317. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +75 -0
  318. package/packages/pi-coding-agent/src/modes/interactive/components/compaction-summary-message.ts +36 -15
  319. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +83 -7
  320. package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +23 -26
  321. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +298 -41
  322. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +44 -0
  323. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +92 -6
  324. package/packages/pi-coding-agent/src/types/ambient-modules.d.ts +69 -0
  325. package/packages/pi-coding-agent/tsconfig.json +2 -2
  326. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  327. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  328. package/packages/pi-tui/dist/tui.js +9 -2
  329. package/packages/pi-tui/dist/tui.js.map +1 -1
  330. package/packages/pi-tui/src/tui.ts +9 -1
  331. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  332. package/src/resources/extensions/gsd/activity-log.ts +21 -0
  333. package/src/resources/extensions/gsd/auto/loop-deps.ts +4 -0
  334. package/src/resources/extensions/gsd/auto/loop.ts +159 -10
  335. package/src/resources/extensions/gsd/auto/phases.ts +193 -9
  336. package/src/resources/extensions/gsd/auto/session.ts +10 -0
  337. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -6
  338. package/src/resources/extensions/gsd/auto-model-selection.ts +69 -8
  339. package/src/resources/extensions/gsd/auto-post-unit.ts +226 -9
  340. package/src/resources/extensions/gsd/auto-recovery.ts +29 -9
  341. package/src/resources/extensions/gsd/auto-unit-closeout.ts +25 -1
  342. package/src/resources/extensions/gsd/auto-verification.ts +129 -2
  343. package/src/resources/extensions/gsd/auto-worktree.ts +1 -0
  344. package/src/resources/extensions/gsd/auto.ts +34 -2
  345. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +5 -3
  346. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +15 -1
  347. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +72 -8
  348. package/src/resources/extensions/gsd/cache.ts +16 -5
  349. package/src/resources/extensions/gsd/commands/catalog.ts +6 -1
  350. package/src/resources/extensions/gsd/commands/handlers/core.ts +5 -1
  351. package/src/resources/extensions/gsd/commands/handlers/ops.ts +5 -0
  352. package/src/resources/extensions/gsd/commands-extract-learnings.ts +304 -0
  353. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +58 -4
  354. package/src/resources/extensions/gsd/docs/preferences-reference.md +16 -1
  355. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +228 -0
  356. package/src/resources/extensions/gsd/ecosystem/loader.ts +201 -0
  357. package/src/resources/extensions/gsd/git-service.ts +68 -0
  358. package/src/resources/extensions/gsd/graph-context.ts +139 -12
  359. package/src/resources/extensions/gsd/gsd-db.ts +321 -3
  360. package/src/resources/extensions/gsd/guided-flow.ts +36 -2
  361. package/src/resources/extensions/gsd/init-wizard.ts +3 -2
  362. package/src/resources/extensions/gsd/journal.ts +30 -0
  363. package/src/resources/extensions/gsd/metrics.ts +26 -0
  364. package/src/resources/extensions/gsd/parallel-orchestrator.ts +40 -1
  365. package/src/resources/extensions/gsd/preferences-models.ts +20 -3
  366. package/src/resources/extensions/gsd/preferences-types.ts +38 -0
  367. package/src/resources/extensions/gsd/preferences-validation.ts +117 -2
  368. package/src/resources/extensions/gsd/preferences.ts +34 -0
  369. package/src/resources/extensions/gsd/safety/evidence-collector.ts +15 -31
  370. package/src/resources/extensions/gsd/session-lock.ts +14 -2
  371. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +20 -1
  372. package/src/resources/extensions/gsd/templates/PREFERENCES.md +19 -0
  373. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +177 -0
  374. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +7 -3
  375. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +20 -0
  376. package/src/resources/extensions/gsd/tests/auto-project-root-env.test.ts +7 -3
  377. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +272 -0
  378. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +117 -0
  379. package/src/resources/extensions/gsd/tests/cold-resume-db-reopen.test.ts +6 -2
  380. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +340 -0
  381. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  382. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  383. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +10 -7
  384. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
  385. package/src/resources/extensions/gsd/tests/health-widget.test.ts +1 -1
  386. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +3 -3
  387. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -2
  388. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -3
  389. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  390. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +2 -1
  391. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +40 -1
  392. package/src/resources/extensions/gsd/tests/preferences.test.ts +145 -0
  393. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +57 -2
  394. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +1 -1
  395. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -5
  396. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +101 -0
  397. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +85 -0
  398. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +69 -0
  399. package/src/resources/extensions/gsd/tests/uok-flags.test.ts +39 -0
  400. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +70 -0
  401. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +85 -0
  402. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +35 -0
  403. package/src/resources/extensions/gsd/tests/uok-model-policy.test.ts +89 -0
  404. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +167 -0
  405. package/src/resources/extensions/gsd/tests/uok-preferences.test.ts +42 -0
  406. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +39 -0
  407. package/src/resources/extensions/gsd/tools/complete-slice.ts +9 -2
  408. package/src/resources/extensions/gsd/tools/validate-milestone.ts +48 -3
  409. package/src/resources/extensions/gsd/types.ts +14 -1
  410. package/src/resources/extensions/gsd/unit-ownership.ts +2 -2
  411. package/src/resources/extensions/gsd/uok/audit-toggle.ts +9 -0
  412. package/src/resources/extensions/gsd/uok/audit.ts +51 -0
  413. package/src/resources/extensions/gsd/uok/contracts.ts +135 -0
  414. package/src/resources/extensions/gsd/uok/execution-graph.ts +241 -0
  415. package/src/resources/extensions/gsd/uok/flags.ts +45 -0
  416. package/src/resources/extensions/gsd/uok/gate-runner.ts +146 -0
  417. package/src/resources/extensions/gsd/uok/gitops.ts +75 -0
  418. package/src/resources/extensions/gsd/uok/kernel.ts +105 -0
  419. package/src/resources/extensions/gsd/uok/loop-adapter.ts +162 -0
  420. package/src/resources/extensions/gsd/uok/model-policy.ts +112 -0
  421. package/src/resources/extensions/gsd/uok/plan-v2.ts +156 -0
  422. package/src/resources/extensions/gsd/workflow-logger.ts +27 -1
  423. package/src/resources/extensions/ttsr/ttsr-manager.ts +10 -5
  424. package/packages/pi-ai/dist/models.custom.d.ts.map +0 -1
  425. package/packages/pi-ai/dist/models.custom.js.map +0 -1
  426. package/packages/pi-ai/dist/models.generated.js +0 -14343
  427. package/packages/pi-ai/dist/models.generated.js.map +0 -1
  428. package/packages/pi-ai/src/models.generated.ts +0 -14345
  429. /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → k1VZ0h-SB6uWSZgPhgNwi}/_buildManifest.js +0 -0
  430. /package/dist/web/standalone/.next/static/{XnHY5eXUsTCFmNodWHetD → k1VZ0h-SB6uWSZgPhgNwi}/_ssgManifest.js +0 -0
@@ -102,6 +102,8 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
102
102
 
103
103
  - `custom_instructions`: extra durable instructions related to skill use. For operational project knowledge (recurring rules, gotchas, patterns), use `.gsd/KNOWLEDGE.md` instead — it's injected into every agent prompt automatically and agents can append to it during execution.
104
104
 
105
+ - `language`: preferred response language for all GSD interactions. Accepts any language name or code — `"Chinese"`, `"zh"`, `"German"`, `"de"`, `"日本語"`, etc. When set, GSD injects "Always respond in \<language\>" into every agent's system prompt, including after `/clear`. Quickest way to set it: `/gsd language <name>`. To clear: `/gsd language off`.
106
+
105
107
  - `models`: per-stage model selection (applies to both auto-mode and guided-flow dispatches). Keys: `research`, `planning`, `discuss`, `execution`, `execution_simple`, `completion`, `validation`, `subagent`. Values can be:
106
108
  - Simple string: `"claude-sonnet-4-6"` — single model, no fallbacks
107
109
  - Provider-qualified string: `"bedrock/claude-sonnet-4-6"` — targets a specific provider when the same model ID exists across multiple providers
@@ -153,7 +155,7 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
153
155
 
154
156
  - `context_pause_threshold`: number (0-100) — context window usage percentage at which auto-mode should pause to suggest checkpointing. Set to `0` to disable. Default: `0` (disabled).
155
157
 
156
- - `token_profile`: `"budget"`, `"balanced"`, or `"quality"` — coordinates model selection, phase skipping, and context compression. `budget` skips research/reassessment and uses cheaper models; `balanced` (default) skips research/reassessment to reduce token burn; `quality` prefers higher-quality models. See token-optimization docs.
158
+ - `token_profile`: `"budget"`, `"balanced"`, `"quality"`, or `"burn-max"` — coordinates model selection, phase skipping, and context compression. `budget` skips research/reassessment and uses cheaper models; `balanced` (default) skips research/reassessment to reduce token burn; `quality` prefers higher-quality models; `burn-max` keeps full-context defaults, disables downgrade routing, and keeps phase skips off.
157
159
 
158
160
  - `phases`: fine-grained control over which phases run. Usually set by `token_profile`, but can be overridden. Keys:
159
161
  - `skip_research`: boolean — skip milestone-level research. Default: `false`.
@@ -191,6 +193,19 @@ Setting `prefer_skills: []` does **not** disable skill discovery — it just mea
191
193
  - `hooks`: boolean — enable routing hooks. Default: `true`.
192
194
  - `capability_routing`: boolean — enable capability-profile scoring for model selection within a tier. Requires `enabled: true`. Default: `false`.
193
195
 
196
+ - `uok`: Unified Orchestration Kernel controls. Keys:
197
+ - `enabled`: boolean — enable kernel wrappers and contract observers. Default: `true`.
198
+ - `legacy_fallback.enabled`: boolean — emergency release fallback that forces legacy orchestration behavior even when `uok.enabled` is `true`. Default: `false`.
199
+ - Runtime override: set `GSD_UOK_FORCE_LEGACY=1` (or `GSD_UOK_LEGACY_FALLBACK=1`) to force legacy behavior for the current process.
200
+ - `gates.enabled`: boolean — route checks through the unified gate runner and persist `gate_runs`.
201
+ - `model_policy.enabled`: boolean — enforce policy filtering before model capability scoring.
202
+ - `execution_graph.enabled`: boolean — enable DAG scheduler facade/adapters for execution.
203
+ - `gitops.enabled`: boolean — persist turn-level git transaction records.
204
+ - `gitops.turn_action`: `"commit"` | `"snapshot"` | `"status-only"` — turn transaction mode.
205
+ - `gitops.turn_push`: boolean — whether turn transactions should include push intent metadata.
206
+ - `audit_unified.enabled`: boolean — dual-write unified audit envelope events.
207
+ - `plan_v2.enabled`: boolean — enable bounded clarify/research/draft/compile planning flow.
208
+
194
209
  - `context_management`: configures context hygiene for auto-mode sessions. Keys:
195
210
  - `observation_masking`: boolean — mask old tool results to reduce context bloat. Default: `true`.
196
211
  - `observation_mask_turns`: number — keep this many recent turns verbatim (1-50). Default: `8`.
@@ -0,0 +1,228 @@
1
+ // GSD2 — Ecosystem Extension API wrapper
2
+ // Wraps pi's ExtensionAPI to expose typed GSD context (phase + active unit)
3
+ // to extensions loaded from `./.gsd/extensions/`. The wrapper intercepts only
4
+ // `on("before_agent_start", ...)` so GSD can dispatch ecosystem handlers AFTER
5
+ // refreshing state — fixing the load-order race where third-party
6
+ // `.pi/extensions/` handlers see a stale module-level snapshot (#3338).
7
+ //
8
+ // SINGLE-SESSION INVARIANT: the module-level `_snapshot` is per-process.
9
+ // Worktree or project switches do NOT reload extensions, matching pi's
10
+ // `.pi/extensions/` behavior. Only re-launching the CLI rebinds the snapshot.
11
+
12
+ import type {
13
+ BeforeAgentStartEvent,
14
+ ExtensionAPI,
15
+ ExtensionHandler,
16
+ } from "@gsd/pi-coding-agent";
17
+
18
+ // Structural mirror of pi's internal BeforeAgentStartEventResult. The internal
19
+ // type is not re-exported from the package root, and constraint #3 forbids
20
+ // changes to packages/pi-coding-agent/, so we mirror the public shape here.
21
+ // `any` on inner fields keeps assignability bidirectional with pi's stricter
22
+ // `Pick<CustomMessage, ...>` shape (CustomMessage is also not re-exported).
23
+ // Source of truth: packages/pi-coding-agent/src/core/extensions/types.ts
24
+ export interface BeforeAgentStartEventResult {
25
+ message?: {
26
+ customType: string;
27
+ content?: any;
28
+ display?: any;
29
+ details?: any;
30
+ };
31
+ systemPrompt?: string;
32
+ }
33
+
34
+ import type { GSDActiveUnit, GSDState, Phase } from "../types.js";
35
+ import { isGSDActive, getCurrentPhase } from "../../shared/gsd-phase-state.js";
36
+ import { logWarning } from "../workflow-logger.js";
37
+
38
+ // ─── Public Interface ───────────────────────────────────────────────────
39
+
40
+ export interface GSDExtensionAPI extends ExtensionAPI {
41
+ /** Current GSD workflow phase, or null if no project state. */
42
+ getPhase(): Phase | null;
43
+ /** Currently active milestone/slice/task triple, or null if none. */
44
+ getActiveUnit(): GSDActiveUnit | null;
45
+ }
46
+
47
+ export type GSDEcosystemBeforeAgentStartHandler = ExtensionHandler<
48
+ BeforeAgentStartEvent,
49
+ BeforeAgentStartEventResult
50
+ >;
51
+
52
+ // ─── Auto-loop phase mapping ────────────────────────────────────────────
53
+
54
+ const AUTO_LOOP_PHASE_MAP: Record<string, Phase> = {
55
+ "plan-milestone": "planning",
56
+ "plan-slice": "planning",
57
+ "research": "researching",
58
+ "discuss": "discussing",
59
+ "execute-task": "executing",
60
+ "verify": "verifying",
61
+ "summarize-task": "summarizing",
62
+ "summarize-slice": "summarizing",
63
+ "advance": "advancing",
64
+ "validate-milestone": "validating-milestone",
65
+ "complete-milestone": "completing-milestone",
66
+ "replan-slice": "replanning-slice",
67
+ };
68
+
69
+ /** Exposed for unit tests. Returns null for unknown keys (does NOT default). */
70
+ export function mapAutoLoopPhase(raw: string): Phase | null {
71
+ return AUTO_LOOP_PHASE_MAP[raw] ?? null;
72
+ }
73
+
74
+ function resolvePhase(state: GSDState | null): Phase | null {
75
+ if (!state) return null;
76
+ if (isGSDActive()) {
77
+ const raw = getCurrentPhase();
78
+ if (raw != null) {
79
+ const mapped = AUTO_LOOP_PHASE_MAP[raw];
80
+ if (mapped) return mapped;
81
+ logWarning("ecosystem", `unknown auto-loop phase: ${raw}`);
82
+ // FALL THROUGH to state.phase rather than defaulting to "executing".
83
+ }
84
+ }
85
+ return state.phase;
86
+ }
87
+
88
+ function resolveActiveUnit(state: GSDState | null): GSDActiveUnit | null {
89
+ if (!state) return null;
90
+ const m = state.activeMilestone;
91
+ const s = state.activeSlice;
92
+ const t = state.activeTask;
93
+ if (!m || !s || !t) return null;
94
+ return {
95
+ milestoneId: m.id,
96
+ milestoneTitle: m.title,
97
+ sliceId: s.id,
98
+ sliceTitle: s.title,
99
+ taskId: t.id,
100
+ taskTitle: t.title,
101
+ };
102
+ }
103
+
104
+ // ─── Module-level snapshot ──────────────────────────────────────────────
105
+
106
+ interface Snapshot {
107
+ phase: Phase | null;
108
+ activeUnit: GSDActiveUnit | null;
109
+ }
110
+
111
+ let _snapshot: Snapshot = { phase: null, activeUnit: null };
112
+
113
+ /** Refresh the snapshot from a freshly derived GSDState (or null on failure). */
114
+ export function updateSnapshot(state: GSDState | null): void {
115
+ _snapshot = {
116
+ phase: resolvePhase(state),
117
+ activeUnit: resolveActiveUnit(state),
118
+ };
119
+ }
120
+
121
+ export function getSnapshotPhase(): Phase | null {
122
+ return _snapshot.phase;
123
+ }
124
+
125
+ export function getSnapshotActiveUnit(): GSDActiveUnit | null {
126
+ return _snapshot.activeUnit;
127
+ }
128
+
129
+ /** Test-only: reset the snapshot to its initial empty state. */
130
+ export function _resetSnapshot(): void {
131
+ _snapshot = { phase: null, activeUnit: null };
132
+ }
133
+
134
+ // ─── Wrapper factory ────────────────────────────────────────────────────
135
+
136
+ /**
137
+ * Build a GSDExtensionAPI by manually delegating every ExtensionAPI method
138
+ * to the underlying pi instance, except `on("before_agent_start", ...)`
139
+ * which is captured into `sharedHandlers` for GSD-owned dispatch.
140
+ *
141
+ * Uses `satisfies GSDExtensionAPI` (NOT `as`) so TypeScript catches drift
142
+ * when pi adds new ExtensionAPI methods.
143
+ */
144
+ export function createGSDExtensionAPI(
145
+ pi: ExtensionAPI,
146
+ sharedHandlers: GSDEcosystemBeforeAgentStartHandler[],
147
+ ): GSDExtensionAPI {
148
+ const wrapper = {
149
+ // ── Event subscription (single intercept point) ────────────────────
150
+ on(event: any, handler: any): void {
151
+ if (event === "before_agent_start") {
152
+ sharedHandlers.push(handler as GSDEcosystemBeforeAgentStartHandler);
153
+ return;
154
+ }
155
+ (pi.on as (e: any, h: any) => void)(event, handler);
156
+ },
157
+
158
+ // ── Event emission ─────────────────────────────────────────────────
159
+ emitBeforeModelSelect: (...args: Parameters<ExtensionAPI["emitBeforeModelSelect"]>) =>
160
+ pi.emitBeforeModelSelect(...args),
161
+ emitAdjustToolSet: (...args: Parameters<ExtensionAPI["emitAdjustToolSet"]>) =>
162
+ pi.emitAdjustToolSet(...args),
163
+
164
+ // ── Tool / command / shortcut / flag registration ──────────────────
165
+ registerTool: ((tool: any) => pi.registerTool(tool)) as ExtensionAPI["registerTool"],
166
+ registerCommand: (...args: Parameters<ExtensionAPI["registerCommand"]>) =>
167
+ pi.registerCommand(...args),
168
+ registerBeforeInstall: (...args: Parameters<ExtensionAPI["registerBeforeInstall"]>) =>
169
+ pi.registerBeforeInstall(...args),
170
+ registerAfterInstall: (...args: Parameters<ExtensionAPI["registerAfterInstall"]>) =>
171
+ pi.registerAfterInstall(...args),
172
+ registerBeforeRemove: (...args: Parameters<ExtensionAPI["registerBeforeRemove"]>) =>
173
+ pi.registerBeforeRemove(...args),
174
+ registerAfterRemove: (...args: Parameters<ExtensionAPI["registerAfterRemove"]>) =>
175
+ pi.registerAfterRemove(...args),
176
+ registerShortcut: (...args: Parameters<ExtensionAPI["registerShortcut"]>) =>
177
+ pi.registerShortcut(...args),
178
+ registerFlag: (...args: Parameters<ExtensionAPI["registerFlag"]>) =>
179
+ pi.registerFlag(...args),
180
+ getFlag: (...args: Parameters<ExtensionAPI["getFlag"]>) => pi.getFlag(...args),
181
+
182
+ // ── Message rendering ──────────────────────────────────────────────
183
+ registerMessageRenderer: ((customType: string, renderer: any) =>
184
+ pi.registerMessageRenderer(customType, renderer)) as ExtensionAPI["registerMessageRenderer"],
185
+
186
+ // ── Actions ────────────────────────────────────────────────────────
187
+ sendMessage: ((message: any, options?: any) =>
188
+ pi.sendMessage(message, options)) as ExtensionAPI["sendMessage"],
189
+ sendUserMessage: (...args: Parameters<ExtensionAPI["sendUserMessage"]>) =>
190
+ pi.sendUserMessage(...args),
191
+ retryLastTurn: () => pi.retryLastTurn(),
192
+ appendEntry: ((customType: string, data?: any) =>
193
+ pi.appendEntry(customType, data)) as ExtensionAPI["appendEntry"],
194
+
195
+ // ── Session metadata ───────────────────────────────────────────────
196
+ setSessionName: (...args: Parameters<ExtensionAPI["setSessionName"]>) =>
197
+ pi.setSessionName(...args),
198
+ getSessionName: () => pi.getSessionName(),
199
+ setLabel: (...args: Parameters<ExtensionAPI["setLabel"]>) => pi.setLabel(...args),
200
+ exec: (...args: Parameters<ExtensionAPI["exec"]>) => pi.exec(...args),
201
+ getActiveTools: () => pi.getActiveTools(),
202
+ getAllTools: () => pi.getAllTools(),
203
+ setActiveTools: (...args: Parameters<ExtensionAPI["setActiveTools"]>) =>
204
+ pi.setActiveTools(...args),
205
+ getCommands: () => pi.getCommands(),
206
+
207
+ // ── Model & thinking ───────────────────────────────────────────────
208
+ setModel: (...args: Parameters<ExtensionAPI["setModel"]>) => pi.setModel(...args),
209
+ getThinkingLevel: () => pi.getThinkingLevel(),
210
+ setThinkingLevel: (...args: Parameters<ExtensionAPI["setThinkingLevel"]>) =>
211
+ pi.setThinkingLevel(...args),
212
+
213
+ // ── Provider registration ──────────────────────────────────────────
214
+ registerProvider: (...args: Parameters<ExtensionAPI["registerProvider"]>) =>
215
+ pi.registerProvider(...args),
216
+ unregisterProvider: (...args: Parameters<ExtensionAPI["unregisterProvider"]>) =>
217
+ pi.unregisterProvider(...args),
218
+
219
+ // ── Shared event bus (passthrough property) ────────────────────────
220
+ events: pi.events,
221
+
222
+ // ── GSD-specific additions ─────────────────────────────────────────
223
+ getPhase: (): Phase | null => _snapshot.phase,
224
+ getActiveUnit: (): GSDActiveUnit | null => _snapshot.activeUnit,
225
+ } satisfies GSDExtensionAPI;
226
+
227
+ return wrapper;
228
+ }
@@ -0,0 +1,201 @@
1
+ // GSD2 — Ecosystem extension loader for ./.gsd/extensions/
2
+ // Discovers and registers single-file extensions that consume GSDExtensionAPI.
3
+ // Trust-gated (mirrors pi's `.pi/extensions/` model) and isolated from pi's
4
+ // own loader chain — handlers run in GSD's own dispatch step, not pi's.
5
+
6
+ import * as fs from "node:fs";
7
+ import * as path from "node:path";
8
+ import { pathToFileURL } from "node:url";
9
+
10
+ import type { ExtensionAPI } from "@gsd/pi-coding-agent";
11
+ import { getAgentDir } from "@gsd/pi-coding-agent";
12
+
13
+ import { logWarning } from "../workflow-logger.js";
14
+ import {
15
+ createGSDExtensionAPI,
16
+ type GSDEcosystemBeforeAgentStartHandler,
17
+ type GSDExtensionAPI,
18
+ } from "./gsd-extension-api.js";
19
+
20
+ // ─── Trust check (inlined; pi does not export isProjectTrusted from its
21
+ // package root, and constraint forbids modifying packages/pi-coding-agent/) ─
22
+
23
+ const TRUSTED_PROJECTS_FILE = "trusted-projects.json";
24
+
25
+ function isProjectTrusted(projectPath: string, agentDir: string): boolean {
26
+ const canonical = path.resolve(projectPath);
27
+ const trustedPath = path.join(agentDir, TRUSTED_PROJECTS_FILE);
28
+ try {
29
+ const content = fs.readFileSync(trustedPath, "utf-8");
30
+ const parsed = JSON.parse(content);
31
+ if (Array.isArray(parsed)) {
32
+ return parsed.includes(canonical);
33
+ }
34
+ } catch {
35
+ // missing or malformed — treat as untrusted
36
+ }
37
+ return false;
38
+ }
39
+
40
+ // ─── Ready-promise singleton ────────────────────────────────────────────
41
+
42
+ let _readyPromise: Promise<void> | null = null;
43
+ let _untrustedWarned = false;
44
+
45
+ /**
46
+ * Discover and register ecosystem extensions from `./.gsd/extensions/`.
47
+ * Idempotent: subsequent calls with the same arguments return the same
48
+ * pending promise (no double-load).
49
+ */
50
+ export function loadEcosystemExtensions(
51
+ pi: ExtensionAPI,
52
+ sharedHandlers: GSDEcosystemBeforeAgentStartHandler[],
53
+ cwd: string = process.cwd(),
54
+ ): Promise<void> {
55
+ if (_readyPromise) return _readyPromise;
56
+ _readyPromise = _loadEcosystemExtensionsImpl(pi, sharedHandlers, cwd);
57
+ return _readyPromise;
58
+ }
59
+
60
+ /**
61
+ * Returns a promise that resolves when ecosystem loading has completed.
62
+ * If loading was never kicked off this returns a resolved promise so the
63
+ * `before_agent_start` handler can `await` unconditionally.
64
+ */
65
+ export function getEcosystemReadyPromise(): Promise<void> {
66
+ return _readyPromise ?? Promise.resolve();
67
+ }
68
+
69
+ /** Test-only: clear the singleton so tests can re-run loading. */
70
+ export function _resetEcosystemLoader(): void {
71
+ _readyPromise = null;
72
+ _untrustedWarned = false;
73
+ }
74
+
75
+ // ─── Implementation ─────────────────────────────────────────────────────
76
+
77
+ async function _loadEcosystemExtensionsImpl(
78
+ pi: ExtensionAPI,
79
+ sharedHandlers: GSDEcosystemBeforeAgentStartHandler[],
80
+ cwd: string,
81
+ ): Promise<void> {
82
+ const extDir = path.join(cwd, ".gsd", "extensions");
83
+ if (!fs.existsSync(extDir)) return;
84
+
85
+ // Trust gate: refuse to load arbitrary code from untrusted project dirs.
86
+ if (!isProjectTrusted(cwd, getAgentDir())) {
87
+ if (!_untrustedWarned) {
88
+ _untrustedWarned = true;
89
+ logWarning(
90
+ "ecosystem",
91
+ ".gsd/extensions present but project is not trusted — skipping ecosystem extensions. Run `pi trust` to opt in.",
92
+ );
93
+ }
94
+ return;
95
+ }
96
+
97
+ // Resolve realpath ONCE so symlink-escape detection has a stable anchor.
98
+ let realExtDir: string;
99
+ try {
100
+ realExtDir = fs.realpathSync(extDir);
101
+ } catch (err) {
102
+ logWarning(
103
+ "ecosystem",
104
+ `failed to resolve extensions dir: ${err instanceof Error ? err.message : String(err)}`,
105
+ );
106
+ return;
107
+ }
108
+
109
+ let entries: string[];
110
+ try {
111
+ entries = fs
112
+ .readdirSync(extDir)
113
+ .filter((f) => f.endsWith(".js") || f.endsWith(".ts"))
114
+ .sort(); // deterministic load order
115
+ } catch (err) {
116
+ logWarning(
117
+ "ecosystem",
118
+ `failed to read extensions dir: ${err instanceof Error ? err.message : String(err)}`,
119
+ );
120
+ return;
121
+ }
122
+
123
+ // The wrapper api is built once per loader run and shared by all extensions
124
+ // so they all read from the same module-level snapshot.
125
+ const api: GSDExtensionAPI = createGSDExtensionAPI(pi, sharedHandlers);
126
+
127
+ for (const entry of entries) {
128
+ await _loadOne(extDir, realExtDir, entry, api);
129
+ }
130
+ }
131
+
132
+ async function _loadOne(
133
+ extDir: string,
134
+ realExtDir: string,
135
+ entry: string,
136
+ api: GSDExtensionAPI,
137
+ ): Promise<void> {
138
+ const fullPath = path.join(extDir, entry);
139
+
140
+ // Symlink-escape guard: reject entries whose realpath is not under realExtDir.
141
+ let realFullPath: string;
142
+ try {
143
+ realFullPath = fs.realpathSync(fullPath);
144
+ } catch (err) {
145
+ logWarning(
146
+ "ecosystem",
147
+ `failed to resolve ${entry}: ${err instanceof Error ? err.message : String(err)}`,
148
+ );
149
+ return;
150
+ }
151
+ const realExtDirWithSep = realExtDir.endsWith(path.sep) ? realExtDir : realExtDir + path.sep;
152
+ if (
153
+ realFullPath !== realExtDir &&
154
+ !realFullPath.startsWith(realExtDirWithSep)
155
+ ) {
156
+ logWarning("ecosystem", `rejecting ${entry}: realpath escapes extensions dir`);
157
+ return;
158
+ }
159
+
160
+ // For .ts files, require a sibling compiled .js — we do not run a TS loader
161
+ // in production. Drop mtime heuristics: if .js exists, prefer it; otherwise warn.
162
+ let importPath = realFullPath;
163
+ if (entry.endsWith(".ts")) {
164
+ const jsSibling = realFullPath.slice(0, -3) + ".js";
165
+ if (fs.existsSync(jsSibling)) {
166
+ importPath = jsSibling;
167
+ } else {
168
+ logWarning(
169
+ "ecosystem",
170
+ `${entry}: TypeScript source has no compiled .js sibling — compile it first`,
171
+ );
172
+ return;
173
+ }
174
+ }
175
+
176
+ let mod: any;
177
+ try {
178
+ mod = await import(pathToFileURL(importPath).href);
179
+ } catch (err) {
180
+ logWarning(
181
+ "ecosystem",
182
+ `failed to import ${entry}: ${err instanceof Error ? err.message : String(err)}`,
183
+ );
184
+ return;
185
+ }
186
+
187
+ const factory = mod?.default;
188
+ if (typeof factory !== "function") {
189
+ logWarning("ecosystem", `${entry}: default export is not a function`);
190
+ return;
191
+ }
192
+
193
+ try {
194
+ await factory(api);
195
+ } catch (err) {
196
+ logWarning(
197
+ "ecosystem",
198
+ `factory threw for ${entry}: ${err instanceof Error ? err.message : String(err)}`,
199
+ );
200
+ }
201
+ }
@@ -34,6 +34,7 @@ import {
34
34
  nativeAddPaths,
35
35
  nativeResetSoft,
36
36
  nativeCommitSubject,
37
+ _resetHasChangesCache,
37
38
  } from "./native-git-bridge.js";
38
39
  import { GSDError, GSD_MERGE_CONFLICT, GSD_GIT_ERROR } from "./errors.js";
39
40
  import { getErrorMessage } from "./error-utils.js";
@@ -93,6 +94,17 @@ export interface CommitOptions {
93
94
  allowEmpty?: boolean;
94
95
  }
95
96
 
97
+ export type TurnGitActionMode = "commit" | "snapshot" | "status-only";
98
+
99
+ export interface TurnGitActionResult {
100
+ action: TurnGitActionMode;
101
+ status: "ok" | "failed";
102
+ commitMessage?: string;
103
+ snapshotLabel?: string;
104
+ dirty?: boolean;
105
+ error?: string;
106
+ }
107
+
96
108
  // ─── Meaningful Commit Message Generation ───────────────────────────────────
97
109
 
98
110
  /** Context for generating a meaningful commit message from task execution results. */
@@ -822,6 +834,62 @@ export function createGitService(basePath: string): GitServiceImpl {
822
834
  return new GitServiceImpl(basePath, gitPrefs);
823
835
  }
824
836
 
837
+ function buildTurnSnapshotLabel(unitType: string, unitId: string): string {
838
+ const raw = `${unitType}/${unitId}`.trim();
839
+ if (!raw) return "turn";
840
+ return raw
841
+ .replace(/[^a-zA-Z0-9._/-]/g, "-")
842
+ .replace(/\/{2,}/g, "/")
843
+ .replace(/-{2,}/g, "-")
844
+ .replace(/^[-/]+|[-/]+$/g, "") || "turn";
845
+ }
846
+
847
+ export function runTurnGitAction(args: {
848
+ basePath: string;
849
+ action: TurnGitActionMode;
850
+ unitType: string;
851
+ unitId: string;
852
+ taskContext?: TaskCommitContext;
853
+ }): TurnGitActionResult {
854
+ try {
855
+ // Force fresh working-tree status per turn; nativeHasChanges caches briefly.
856
+ _resetHasChangesCache();
857
+ if (args.action === "status-only") {
858
+ return {
859
+ action: args.action,
860
+ status: "ok",
861
+ dirty: nativeHasChanges(args.basePath),
862
+ };
863
+ }
864
+
865
+ const git = createGitService(args.basePath);
866
+ if (args.action === "snapshot") {
867
+ const label = buildTurnSnapshotLabel(args.unitType, args.unitId);
868
+ git.createSnapshot(label);
869
+ return {
870
+ action: args.action,
871
+ status: "ok",
872
+ snapshotLabel: label,
873
+ dirty: nativeHasChanges(args.basePath),
874
+ };
875
+ }
876
+
877
+ const commitMessage = git.autoCommit(args.unitType, args.unitId, [], args.taskContext) ?? undefined;
878
+ return {
879
+ action: args.action,
880
+ status: "ok",
881
+ commitMessage,
882
+ dirty: nativeHasChanges(args.basePath),
883
+ };
884
+ } catch (err) {
885
+ return {
886
+ action: args.action,
887
+ status: "failed",
888
+ error: getErrorMessage(err),
889
+ };
890
+ }
891
+ }
892
+
825
893
  // ─── Commit Type Inference ─────────────────────────────────────────────────
826
894
 
827
895
  /**