gsd-pi 2.78.0-dev.aeeb2ca00 → 2.78.1-dev.0fdacd524

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 (471) hide show
  1. package/README.md +8 -7
  2. package/dist/bundled-resource-path.d.ts +7 -0
  3. package/dist/bundled-resource-path.js +34 -2
  4. package/dist/claude-cli-check.js +77 -38
  5. package/dist/cli-policy.d.ts +13 -0
  6. package/dist/cli-policy.js +17 -0
  7. package/dist/cli.js +95 -55
  8. package/dist/headless-query.d.ts +22 -0
  9. package/dist/headless-query.js +43 -8
  10. package/dist/headless.d.ts +10 -0
  11. package/dist/headless.js +16 -1
  12. package/dist/loader.js +9 -13
  13. package/dist/onboarding.d.ts +10 -0
  14. package/dist/onboarding.js +2 -2
  15. package/dist/provider-migrations.d.ts +2 -2
  16. package/dist/provider-migrations.js +5 -2
  17. package/dist/resource-loader.d.ts +5 -2
  18. package/dist/resource-loader.js +30 -13
  19. package/dist/resources/.managed-resources-content-hash +1 -0
  20. package/dist/resources/extensions/claude-code-cli/readiness.js +90 -46
  21. package/dist/resources/extensions/google-search/index.js +2 -6
  22. package/dist/resources/extensions/gsd/auto/loop.js +23 -0
  23. package/dist/resources/extensions/gsd/auto/phases.js +5 -13
  24. package/dist/resources/extensions/gsd/auto/run-unit.js +26 -12
  25. package/dist/resources/extensions/gsd/auto/session.js +5 -6
  26. package/dist/resources/extensions/gsd/auto-dashboard.js +3 -2
  27. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +55 -21
  28. package/dist/resources/extensions/gsd/auto-dispatch.js +18 -6
  29. package/dist/resources/extensions/gsd/auto-prompts.js +69 -2
  30. package/dist/resources/extensions/gsd/auto-recovery.js +43 -4
  31. package/dist/resources/extensions/gsd/auto-runtime-state.js +31 -0
  32. package/dist/resources/extensions/gsd/auto-start.js +1 -1
  33. package/dist/resources/extensions/gsd/auto-tool-tracking.js +2 -2
  34. package/dist/resources/extensions/gsd/auto-worktree.js +60 -13
  35. package/dist/resources/extensions/gsd/auto.js +39 -14
  36. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +14 -2
  37. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +7 -5
  38. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
  39. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +5 -4
  40. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +112 -31
  41. package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +11 -6
  42. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +22 -0
  43. package/dist/resources/extensions/gsd/bootstrap/system-context.js +45 -8
  44. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +121 -3
  45. package/dist/resources/extensions/gsd/commands/catalog.js +76 -5
  46. package/dist/resources/extensions/gsd/commands/handlers/core.js +23 -1
  47. package/dist/resources/extensions/gsd/commands/handlers/ops.js +8 -0
  48. package/dist/resources/extensions/gsd/commands-config.js +3 -2
  49. package/dist/resources/extensions/gsd/commands-extensions.js +46 -3
  50. package/dist/resources/extensions/gsd/commands-handlers.js +3 -2
  51. package/dist/resources/extensions/gsd/commands-mcp-status.js +3 -1
  52. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +10 -1
  53. package/dist/resources/extensions/gsd/commands-worktree.js +309 -0
  54. package/dist/resources/extensions/gsd/dashboard-overlay.js +1 -1
  55. package/dist/resources/extensions/gsd/docs/preferences-reference.md +10 -0
  56. package/dist/resources/extensions/gsd/doctor-providers.js +2 -1
  57. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +39 -1
  58. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  59. package/dist/resources/extensions/gsd/forensics.js +10 -8
  60. package/dist/resources/extensions/gsd/git-service.js +12 -5
  61. package/dist/resources/extensions/gsd/gsd-db.js +11 -2
  62. package/dist/resources/extensions/gsd/guided-flow.js +25 -24
  63. package/dist/resources/extensions/gsd/home-dir.js +16 -0
  64. package/dist/resources/extensions/gsd/key-manager.js +2 -1
  65. package/dist/resources/extensions/gsd/memory-store.js +66 -31
  66. package/dist/resources/extensions/gsd/migrate/command.js +3 -2
  67. package/dist/resources/extensions/gsd/milestone-id-reservation.js +36 -0
  68. package/dist/resources/extensions/gsd/model-router.js +114 -9
  69. package/dist/resources/extensions/gsd/native-git-bridge.js +7 -1
  70. package/dist/resources/extensions/gsd/preferences-models.js +91 -15
  71. package/dist/resources/extensions/gsd/preferences-types.js +2 -0
  72. package/dist/resources/extensions/gsd/preferences-validation.js +32 -0
  73. package/dist/resources/extensions/gsd/preferences.js +5 -3
  74. package/dist/resources/extensions/gsd/prompt-loader.js +23 -12
  75. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +10 -0
  76. package/dist/resources/extensions/gsd/prompts/complete-slice.md +10 -0
  77. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
  78. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +2 -0
  79. package/dist/resources/extensions/gsd/prompts/plan-slice.md +10 -0
  80. package/dist/resources/extensions/gsd/prompts/refine-slice.md +10 -0
  81. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +2 -0
  82. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +9 -3
  83. package/dist/resources/extensions/gsd/state.js +42 -0
  84. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  85. package/dist/resources/extensions/gsd/tools/memory-tools.js +18 -1
  86. package/dist/resources/extensions/gsd/unit-context-manifest.js +29 -4
  87. package/dist/resources/extensions/gsd/visualizer-overlay.js +1 -1
  88. package/dist/resources/extensions/gsd/watch/header-renderer.js +3 -1
  89. package/dist/resources/extensions/gsd/worktree-command.js +26 -46
  90. package/dist/resources/extensions/gsd/worktree-manager.js +20 -1
  91. package/dist/resources/extensions/gsd/worktree-resolver.js +4 -13
  92. package/dist/resources/extensions/gsd/worktree-root.js +124 -0
  93. package/dist/resources/extensions/gsd/worktree-session-state.js +33 -0
  94. package/dist/resources/extensions/gsd/worktree.js +4 -115
  95. package/dist/resources/extensions/mcp-client/index.js +6 -9
  96. package/dist/resources/extensions/ollama/index.js +15 -2
  97. package/dist/resources/extensions/ollama/model-capabilities.js +31 -0
  98. package/dist/resources/extensions/ollama/ollama-client.js +40 -4
  99. package/dist/resources/extensions/slash-commands/create-extension.js +36 -22
  100. package/dist/resources/extensions/subagent/index.js +324 -178
  101. package/dist/resources/skills/create-gsd-extension/SKILL.md +9 -5
  102. package/dist/resources/skills/create-gsd-extension/references/custom-commands.md +1 -1
  103. package/dist/resources/skills/create-gsd-extension/references/custom-rendering.md +5 -5
  104. package/dist/resources/skills/create-gsd-extension/references/custom-tools.md +4 -4
  105. package/dist/resources/skills/create-gsd-extension/references/custom-ui.md +6 -6
  106. package/dist/resources/skills/create-gsd-extension/references/events-reference.md +3 -3
  107. package/dist/resources/skills/create-gsd-extension/references/packaging-distribution.md +1 -1
  108. package/dist/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +3 -3
  109. package/dist/resources/skills/create-gsd-extension/workflows/create-extension.md +32 -12
  110. package/dist/resources/skills/lint/SKILL.md +4 -0
  111. package/dist/resources/skills/review/SKILL.md +4 -0
  112. package/dist/resources/skills/test/SKILL.md +3 -0
  113. package/dist/rtk-shared.d.ts +3 -0
  114. package/dist/rtk-shared.js +17 -0
  115. package/dist/rtk.d.ts +2 -5
  116. package/dist/rtk.js +3 -20
  117. package/dist/runtime-checks.d.ts +27 -0
  118. package/dist/runtime-checks.js +38 -0
  119. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  120. package/dist/web/standalone/.next/BUILD_ID +1 -1
  121. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  122. package/dist/web/standalone/.next/build-manifest.json +3 -3
  123. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  124. package/dist/web/standalone/.next/react-loadable-manifest.json +44 -4
  125. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  127. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  128. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  129. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  130. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  131. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  132. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  134. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  135. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  136. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  138. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  140. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  141. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  142. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  143. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/session/events/route.js +4 -2
  146. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  147. package/dist/web/standalone/.next/server/app/index.html +1 -1
  148. package/dist/web/standalone/.next/server/app/index.rsc +2 -2
  149. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  150. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
  151. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  152. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  153. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  154. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  155. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  156. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  157. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  158. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  160. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  161. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  162. package/dist/web/standalone/.next/server/webpack-runtime.js +1 -1
  163. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +1 -0
  164. package/dist/web/standalone/.next/static/chunks/2824.08296bc2f9654698.js +1 -0
  165. package/dist/web/standalone/.next/static/chunks/3026.3af53b279375f082.js +1 -0
  166. package/dist/web/standalone/.next/static/chunks/315.6f68ae79b67d25cf.js +1 -0
  167. package/dist/web/standalone/.next/static/chunks/3497.4bfc60a3b3dea717.js +1 -0
  168. package/dist/web/standalone/.next/static/chunks/5516.4a07c872b5c3a663.js +1 -0
  169. package/dist/web/standalone/.next/static/chunks/8336.31b019697882acfb.js +10 -0
  170. package/dist/web/standalone/.next/static/chunks/8845.c9702695e8c5a9c5.js +2 -0
  171. package/dist/web/standalone/.next/static/chunks/9058.01ef3a463bda88f1.js +20 -0
  172. package/dist/web/standalone/.next/static/chunks/9441.1081da1125d1764f.js +1 -0
  173. package/dist/web/standalone/.next/static/chunks/app/{page-5b113fd32bc2a1c3.js → page-9bf2e0c50fb2ca05.js} +1 -1
  174. package/dist/web/standalone/.next/static/chunks/webpack-f9f0dc45e4f3ac10.js +1 -0
  175. package/dist/web/standalone/package.json +2 -1
  176. package/dist/welcome-screen.js +27 -1
  177. package/dist/worktree-cli.d.ts +1 -0
  178. package/dist/worktree-cli.js +9 -3
  179. package/dist/worktree-status-banner.d.ts +1 -0
  180. package/dist/worktree-status-banner.js +132 -0
  181. package/package.json +1 -3
  182. package/packages/daemon/package.json +2 -2
  183. package/packages/mcp-server/dist/alias-telemetry.d.ts +8 -0
  184. package/packages/mcp-server/dist/alias-telemetry.d.ts.map +1 -0
  185. package/packages/mcp-server/dist/alias-telemetry.js +30 -0
  186. package/packages/mcp-server/dist/alias-telemetry.js.map +1 -0
  187. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  188. package/packages/mcp-server/dist/workflow-tools.js +74 -46
  189. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  190. package/packages/mcp-server/package.json +2 -2
  191. package/packages/mcp-server/src/alias-telemetry.test.ts +78 -0
  192. package/packages/mcp-server/src/alias-telemetry.ts +30 -0
  193. package/packages/mcp-server/src/workflow-tools.test.ts +78 -0
  194. package/packages/mcp-server/src/workflow-tools.ts +93 -58
  195. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  196. package/packages/native/package.json +1 -1
  197. package/packages/native/tsconfig.tsbuildinfo +1 -1
  198. package/packages/pi-agent-core/package.json +1 -1
  199. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  200. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.d.ts +2 -0
  201. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.d.ts.map +1 -0
  202. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.js +231 -0
  203. package/packages/pi-ai/dist/providers/anthropic-shared.cache-breakpoint.test.js.map +1 -0
  204. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  205. package/packages/pi-ai/dist/providers/anthropic-shared.js +48 -19
  206. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  207. package/packages/pi-ai/dist/types.d.ts +13 -0
  208. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  209. package/packages/pi-ai/dist/types.js.map +1 -1
  210. package/packages/pi-ai/dist/utils/repair-tool-json.d.ts.map +1 -1
  211. package/packages/pi-ai/dist/utils/repair-tool-json.js +24 -3
  212. package/packages/pi-ai/dist/utils/repair-tool-json.js.map +1 -1
  213. package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js +26 -0
  214. package/packages/pi-ai/dist/utils/tests/repair-tool-json.test.js.map +1 -1
  215. package/packages/pi-ai/package.json +1 -1
  216. package/packages/pi-ai/src/providers/anthropic-shared.cache-breakpoint.test.ts +289 -0
  217. package/packages/pi-ai/src/providers/anthropic-shared.ts +52 -20
  218. package/packages/pi-ai/src/types.ts +13 -0
  219. package/packages/pi-ai/src/utils/repair-tool-json.ts +24 -3
  220. package/packages/pi-ai/src/utils/tests/repair-tool-json.test.ts +32 -0
  221. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  222. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  223. package/packages/pi-coding-agent/dist/core/agent-session.js +6 -0
  224. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  225. package/packages/pi-coding-agent/dist/core/messages.d.ts.map +1 -1
  226. package/packages/pi-coding-agent/dist/core/messages.js +4 -0
  227. package/packages/pi-coding-agent/dist/core/messages.js.map +1 -1
  228. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +19 -2
  229. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  230. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +10 -0
  231. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  232. package/packages/pi-coding-agent/dist/core/model-registry.js +18 -0
  233. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  234. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +13 -0
  235. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  236. package/packages/pi-coding-agent/dist/core/system-prompt.js +20 -16
  237. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  238. package/packages/pi-coding-agent/dist/core/token-telemetry.d.ts +37 -0
  239. package/packages/pi-coding-agent/dist/core/token-telemetry.d.ts.map +1 -0
  240. package/packages/pi-coding-agent/dist/core/token-telemetry.js +49 -0
  241. package/packages/pi-coding-agent/dist/core/token-telemetry.js.map +1 -0
  242. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.d.ts +2 -0
  243. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.d.ts.map +1 -0
  244. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js +133 -0
  245. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.js.map +1 -0
  246. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +1 -1
  247. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  248. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +14 -1
  249. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  250. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.d.ts +2 -0
  251. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.d.ts.map +1 -0
  252. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.js +78 -0
  253. package/packages/pi-coding-agent/dist/tests/system-prompt-cache-stability.test.js.map +1 -0
  254. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.d.ts +2 -0
  255. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.d.ts.map +1 -0
  256. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.js +181 -0
  257. package/packages/pi-coding-agent/dist/tests/token-telemetry.test.js.map +1 -0
  258. package/packages/pi-coding-agent/package.json +1 -1
  259. package/packages/pi-coding-agent/src/core/agent-session.ts +7 -0
  260. package/packages/pi-coding-agent/src/core/messages.ts +4 -0
  261. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +32 -2
  262. package/packages/pi-coding-agent/src/core/model-registry.ts +21 -0
  263. package/packages/pi-coding-agent/src/core/system-prompt.ts +33 -15
  264. package/packages/pi-coding-agent/src/core/token-telemetry.ts +77 -0
  265. package/packages/pi-coding-agent/src/modes/interactive/components/tool-card-cleanup-and-success-runtime.test.ts +212 -0
  266. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +17 -1
  267. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +1 -1
  268. package/packages/pi-coding-agent/src/tests/system-prompt-cache-stability.test.ts +102 -0
  269. package/packages/pi-coding-agent/src/tests/token-telemetry.test.ts +200 -0
  270. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  271. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +17 -3
  272. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  273. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.d.ts +2 -0
  274. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.d.ts.map +1 -0
  275. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.js +161 -0
  276. package/packages/pi-tui/dist/components/__tests__/leak-fixes-runtime.test.js.map +1 -0
  277. package/packages/pi-tui/package.json +1 -1
  278. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +20 -3
  279. package/packages/pi-tui/src/components/__tests__/leak-fixes-runtime.test.ts +219 -0
  280. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  281. package/packages/rpc-client/package.json +1 -1
  282. package/pkg/package.json +1 -1
  283. package/src/resources/extensions/claude-code-cli/readiness.ts +92 -47
  284. package/src/resources/extensions/google-search/index.ts +2 -9
  285. package/src/resources/extensions/gsd/auto/loop.ts +24 -2
  286. package/src/resources/extensions/gsd/auto/phases.ts +6 -14
  287. package/src/resources/extensions/gsd/auto/run-unit.ts +26 -12
  288. package/src/resources/extensions/gsd/auto/session.ts +5 -6
  289. package/src/resources/extensions/gsd/auto/types.ts +1 -0
  290. package/src/resources/extensions/gsd/auto-dashboard.ts +3 -2
  291. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +60 -24
  292. package/src/resources/extensions/gsd/auto-dispatch.ts +18 -6
  293. package/src/resources/extensions/gsd/auto-prompts.ts +66 -2
  294. package/src/resources/extensions/gsd/auto-recovery.ts +46 -8
  295. package/src/resources/extensions/gsd/auto-runtime-state.ts +51 -0
  296. package/src/resources/extensions/gsd/auto-start.ts +1 -1
  297. package/src/resources/extensions/gsd/auto-tool-tracking.ts +2 -4
  298. package/src/resources/extensions/gsd/auto-worktree.ts +82 -12
  299. package/src/resources/extensions/gsd/auto.ts +37 -10
  300. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +15 -13
  301. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +8 -7
  302. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
  303. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +10 -9
  304. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +121 -31
  305. package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +12 -6
  306. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +20 -0
  307. package/src/resources/extensions/gsd/bootstrap/system-context.ts +50 -8
  308. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +141 -11
  309. package/src/resources/extensions/gsd/commands/catalog.ts +82 -5
  310. package/src/resources/extensions/gsd/commands/handlers/core.ts +23 -1
  311. package/src/resources/extensions/gsd/commands/handlers/ops.ts +10 -0
  312. package/src/resources/extensions/gsd/commands-config.ts +3 -2
  313. package/src/resources/extensions/gsd/commands-extensions.ts +43 -3
  314. package/src/resources/extensions/gsd/commands-handlers.ts +3 -2
  315. package/src/resources/extensions/gsd/commands-mcp-status.ts +3 -1
  316. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +15 -1
  317. package/src/resources/extensions/gsd/commands-worktree.ts +383 -0
  318. package/src/resources/extensions/gsd/dashboard-overlay.ts +1 -1
  319. package/src/resources/extensions/gsd/docs/preferences-reference.md +10 -0
  320. package/src/resources/extensions/gsd/doctor-providers.ts +2 -1
  321. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +39 -1
  322. package/src/resources/extensions/gsd/doctor-types.ts +3 -1
  323. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  324. package/src/resources/extensions/gsd/forensics.ts +12 -7
  325. package/src/resources/extensions/gsd/git-service.ts +13 -5
  326. package/src/resources/extensions/gsd/gsd-db.ts +12 -2
  327. package/src/resources/extensions/gsd/guided-flow.ts +27 -26
  328. package/src/resources/extensions/gsd/home-dir.ts +19 -0
  329. package/src/resources/extensions/gsd/journal.ts +4 -1
  330. package/src/resources/extensions/gsd/key-manager.ts +2 -1
  331. package/src/resources/extensions/gsd/memory-store.ts +81 -28
  332. package/src/resources/extensions/gsd/migrate/command.ts +3 -2
  333. package/src/resources/extensions/gsd/milestone-id-reservation.ts +47 -0
  334. package/src/resources/extensions/gsd/model-router.ts +172 -9
  335. package/src/resources/extensions/gsd/native-git-bridge.ts +7 -1
  336. package/src/resources/extensions/gsd/preferences-models.ts +101 -15
  337. package/src/resources/extensions/gsd/preferences-types.ts +6 -0
  338. package/src/resources/extensions/gsd/preferences-validation.ts +35 -0
  339. package/src/resources/extensions/gsd/preferences.ts +16 -2
  340. package/src/resources/extensions/gsd/prompt-loader.ts +26 -12
  341. package/src/resources/extensions/gsd/prompts/complete-milestone.md +10 -0
  342. package/src/resources/extensions/gsd/prompts/complete-slice.md +10 -0
  343. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
  344. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +2 -0
  345. package/src/resources/extensions/gsd/prompts/plan-slice.md +10 -0
  346. package/src/resources/extensions/gsd/prompts/refine-slice.md +10 -0
  347. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +2 -0
  348. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +9 -3
  349. package/src/resources/extensions/gsd/state.ts +42 -0
  350. package/src/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  351. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +179 -1
  352. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +58 -0
  353. package/src/resources/extensions/gsd/tests/auto-session-encapsulation.test.ts +24 -5
  354. package/src/resources/extensions/gsd/tests/auto-supervisor.test.mjs +21 -4
  355. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +1 -1
  356. package/src/resources/extensions/gsd/tests/budget-prediction.test.ts +138 -211
  357. package/src/resources/extensions/gsd/tests/bundled-skill-triggers.test.ts +50 -27
  358. package/src/resources/extensions/gsd/tests/commands-extensions-version-compare.test.ts +58 -0
  359. package/src/resources/extensions/gsd/tests/commands-worktree-clean.test.ts +48 -0
  360. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +142 -59
  361. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +7 -4
  362. package/src/resources/extensions/gsd/tests/completed-at-reconcile.test.ts +89 -32
  363. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +41 -23
  364. package/src/resources/extensions/gsd/tests/db-path-worktree-symlink.test.ts +3 -43
  365. package/src/resources/extensions/gsd/tests/debug-logger.test.ts +5 -3
  366. package/src/resources/extensions/gsd/tests/deferred-milestone-dir-4996.test.ts +116 -0
  367. package/src/resources/extensions/gsd/tests/discuss-empty-db-fallback.test.ts +22 -87
  368. package/src/resources/extensions/gsd/tests/discuss-queued-milestones.test.ts +7 -118
  369. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +18 -60
  370. package/src/resources/extensions/gsd/tests/doctor-orphan-milestone-4996.test.ts +100 -0
  371. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +14 -76
  372. package/src/resources/extensions/gsd/tests/ensure-preconditions-guard-4996.test.ts +93 -0
  373. package/src/resources/extensions/gsd/tests/false-degraded-mode-warning.test.ts +22 -83
  374. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +1 -63
  375. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed-runtime.test.ts +47 -0
  376. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +26 -1
  377. package/src/resources/extensions/gsd/tests/gitignore-bg-shell-runtime.test.ts +63 -0
  378. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +25 -65
  379. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +30 -0
  380. package/src/resources/extensions/gsd/tests/gsd-no-project-error-runtime.test.ts +81 -0
  381. package/src/resources/extensions/gsd/tests/headless-answers.test.ts +14 -4
  382. package/src/resources/extensions/gsd/tests/health-widget.test.ts +22 -12
  383. package/src/resources/extensions/gsd/tests/help-menu-coverage.test.ts +57 -0
  384. package/src/resources/extensions/gsd/tests/home-dir.test.ts +52 -0
  385. package/src/resources/extensions/gsd/tests/import-done-milestones-runtime.test.ts +145 -0
  386. package/src/resources/extensions/gsd/tests/init-prefs-routing.test.ts +64 -1
  387. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +72 -1
  388. package/src/resources/extensions/gsd/tests/integration/token-savings.test.ts +0 -23
  389. package/src/resources/extensions/gsd/tests/memory-store.test.ts +128 -0
  390. package/src/resources/extensions/gsd/tests/memory-tools.test.ts +33 -1
  391. package/src/resources/extensions/gsd/tests/merge-self-branch-guard.test.ts +124 -0
  392. package/src/resources/extensions/gsd/tests/milestone-id-gap-reuse-4996.test.ts +152 -0
  393. package/src/resources/extensions/gsd/tests/milestone-report-path.test.ts +18 -1
  394. package/src/resources/extensions/gsd/tests/model-router.test.ts +169 -8
  395. package/src/resources/extensions/gsd/tests/native-git-infra-errors.test.ts +50 -0
  396. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +8 -0
  397. package/src/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +32 -43
  398. package/src/resources/extensions/gsd/tests/phases-merge-error-stops-auto.test.ts +4 -10
  399. package/src/resources/extensions/gsd/tests/preferences.test.ts +127 -0
  400. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +16 -0
  401. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +7 -0
  402. package/src/resources/extensions/gsd/tests/quick-turn-end-cleanup.test.ts +6 -6
  403. package/src/resources/extensions/gsd/tests/register-hooks-compaction-checkpoint.test.ts +93 -0
  404. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +34 -0
  405. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +168 -19
  406. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +7 -1
  407. package/src/resources/extensions/gsd/tests/smart-entry-complete.test.ts +23 -1
  408. package/src/resources/extensions/gsd/tests/steer-worktree-path.test.ts +17 -1
  409. package/src/resources/extensions/gsd/tests/system-context-message-routing.test.ts +101 -0
  410. package/src/resources/extensions/gsd/tests/token-profile.test.ts +51 -4
  411. package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +7 -16
  412. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +38 -3
  413. package/src/resources/extensions/gsd/tests/unstructured-continue-context-injection.test.ts +5 -7
  414. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +15 -1
  415. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +6 -6
  416. package/src/resources/extensions/gsd/tests/worktree-path-injection.test.ts +235 -0
  417. package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +34 -33
  418. package/src/resources/extensions/gsd/tests/worktree.test.ts +8 -0
  419. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +131 -1
  420. package/src/resources/extensions/gsd/tools/memory-tools.ts +17 -1
  421. package/src/resources/extensions/gsd/unit-context-manifest.ts +44 -12
  422. package/src/resources/extensions/gsd/visualizer-overlay.ts +1 -1
  423. package/src/resources/extensions/gsd/watch/header-renderer.ts +3 -1
  424. package/src/resources/extensions/gsd/workflow-logger.ts +1 -0
  425. package/src/resources/extensions/gsd/worktree-command.ts +31 -44
  426. package/src/resources/extensions/gsd/worktree-manager.ts +40 -1
  427. package/src/resources/extensions/gsd/worktree-resolver.ts +4 -14
  428. package/src/resources/extensions/gsd/worktree-root.ts +144 -0
  429. package/src/resources/extensions/gsd/worktree-session-state.ts +35 -0
  430. package/src/resources/extensions/gsd/worktree.ts +8 -119
  431. package/src/resources/extensions/mcp-client/index.ts +6 -10
  432. package/src/resources/extensions/mcp-client/tests/global-config.test.ts +91 -0
  433. package/src/resources/extensions/ollama/index.ts +16 -2
  434. package/src/resources/extensions/ollama/model-capabilities.ts +34 -0
  435. package/src/resources/extensions/ollama/ollama-client.ts +41 -4
  436. package/src/resources/extensions/ollama/tests/model-capabilities.test.ts +96 -0
  437. package/src/resources/extensions/ollama/tests/ollama-client-timeout-env.test.ts +147 -0
  438. package/src/resources/extensions/slash-commands/create-extension.ts +38 -24
  439. package/src/resources/extensions/subagent/index.ts +165 -7
  440. package/src/resources/skills/create-gsd-extension/SKILL.md +9 -5
  441. package/src/resources/skills/create-gsd-extension/references/custom-commands.md +1 -1
  442. package/src/resources/skills/create-gsd-extension/references/custom-rendering.md +5 -5
  443. package/src/resources/skills/create-gsd-extension/references/custom-tools.md +4 -4
  444. package/src/resources/skills/create-gsd-extension/references/custom-ui.md +6 -6
  445. package/src/resources/skills/create-gsd-extension/references/events-reference.md +3 -3
  446. package/src/resources/skills/create-gsd-extension/references/packaging-distribution.md +1 -1
  447. package/src/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +3 -3
  448. package/src/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +2 -2
  449. package/src/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +3 -3
  450. package/src/resources/skills/create-gsd-extension/templates/templates.test.ts +58 -0
  451. package/src/resources/skills/create-gsd-extension/workflows/create-extension.md +32 -12
  452. package/src/resources/skills/lint/SKILL.md +4 -0
  453. package/src/resources/skills/review/SKILL.md +4 -0
  454. package/src/resources/skills/test/SKILL.md +3 -0
  455. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +0 -601
  456. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +0 -651
  457. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +0 -91
  458. package/dist/resources/extensions/gsd/tests/auto-supervisor.test.mjs +0 -53
  459. package/dist/resources/extensions/gsd/tests/dist-redirect.mjs +0 -112
  460. package/dist/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +0 -23
  461. package/dist/resources/extensions/gsd/tests/resolve-ts.mjs +0 -5
  462. package/dist/resources/skills/github-workflows/references/gh/tests/__init__.py +0 -0
  463. package/dist/resources/skills/github-workflows/references/gh/tests/test_github_project_setup.py +0 -608
  464. package/dist/web/standalone/.next/static/chunks/2826.e9f5195e91f9cad2.js +0 -11
  465. package/dist/web/standalone/.next/static/chunks/3621.fc7480022c972438.js +0 -20
  466. package/dist/web/standalone/.next/static/chunks/webpack-2e68521d7c82f7c2.js +0 -1
  467. package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +0 -22
  468. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +0 -47
  469. package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +0 -75
  470. /package/dist/web/standalone/.next/static/{cAJH99yNS1UPbeSEiNRrV → 4iu6IYeYfxOq8OidlDqp6}/_buildManifest.js +0 -0
  471. /package/dist/web/standalone/.next/static/{cAJH99yNS1UPbeSEiNRrV → 4iu6IYeYfxOq8OidlDqp6}/_ssgManifest.js +0 -0
@@ -84,6 +84,102 @@ describe("getModelCapabilities", () => {
84
84
  });
85
85
  });
86
86
 
87
+ // ─── Ordering / prefix-shadowing regression (#4991) ──────────────────────────
88
+ //
89
+ // The lookup is a linear scan over KNOWN_MODELS using `baseName.startsWith(pattern)`.
90
+ // Cloud and long-variant model names share prefixes with their base families,
91
+ // so the longer entries MUST appear earlier in the table — otherwise a base
92
+ // like `qwen3` shadows `qwen3-coder`/`qwen3-next`/`qwen3.5` and the picker
93
+ // reports the wrong context window. These tests pin the ordering.
94
+
95
+ describe("getModelCapabilities — long-variant overrides aren't shadowed (#4991)", () => {
96
+ it("qwen3-coder reports 256K, not the qwen3 131K", () => {
97
+ const caps = getModelCapabilities("qwen3-coder:480b");
98
+ assert.equal(caps.contextWindow, 262144);
99
+ assert.equal(caps.ollamaOptions?.num_ctx, 262144);
100
+ });
101
+
102
+ it("qwen3-coder-next still resolves via the qwen3-coder entry", () => {
103
+ const caps = getModelCapabilities("qwen3-coder-next");
104
+ assert.equal(caps.contextWindow, 262144);
105
+ });
106
+
107
+ it("qwen3-next:80b reports 1M, not the qwen3 131K", () => {
108
+ const caps = getModelCapabilities("qwen3-next:80b");
109
+ assert.equal(caps.contextWindow, 1048576);
110
+ });
111
+
112
+ it("qwen3.5 / qwen3.6 cloud variants report 1M", () => {
113
+ assert.equal(getModelCapabilities("qwen3.5:397b").contextWindow, 1048576);
114
+ assert.equal(getModelCapabilities("qwen3.6:cloud").contextWindow, 1048576);
115
+ });
116
+
117
+ it("base qwen3 still resolves to its 131K entry", () => {
118
+ const caps = getModelCapabilities("qwen3:8b");
119
+ assert.equal(caps.contextWindow, 131072);
120
+ });
121
+
122
+ it("glm-5.1:cloud reports 200K", () => {
123
+ const caps = getModelCapabilities("glm-5.1:cloud");
124
+ assert.equal(caps.contextWindow, 204800);
125
+ });
126
+
127
+ it("glm-4.6:cloud reports 200K", () => {
128
+ const caps = getModelCapabilities("glm-4.6:cloud");
129
+ assert.equal(caps.contextWindow, 204800);
130
+ });
131
+
132
+ it("glm-4 base still resolves to its 131K entry", () => {
133
+ const caps = getModelCapabilities("glm-4:9b");
134
+ assert.equal(caps.contextWindow, 131072);
135
+ });
136
+
137
+ it("kimi-k2-thinking reports 256K (not shadowed by kimi-k2)", () => {
138
+ const caps = getModelCapabilities("kimi-k2-thinking");
139
+ assert.equal(caps.contextWindow, 262144);
140
+ });
141
+
142
+ it("kimi-k2.5:cloud and kimi-k2.6:cloud both report 256K", () => {
143
+ assert.equal(getModelCapabilities("kimi-k2.5:cloud").contextWindow, 262144);
144
+ assert.equal(getModelCapabilities("kimi-k2.6:cloud").contextWindow, 262144);
145
+ });
146
+
147
+ it("kimi-k2 base resolves to 256K", () => {
148
+ const caps = getModelCapabilities("kimi-k2:cloud");
149
+ assert.equal(caps.contextWindow, 262144);
150
+ });
151
+
152
+ it("minimax-m2.5:cloud and minimax-m2.7:cloud report 1M", () => {
153
+ assert.equal(getModelCapabilities("minimax-m2.5:cloud").contextWindow, 1048576);
154
+ assert.equal(getModelCapabilities("minimax-m2.7:cloud").contextWindow, 1048576);
155
+ });
156
+
157
+ it("minimax-m2 base resolves to 1M", () => {
158
+ const caps = getModelCapabilities("minimax-m2:cloud");
159
+ assert.equal(caps.contextWindow, 1048576);
160
+ });
161
+
162
+ it("ollamaOptions.num_ctx mirrors contextWindow for all new entries", () => {
163
+ // Inference time: num_ctx is what gets sent to Ollama on each chat.
164
+ // If contextWindow is right but num_ctx is stale, the model still
165
+ // gets truncated. Pin both sides.
166
+ for (const name of [
167
+ "qwen3-next:80b",
168
+ "qwen3-coder:480b",
169
+ "glm-5.1:cloud",
170
+ "kimi-k2-thinking",
171
+ "minimax-m2.7:cloud",
172
+ ]) {
173
+ const caps = getModelCapabilities(name);
174
+ assert.equal(
175
+ caps.ollamaOptions?.num_ctx,
176
+ caps.contextWindow,
177
+ `${name}: num_ctx (${caps.ollamaOptions?.num_ctx}) must equal contextWindow (${caps.contextWindow})`,
178
+ );
179
+ }
180
+ });
181
+ });
182
+
87
183
  // ─── estimateContextFromParams ───────────────────────────────────────────────
88
184
 
89
185
  describe("estimateContextFromParams", () => {
@@ -0,0 +1,147 @@
1
+ // GSD2 — Tests for OLLAMA_PROBE_TIMEOUT_MS / OLLAMA_REQUEST_TIMEOUT_MS env vars (#5003 / #4982)
2
+ //
3
+ // Pinned defaults: 1500 ms probe, 10 000 ms request. The defaults must be
4
+ // preserved when the env var is unset, empty, non-numeric, zero, or negative
5
+ // so a typo or fat-fingered value can't silently disable the timeout. When
6
+ // the env var is set to a valid positive integer it overrides the default.
7
+
8
+ import { describe, it, beforeEach, afterEach } from "node:test";
9
+ import assert from "node:assert/strict";
10
+ import {
11
+ envPositiveInt,
12
+ getProbeTimeoutMs,
13
+ getRequestTimeoutMs,
14
+ MAX_TIMER_DELAY_MS,
15
+ } from "../ollama-client.js";
16
+
17
+ const PROBE_VAR = "OLLAMA_PROBE_TIMEOUT_MS";
18
+ const REQUEST_VAR = "OLLAMA_REQUEST_TIMEOUT_MS";
19
+
20
+ function withEnv(name: string, value: string | undefined, run: () => void): void {
21
+ const prior = process.env[name];
22
+ if (value === undefined) {
23
+ delete process.env[name];
24
+ } else {
25
+ process.env[name] = value;
26
+ }
27
+ try {
28
+ run();
29
+ } finally {
30
+ if (prior === undefined) {
31
+ delete process.env[name];
32
+ } else {
33
+ process.env[name] = prior;
34
+ }
35
+ }
36
+ }
37
+
38
+ describe("envPositiveInt — defensive fallback", () => {
39
+ it("returns fallback when var is unset", () => {
40
+ withEnv("__GSD_TEST_INT__", undefined, () => {
41
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 42);
42
+ });
43
+ });
44
+
45
+ it("returns fallback when var is empty string", () => {
46
+ withEnv("__GSD_TEST_INT__", "", () => {
47
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 42);
48
+ });
49
+ });
50
+
51
+ it("returns fallback when var is non-numeric", () => {
52
+ withEnv("__GSD_TEST_INT__", "abc", () => {
53
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 42);
54
+ });
55
+ });
56
+
57
+ it("returns fallback when var is zero (would silently disable timeout)", () => {
58
+ withEnv("__GSD_TEST_INT__", "0", () => {
59
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 42);
60
+ });
61
+ });
62
+
63
+ it("returns fallback when var is negative", () => {
64
+ withEnv("__GSD_TEST_INT__", "-100", () => {
65
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 42);
66
+ });
67
+ });
68
+
69
+ it("returns parsed value when var is a positive integer", () => {
70
+ withEnv("__GSD_TEST_INT__", "5000", () => {
71
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 5000);
72
+ });
73
+ });
74
+
75
+ it("parses leading digits and discards trailing junk (parseInt semantics)", () => {
76
+ withEnv("__GSD_TEST_INT__", "1500ms", () => {
77
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), 1500);
78
+ });
79
+ });
80
+
81
+ it("clamps values above MAX_TIMER_DELAY_MS to prevent setTimeout overflow", () => {
82
+ withEnv("__GSD_TEST_INT__", String(MAX_TIMER_DELAY_MS + 1), () => {
83
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), MAX_TIMER_DELAY_MS);
84
+ });
85
+ });
86
+
87
+ it("accepts MAX_TIMER_DELAY_MS exactly", () => {
88
+ withEnv("__GSD_TEST_INT__", String(MAX_TIMER_DELAY_MS), () => {
89
+ assert.equal(envPositiveInt("__GSD_TEST_INT__", 42), MAX_TIMER_DELAY_MS);
90
+ });
91
+ });
92
+ });
93
+
94
+ describe("getProbeTimeoutMs — OLLAMA_PROBE_TIMEOUT_MS override", () => {
95
+ beforeEach(() => {
96
+ delete process.env[PROBE_VAR];
97
+ });
98
+ afterEach(() => {
99
+ delete process.env[PROBE_VAR];
100
+ });
101
+
102
+ it("defaults to 1500 ms when unset", () => {
103
+ assert.equal(getProbeTimeoutMs(), 1500);
104
+ });
105
+
106
+ it("honours a positive override", () => {
107
+ process.env[PROBE_VAR] = "5000";
108
+ assert.equal(getProbeTimeoutMs(), 5000);
109
+ });
110
+
111
+ it("falls back to 1500 ms on a zero override (typo guard)", () => {
112
+ process.env[PROBE_VAR] = "0";
113
+ assert.equal(getProbeTimeoutMs(), 1500);
114
+ });
115
+
116
+ it("re-reads the env var on every call", () => {
117
+ process.env[PROBE_VAR] = "2000";
118
+ assert.equal(getProbeTimeoutMs(), 2000);
119
+ process.env[PROBE_VAR] = "8000";
120
+ assert.equal(getProbeTimeoutMs(), 8000);
121
+ delete process.env[PROBE_VAR];
122
+ assert.equal(getProbeTimeoutMs(), 1500);
123
+ });
124
+ });
125
+
126
+ describe("getRequestTimeoutMs — OLLAMA_REQUEST_TIMEOUT_MS override", () => {
127
+ beforeEach(() => {
128
+ delete process.env[REQUEST_VAR];
129
+ });
130
+ afterEach(() => {
131
+ delete process.env[REQUEST_VAR];
132
+ });
133
+
134
+ it("defaults to 10 000 ms when unset", () => {
135
+ assert.equal(getRequestTimeoutMs(), 10000);
136
+ });
137
+
138
+ it("honours a positive override", () => {
139
+ process.env[REQUEST_VAR] = "30000";
140
+ assert.equal(getRequestTimeoutMs(), 30000);
141
+ });
142
+
143
+ it("falls back to 10 000 ms on non-numeric input", () => {
144
+ process.env[REQUEST_VAR] = "thirty-seconds";
145
+ assert.equal(getRequestTimeoutMs(), 10000);
146
+ });
147
+ });
@@ -221,41 +221,38 @@ function sendPrompt(description: string, result: RoundResult, pi: ExtensionAPI):
221
221
  : "";
222
222
 
223
223
  const docHints: string[] = [
224
- "- `~/.gsd/agent/docs/extending-pi/01-what-are-extensions.md` — capabilities overview",
225
- "- `~/.gsd/agent/docs/extending-pi/03-getting-started.md` — minimal extension, hot reload",
226
- "- `~/.gsd/agent/docs/extending-pi/08-extensioncontext-what-you-can-access.md` — ExtensionContext API",
227
- "- `~/.gsd/agent/docs/extending-pi/09-extensionapi-what-you-can-do.md` — ExtensionAPI: registration, messaging",
228
- "- `~/.gsd/agent/docs/extending-pi/22-key-rules-gotchas.md` — must-read rules before shipping",
224
+ "- `docs/extension-sdk/README.md` — overview, quick start, directory layout",
225
+ "- `docs/extension-sdk/api-reference.md` — ExtensionAPI and ExtensionContext surfaces",
226
+ "- `docs/extension-sdk/building-extensions.md` — tools, commands, events, UI, state",
227
+ "- `docs/extension-sdk/rules.md` — non-negotiable rules and gotchas",
229
228
  ];
230
229
 
231
230
  if (uiSelected.includes("custom component")) {
232
- docHints.push("- `~/.gsd/agent/docs/extending-pi/12-custom-ui-visual-components.md` — dialogs, widgets, overlays");
233
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/06-ctx-ui-custom-full-custom-components.md` — ctx.ui.custom() API");
234
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/07-built-in-components-the-building-blocks.md` — Text, Box, SelectList");
235
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/09-keyboard-input-how-to-handle-keys.md` — Key, matchesKey");
236
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/10-line-width-the-cardinal-rule.md` — truncation, width rules");
237
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/19-building-a-complete-component-step-by-step.md` — step-by-step guide");
238
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/21-common-mistakes-and-how-to-avoid-them.md` — pitfalls");
231
+ docHints.push("- `docs/extension-sdk/building-extensions.md#custom-components` — ctx.ui.custom() API");
232
+ docHints.push("- `docs/dev/pi-ui-tui/06-ctx-ui-custom-full-custom-components.md` — step-by-step component guide");
233
+ docHints.push("- `docs/dev/pi-ui-tui/07-built-in-components-the-building-blocks.md` — Text, Box, SelectList");
234
+ docHints.push("- `docs/dev/pi-ui-tui/09-keyboard-input-how-to-handle-keys.md` — Key, matchesKey");
235
+ docHints.push("- `docs/dev/pi-ui-tui/10-line-width-the-cardinal-rule.md` — truncation, width rules");
239
236
  } else if (uiSelected.includes("Dialogs")) {
240
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/04-built-in-dialog-methods.md` — select, confirm, input, editor");
237
+ docHints.push("- `docs/extension-sdk/building-extensions.md#built-in-dialogs` — select, confirm, input");
241
238
  } else if (uiSelected.includes("Status")) {
242
- docHints.push("- `~/.gsd/agent/docs/pi-ui-tui/05-persistent-ui-elements.md` — status, widgets, footer, header");
239
+ docHints.push("- `docs/extension-sdk/building-extensions.md#persistent-ui-elements` — status, widgets");
243
240
  }
244
241
 
245
242
  if (uiSelected.includes("tool") || result.answers["purpose"]) {
246
- docHints.push("- `~/.gsd/agent/docs/extending-pi/14-custom-rendering-controlling-what-the-user-sees.md` — renderCall / renderResult");
243
+ docHints.push("- `docs/dev/extending-pi/14-custom-rendering-controlling-what-the-user-sees.md` — renderCall / renderResult");
247
244
  }
248
245
 
249
246
  if (eventsSelected && !eventsSelected.includes("standalone")) {
250
- docHints.push("- `~/.gsd/agent/docs/extending-pi/07-events-the-nervous-system.md` — all events reference");
247
+ docHints.push("- `docs/dev/extending-pi/07-events-the-nervous-system.md` — all events reference");
251
248
  }
252
249
 
253
250
  if (eventsSelected.includes("context / prompt")) {
254
- docHints.push("- `~/.gsd/agent/docs/extending-pi/15-system-prompt-modification.md` — system prompt hooks");
251
+ docHints.push("- `docs/dev/extending-pi/15-system-prompt-modification.md` — system prompt hooks");
255
252
  }
256
253
 
257
254
  if (persistenceSelected.includes("session")) {
258
- docHints.push("- `~/.gsd/agent/docs/extending-pi/13-state-management-persistence.md` — pi.appendEntry, session state");
255
+ docHints.push("- `docs/extension-sdk/building-extensions.md#state-management` — state reconstruction, appendEntry");
259
256
  }
260
257
 
261
258
  const prompt = `Create a new pi extension based on this description:
@@ -270,13 +267,30 @@ ${docHints.join("\n")}
270
267
 
271
268
  ## Output
272
269
 
273
- Write the complete implementation as a single self-contained extension file:
274
-
275
- \`~/.gsd/agent/extensions/<kebab-case-name>.ts\`
276
-
277
- Then register it in the main extensions index:
270
+ Write the complete implementation as a directory-based extension:
271
+
272
+ \`~/.gsd/agent/extensions/<kebab-case-name>/index.ts\`
273
+ \`~/.gsd/agent/extensions/<kebab-case-name>/extension-manifest.json\`
274
+
275
+ The manifest must follow this format:
276
+ \`\`\`json
277
+ {
278
+ "id": "<kebab-case-name>",
279
+ "name": "<Human Name>",
280
+ "version": "1.0.0",
281
+ "description": "<one-line description>",
282
+ "tier": "community",
283
+ "requires": { "platform": ">=2.29.0" },
284
+ "provides": {
285
+ "tools": ["<tool_names_registered>"],
286
+ "commands": ["<command_names_registered>"],
287
+ "hooks": ["<event_names_subscribed>"],
288
+ "shortcuts": ["<shortcut_keys_registered>"]
289
+ }
290
+ }
291
+ \`\`\`
278
292
 
279
- \`~/.gsd/agent/extensions/index.ts\` import and call the new extension's default export alongside existing ones
293
+ Only include non-empty arrays in \`provides\`. See \`docs/extension-sdk/manifest-spec.md\` for the full spec.
280
294
 
281
295
  ## Rules you must follow exactly
282
296
 
@@ -36,6 +36,7 @@ import {
36
36
  } from "./isolation.js";
37
37
  import { registerWorker, updateWorker } from "./worker-registry.js";
38
38
  import { loadEffectiveGSDPreferences } from "../gsd/preferences.js";
39
+ import { emitJournalEvent } from "../gsd/journal.js";
39
40
  import { CmuxClient, shellEscape } from "../cmux/index.js";
40
41
 
41
42
  const MAX_PARALLEL_TASKS = 8;
@@ -662,11 +663,13 @@ export default function (pi: ExtensionAPI) {
662
663
  "Use chain mode to pipeline: scout finds context, planner designs, worker implements.",
663
664
  ].join(" "),
664
665
  promptGuidelines: [
665
- "Use subagent to delegate self-contained tasks that benefit from an isolated context window.",
666
- "Use scout agent first when you need codebase context before implementing.",
667
- "Use chain mode for scout→planner→worker or worker→reviewer→worker pipelines.",
668
- "Use parallel mode when tasks are independent and don't need each other's output.",
669
- "Always check available agents with /subagent before choosing one.",
666
+ "Prefer subagent dispatch over inline work whenever a task is self-contained recon, planning, review, refactor, test writing, security audit, doc writing. Each dispatch gets a fresh context window, so your main session stays focused on synthesis.",
667
+ "Before reading more than ~3 files to understand something, dispatch the scout agent and work from its compressed report instead.",
668
+ "Before any change touching ≥2 packages, the orchestration kernel, auto-mode, or a public API, dispatch the planner agent first. Plan first, then implement.",
669
+ "You MUST use parallel mode when ≥2 ready tasks are independent of each other's output. Do not serialize independent tasks manually — that wastes wall time and context.",
670
+ "Use chain mode for sequential pipelines where each step's output feeds the next: scout → planner → worker, or worker → reviewer → worker.",
671
+ "Before opening a PR or marking a slice complete, dispatch the reviewer agent (and security agent if the change touches auth, network, parsing, file IO, or shell exec).",
672
+ "Always check available agents with /subagent before choosing one — there are bundled specialists plus any project-scoped agents.",
670
673
  ],
671
674
  parameters: SubagentParams,
672
675
 
@@ -709,6 +712,141 @@ export default function (pi: ExtensionAPI) {
709
712
  };
710
713
  }
711
714
 
715
+ // Dispatch telemetry — emit invoked once per dispatch and completed before each return.
716
+ // Fresh flowId per dispatch (subagent runs aren't currently plumbed with the parent
717
+ // auto-mode flowId; per-dispatch ids still let us measure frequency, batch size, mode).
718
+ const dispatchMode: "single" | "parallel" | "chain" = hasChain ? "chain" : hasTasks ? "parallel" : "single";
719
+ const dispatchAgents = hasChain
720
+ ? params.chain!.map((s) => s.agent)
721
+ : hasTasks
722
+ ? params.tasks!.map((t) => t.agent)
723
+ : params.agent
724
+ ? [params.agent]
725
+ : [];
726
+ const dispatchTasks = hasChain
727
+ ? params.chain!.map((s) => s.task)
728
+ : hasTasks
729
+ ? params.tasks!.map((t) => t.task)
730
+ : params.task
731
+ ? [params.task]
732
+ : [];
733
+ const dispatchId = crypto.randomUUID();
734
+ const dispatchStartMs = Date.now();
735
+ let finalResults: SingleResult[] = [];
736
+ let dispatchCompletedEmitted = false;
737
+
738
+ emitJournalEvent(ctx.cwd, {
739
+ ts: new Date().toISOString(),
740
+ flowId: dispatchId,
741
+ seq: 0,
742
+ eventType: "subagent-invoked",
743
+ data: {
744
+ dispatchId,
745
+ mode: dispatchMode,
746
+ agents: dispatchAgents,
747
+ batchSize: dispatchAgents.length,
748
+ unitType: getCurrentPhase() ?? null,
749
+ isolated: useIsolation,
750
+ },
751
+ });
752
+
753
+ const zeroUsage = (): UsageStats => ({
754
+ input: 0,
755
+ output: 0,
756
+ cacheRead: 0,
757
+ cacheWrite: 0,
758
+ cost: 0,
759
+ contextTokens: 0,
760
+ turns: 0,
761
+ });
762
+ const errorMessageFor = (err: unknown): string =>
763
+ err instanceof Error ? err.message : String(err || "subagent dispatch failed");
764
+ const makeFailureResult = (err: unknown, agent: string, task: string, step?: number): SingleResult => {
765
+ const message = errorMessageFor(err);
766
+ return {
767
+ agent,
768
+ agentSource: "unknown",
769
+ task,
770
+ exitCode: 1,
771
+ messages: [],
772
+ stderr: message,
773
+ usage: zeroUsage(),
774
+ stopReason: signal?.aborted ? "aborted" : "error",
775
+ errorMessage: message,
776
+ ...(step !== undefined ? { step } : {}),
777
+ };
778
+ };
779
+ const synthesizeFailureResults = (err: unknown): SingleResult[] => {
780
+ if (finalResults.length > 0) {
781
+ let patchedRunning = false;
782
+ const patched = finalResults.map((result) => {
783
+ if (result.exitCode !== -1) return result;
784
+ patchedRunning = true;
785
+ const message = errorMessageFor(err);
786
+ return {
787
+ ...result,
788
+ exitCode: 1,
789
+ stderr: result.stderr || message,
790
+ stopReason: signal?.aborted ? "aborted" : "error",
791
+ errorMessage: result.errorMessage || message,
792
+ usage: result.usage ?? zeroUsage(),
793
+ };
794
+ });
795
+ if (patchedRunning || patched.some((result) => result.exitCode !== 0)) return patched;
796
+
797
+ const nextIndex = finalResults.length < dispatchAgents.length ? finalResults.length : 0;
798
+ if (nextIndex > 0) {
799
+ return [
800
+ ...finalResults,
801
+ makeFailureResult(
802
+ err,
803
+ dispatchAgents[nextIndex] ?? "unknown",
804
+ dispatchTasks[nextIndex] ?? "",
805
+ dispatchMode === "chain" ? nextIndex + 1 : undefined,
806
+ ),
807
+ ];
808
+ }
809
+ }
810
+
811
+ const agentsForFailure = dispatchAgents.length > 0 ? dispatchAgents : ["unknown"];
812
+ return agentsForFailure.map((agent, index) =>
813
+ makeFailureResult(
814
+ err,
815
+ agent,
816
+ dispatchTasks[index] ?? "",
817
+ dispatchMode === "chain" ? index + 1 : undefined,
818
+ ),
819
+ );
820
+ };
821
+ const finishDispatch = (results: SingleResult[]): void => {
822
+ if (dispatchCompletedEmitted) return;
823
+ finalResults = results;
824
+ dispatchCompletedEmitted = true;
825
+ const successCount = results.filter((r) => r.exitCode === 0).length;
826
+ const failureCount = results.filter((r) => r.exitCode !== 0).length;
827
+ const totalCost = results.reduce((s, r) => s + (r.usage?.cost ?? 0), 0);
828
+ const totalInputTokens = results.reduce((s, r) => s + (r.usage?.input ?? 0), 0);
829
+ const totalOutputTokens = results.reduce((s, r) => s + (r.usage?.output ?? 0), 0);
830
+ emitJournalEvent(ctx.cwd, {
831
+ ts: new Date().toISOString(),
832
+ flowId: dispatchId,
833
+ seq: 1,
834
+ eventType: "subagent-completed",
835
+ data: {
836
+ dispatchId,
837
+ mode: dispatchMode,
838
+ agents: dispatchAgents,
839
+ successCount,
840
+ failureCount,
841
+ totalCost,
842
+ totalInputTokens,
843
+ totalOutputTokens,
844
+ wallTimeMs: Date.now() - dispatchStartMs,
845
+ },
846
+ });
847
+ };
848
+
849
+ try {
712
850
  if ((agentScope === "project" || agentScope === "both") && confirmProjectAgents && ctx.hasUI) {
713
851
  const requestedAgentNames = new Set<string>();
714
852
  if (params.chain) for (const step of params.chain) requestedAgentNames.add(step.agent);
@@ -726,16 +864,19 @@ export default function (pi: ExtensionAPI) {
726
864
  "Run project-local agents?",
727
865
  `Agents: ${names}\nSource: ${dir}\n\nProject agents are repo-controlled. Only continue for trusted repositories.`,
728
866
  );
729
- if (!ok)
867
+ if (!ok) {
868
+ finishDispatch([]);
730
869
  return {
731
870
  content: [{ type: "text", text: "Canceled: project-local agents not approved." }],
732
871
  details: makeDetails(hasChain ? "chain" : hasTasks ? "parallel" : "single")([]),
733
872
  };
873
+ }
734
874
  }
735
875
  }
736
876
 
737
877
  if (params.chain && params.chain.length > 0) {
738
878
  const results: SingleResult[] = [];
879
+ finalResults = results;
739
880
  let previousOutput = "";
740
881
 
741
882
  for (let i = 0; i < params.chain.length; i++) {
@@ -775,6 +916,7 @@ export default function (pi: ExtensionAPI) {
775
916
  if (isError) {
776
917
  const errorMsg =
777
918
  result.errorMessage || result.stderr || getFinalOutput(result.messages) || "(no output)";
919
+ finishDispatch(results);
778
920
  return {
779
921
  content: [{ type: "text", text: `Chain stopped at step ${i + 1} (${step.agent}): ${errorMsg}` }],
780
922
  details: makeDetails("chain")(results),
@@ -783,6 +925,7 @@ export default function (pi: ExtensionAPI) {
783
925
  }
784
926
  previousOutput = getFinalOutput(result.messages);
785
927
  }
928
+ finishDispatch(results);
786
929
  return {
787
930
  content: [{ type: "text", text: getFinalOutput(results[results.length - 1].messages) || "(no output)" }],
788
931
  details: makeDetails("chain")(results),
@@ -790,7 +933,8 @@ export default function (pi: ExtensionAPI) {
790
933
  }
791
934
 
792
935
  if (params.tasks && params.tasks.length > 0) {
793
- if (params.tasks.length > MAX_PARALLEL_TASKS)
936
+ if (params.tasks.length > MAX_PARALLEL_TASKS) {
937
+ finishDispatch([]);
794
938
  return {
795
939
  content: [
796
940
  {
@@ -800,6 +944,7 @@ export default function (pi: ExtensionAPI) {
800
944
  ],
801
945
  details: makeDetails("parallel")([]),
802
946
  };
947
+ }
803
948
 
804
949
  // Track all results for streaming updates
805
950
  const allResults: SingleResult[] = new Array(params.tasks.length);
@@ -816,6 +961,7 @@ export default function (pi: ExtensionAPI) {
816
961
  usage: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, cost: 0, contextTokens: 0, turns: 0 },
817
962
  };
818
963
  }
964
+ finalResults = allResults;
819
965
 
820
966
  const emitParallelUpdate = () => {
821
967
  if (onUpdate) {
@@ -887,6 +1033,7 @@ export default function (pi: ExtensionAPI) {
887
1033
  emitParallelUpdate();
888
1034
  return result;
889
1035
  });
1036
+ finalResults = results;
890
1037
 
891
1038
  const successCount = results.filter((r) => r.exitCode === 0).length;
892
1039
  const summaries = results.map((r) => {
@@ -896,6 +1043,7 @@ export default function (pi: ExtensionAPI) {
896
1043
  : getFinalOutput(r.messages);
897
1044
  return `[${r.agent}] ${r.exitCode === 0 ? "completed" : `failed (exit ${r.exitCode})`}: ${output || "(no output)"}`;
898
1045
  });
1046
+ finishDispatch(results);
899
1047
  return {
900
1048
  content: [
901
1049
  {
@@ -943,6 +1091,7 @@ export default function (pi: ExtensionAPI) {
943
1091
  onUpdate,
944
1092
  makeDetails("single"),
945
1093
  );
1094
+ finalResults = [result];
946
1095
 
947
1096
  // Capture and merge delta if isolated
948
1097
  if (isolation) {
@@ -956,6 +1105,7 @@ export default function (pi: ExtensionAPI) {
956
1105
  if (isError) {
957
1106
  const errorMsg =
958
1107
  result.errorMessage || result.stderr || getFinalOutput(result.messages) || "(no output)";
1108
+ finishDispatch([result]);
959
1109
  return {
960
1110
  content: [{ type: "text", text: `Agent ${result.stopReason || "failed"}: ${errorMsg}` }],
961
1111
  details: makeDetails("single")([result]),
@@ -967,6 +1117,7 @@ export default function (pi: ExtensionAPI) {
967
1117
  if (mergeResult && !mergeResult.success) {
968
1118
  outputText += `\n\n⚠ Patch merge failed: ${mergeResult.error || "unknown error"}`;
969
1119
  }
1120
+ finishDispatch([result]);
970
1121
  return {
971
1122
  content: [{ type: "text", text: outputText }],
972
1123
  details: makeDetails("single")([result]),
@@ -978,11 +1129,18 @@ export default function (pi: ExtensionAPI) {
978
1129
  }
979
1130
  }
980
1131
 
1132
+ finishDispatch([]);
981
1133
  const available = agents.map((a) => `${a.name} (${a.source})`).join(", ") || "none";
982
1134
  return {
983
1135
  content: [{ type: "text", text: `Invalid parameters. Available agents: ${available}` }],
984
1136
  details: makeDetails("single")([]),
985
1137
  };
1138
+ } catch (err) {
1139
+ if (!dispatchCompletedEmitted) finalResults = synthesizeFailureResults(err);
1140
+ throw err;
1141
+ } finally {
1142
+ finishDispatch(finalResults);
1143
+ }
986
1144
  },
987
1145
 
988
1146
  renderCall(args, theme) {
@@ -19,8 +19,8 @@ Note: `~/.gsd/agent/extensions/` is reserved for bundled extensions synced from
19
19
  3. **Commands** — Give users slash commands (`pi.registerCommand()`). Users type `/mycommand`.
20
20
 
21
21
  **Non-negotiable rules:**
22
- - Use `StringEnum` from `@mariozechner/pi-ai` for string enum params (NOT `Type.Union`/`Type.Literal` — breaks Google's API)
23
- - Truncate tool output to 50KB / 2000 lines max (use `truncateHead`/`truncateTail` from `@mariozechner/pi-coding-agent`)
22
+ - Use `StringEnum` from `@gsd/pi-ai` for string enum params (NOT `Type.Union`/`Type.Literal` — breaks Google's API)
23
+ - Truncate tool output to 50KB / 2000 lines max (use `truncateHead`/`truncateTail` from `@gsd/pi-coding-agent`)
24
24
  - Store stateful tool state in `details` for branching support
25
25
  - Check `signal?.aborted` in long-running tool executions
26
26
  - Use `pi.exec()` not `child_process` for shell commands
@@ -34,10 +34,10 @@ Note: `~/.gsd/agent/extensions/` is reserved for bundled extensions synced from
34
34
 
35
35
  | Package | Purpose |
36
36
  |---------|---------|
37
- | `@mariozechner/pi-coding-agent` | `ExtensionAPI`, `ExtensionContext`, `Theme`, event types, tool utilities, `DynamicBorder`, `BorderedLoader`, `CustomEditor`, `highlightCode` |
37
+ | `@gsd/pi-coding-agent` | `ExtensionAPI`, `ExtensionContext`, `Theme`, event types, tool utilities, `DynamicBorder`, `BorderedLoader`, `CustomEditor`, `highlightCode` |
38
38
  | `@sinclair/typebox` | `Type.Object`, `Type.String`, `Type.Number`, `Type.Optional`, `Type.Boolean`, `Type.Array` |
39
- | `@mariozechner/pi-ai` | `StringEnum` (required for string enums), `Type` re-export |
40
- | `@mariozechner/pi-tui` | `Text`, `Box`, `Container`, `Spacer`, `Markdown`, `SelectList`, `Input`, `matchesKey`, `Key`, `truncateToWidth`, `visibleWidth` |
39
+ | `@gsd/pi-ai` | `StringEnum` (required for string enums), `Type` re-export |
40
+ | `@gsd/pi-tui` | `Text`, `Box`, `Container`, `Spacer`, `Markdown`, `SelectList`, `Input`, `matchesKey`, `Key`, `truncateToWidth`, `visibleWidth` |
41
41
  | Node.js built-ins | `node:fs`, `node:path`, `node:child_process`, etc. |
42
42
 
43
43
  </essential_principles>
@@ -65,6 +65,9 @@ All domain knowledge in `references/`:
65
65
  **Capabilities:** custom-tools.md, custom-commands.md, custom-ui.md, custom-rendering.md
66
66
  **Patterns:** state-management.md, system-prompt-modification.md, compaction-session-control.md
67
67
  **Infrastructure:** model-provider-management.md, remote-execution-overrides.md, packaging-distribution.md, mode-behavior.md
68
+ **Spec:** `docs/extension-sdk/manifest-spec.md` — manifest format, tiers, validation
69
+ **Testing:** `docs/extension-sdk/testing.md` — mock patterns, test conventions
70
+ **SDK:** `docs/extension-sdk/` — the authoritative GSD-2 extension guide
68
71
  **Gotchas:** key-rules-gotchas.md
69
72
  </reference_index>
70
73
 
@@ -78,6 +81,7 @@ All domain knowledge in `references/`:
78
81
 
79
82
  <success_criteria>
80
83
  Extension is complete when:
84
+ - `extension-manifest.json` exists with accurate `provides` listing all registered tools/commands/hooks/shortcuts
81
85
  - TypeScript compiles without errors (jiti handles this at runtime)
82
86
  - Extension loads on GSD startup or `/reload` without errors
83
87
  - Tools appear in the LLM's system prompt and are callable