gsd-pi 2.28.0-dev.e19bf89 → 2.29.0-dev.23d50d0

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 (285) hide show
  1. package/README.md +24 -17
  2. package/dist/cli.js +15 -9
  3. package/dist/headless.js +4 -0
  4. package/dist/resource-loader.js +80 -8
  5. package/dist/resources/extensions/bg-shell/process-manager.ts +13 -0
  6. package/dist/resources/extensions/gsd/auto-dashboard.ts +217 -65
  7. package/dist/resources/extensions/gsd/auto-dispatch.ts +2 -2
  8. package/dist/resources/extensions/gsd/auto-post-unit.ts +53 -6
  9. package/dist/resources/extensions/gsd/auto-prompts.ts +27 -14
  10. package/dist/resources/extensions/gsd/auto-recovery.ts +33 -23
  11. package/dist/resources/extensions/gsd/auto-start.ts +25 -10
  12. package/dist/resources/extensions/gsd/auto-verification.ts +41 -7
  13. package/dist/resources/extensions/gsd/auto-worktree-sync.ts +21 -6
  14. package/dist/resources/extensions/gsd/auto-worktree.ts +9 -0
  15. package/dist/resources/extensions/gsd/auto.ts +67 -22
  16. package/dist/resources/extensions/gsd/commands-handlers.ts +3 -11
  17. package/dist/resources/extensions/gsd/commands-logs.ts +536 -0
  18. package/dist/resources/extensions/gsd/commands-prefs-wizard.ts +90 -47
  19. package/dist/resources/extensions/gsd/commands-workflow-templates.ts +544 -0
  20. package/dist/resources/extensions/gsd/commands.ts +75 -29
  21. package/dist/resources/extensions/gsd/dashboard-overlay.ts +2 -1
  22. package/dist/resources/extensions/gsd/doctor-types.ts +13 -0
  23. package/dist/resources/extensions/gsd/doctor.ts +2 -6
  24. package/dist/resources/extensions/gsd/export.ts +28 -2
  25. package/dist/resources/extensions/gsd/gsd-db.ts +19 -0
  26. package/dist/resources/extensions/gsd/index.ts +2 -1
  27. package/dist/resources/extensions/gsd/json-persistence.ts +67 -0
  28. package/dist/resources/extensions/gsd/mechanical-completion.ts +430 -0
  29. package/dist/resources/extensions/gsd/metrics.ts +17 -31
  30. package/dist/resources/extensions/gsd/paths.ts +17 -8
  31. package/dist/resources/extensions/gsd/preferences-models.ts +7 -1
  32. package/dist/resources/extensions/gsd/preferences-validation.ts +2 -1
  33. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +4 -2
  34. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
  35. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +26 -2
  36. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -1
  37. package/dist/resources/extensions/gsd/prompts/workflow-start.md +28 -0
  38. package/dist/resources/extensions/gsd/queue-order.ts +10 -11
  39. package/dist/resources/extensions/gsd/routing-history.ts +13 -17
  40. package/dist/resources/extensions/gsd/session-lock.ts +284 -0
  41. package/dist/resources/extensions/gsd/session-status-io.ts +23 -41
  42. package/dist/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +1 -1
  43. package/dist/resources/extensions/gsd/tests/auto-skip-loop.test.ts +1 -1
  44. package/dist/resources/extensions/gsd/tests/commands-logs.test.ts +241 -0
  45. package/dist/resources/extensions/gsd/tests/extension-selector-separator.test.ts +60 -38
  46. package/dist/resources/extensions/gsd/tests/gsd-inspect.test.ts +1 -1
  47. package/dist/resources/extensions/gsd/tests/mechanical-completion.test.ts +356 -0
  48. package/dist/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +1 -1
  49. package/dist/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +1 -0
  50. package/dist/resources/extensions/gsd/tests/session-lock.test.ts +315 -0
  51. package/dist/resources/extensions/gsd/tests/token-profile.test.ts +14 -16
  52. package/dist/resources/extensions/gsd/tests/validate-milestone.test.ts +55 -0
  53. package/dist/resources/extensions/gsd/tests/verification-evidence.test.ts +26 -24
  54. package/dist/resources/extensions/gsd/tests/verification-gate.test.ts +136 -7
  55. package/dist/resources/extensions/gsd/tests/workflow-templates.test.ts +173 -0
  56. package/dist/resources/extensions/gsd/types.ts +3 -0
  57. package/dist/resources/extensions/gsd/unit-runtime.ts +16 -13
  58. package/dist/resources/extensions/gsd/verification-evidence.ts +2 -0
  59. package/dist/resources/extensions/gsd/verification-gate.ts +13 -2
  60. package/dist/resources/extensions/gsd/workflow-templates/bugfix.md +87 -0
  61. package/dist/resources/extensions/gsd/workflow-templates/dep-upgrade.md +74 -0
  62. package/dist/resources/extensions/gsd/workflow-templates/full-project.md +41 -0
  63. package/dist/resources/extensions/gsd/workflow-templates/hotfix.md +45 -0
  64. package/dist/resources/extensions/gsd/workflow-templates/refactor.md +83 -0
  65. package/dist/resources/extensions/gsd/workflow-templates/registry.json +85 -0
  66. package/dist/resources/extensions/gsd/workflow-templates/security-audit.md +73 -0
  67. package/dist/resources/extensions/gsd/workflow-templates/small-feature.md +81 -0
  68. package/dist/resources/extensions/gsd/workflow-templates/spike.md +69 -0
  69. package/dist/resources/extensions/gsd/workflow-templates.ts +241 -0
  70. package/dist/resources/extensions/mcp-client/index.ts +459 -0
  71. package/dist/resources/extensions/remote-questions/discord-adapter.ts +9 -20
  72. package/dist/resources/extensions/remote-questions/http-client.ts +76 -0
  73. package/dist/resources/extensions/remote-questions/notify.ts +1 -2
  74. package/dist/resources/extensions/remote-questions/slack-adapter.ts +11 -18
  75. package/dist/resources/extensions/remote-questions/telegram-adapter.ts +8 -20
  76. package/dist/resources/extensions/remote-questions/types.ts +3 -0
  77. package/dist/resources/extensions/shared/mod.ts +3 -0
  78. package/dist/resources/skills/create-gsd-extension/SKILL.md +87 -0
  79. package/dist/resources/skills/create-gsd-extension/references/compaction-session-control.md +77 -0
  80. package/dist/resources/skills/create-gsd-extension/references/custom-commands.md +139 -0
  81. package/dist/resources/skills/create-gsd-extension/references/custom-rendering.md +108 -0
  82. package/dist/resources/skills/create-gsd-extension/references/custom-tools.md +183 -0
  83. package/dist/resources/skills/create-gsd-extension/references/custom-ui.md +490 -0
  84. package/dist/resources/skills/create-gsd-extension/references/events-reference.md +126 -0
  85. package/dist/resources/skills/create-gsd-extension/references/extension-lifecycle.md +64 -0
  86. package/dist/resources/skills/create-gsd-extension/references/extensionapi-reference.md +75 -0
  87. package/dist/resources/skills/create-gsd-extension/references/extensioncontext-reference.md +53 -0
  88. package/dist/resources/skills/create-gsd-extension/references/key-rules-gotchas.md +36 -0
  89. package/dist/resources/skills/create-gsd-extension/references/mode-behavior.md +32 -0
  90. package/dist/resources/skills/create-gsd-extension/references/model-provider-management.md +89 -0
  91. package/dist/resources/skills/create-gsd-extension/references/packaging-distribution.md +55 -0
  92. package/dist/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +90 -0
  93. package/dist/resources/skills/create-gsd-extension/references/state-management.md +70 -0
  94. package/dist/resources/skills/create-gsd-extension/references/system-prompt-modification.md +52 -0
  95. package/dist/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +51 -0
  96. package/dist/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +143 -0
  97. package/dist/resources/skills/create-gsd-extension/workflows/add-capability.md +57 -0
  98. package/dist/resources/skills/create-gsd-extension/workflows/create-extension.md +156 -0
  99. package/dist/resources/skills/create-gsd-extension/workflows/debug-extension.md +74 -0
  100. package/dist/resources/skills/create-skill/SKILL.md +184 -0
  101. package/dist/resources/skills/create-skill/references/api-security.md +226 -0
  102. package/dist/resources/skills/create-skill/references/be-clear-and-direct.md +531 -0
  103. package/dist/resources/skills/create-skill/references/common-patterns.md +595 -0
  104. package/dist/resources/skills/create-skill/references/core-principles.md +437 -0
  105. package/dist/resources/skills/create-skill/references/executable-code.md +175 -0
  106. package/dist/resources/skills/create-skill/references/gsd-skill-ecosystem.md +68 -0
  107. package/dist/resources/skills/create-skill/references/iteration-and-testing.md +474 -0
  108. package/dist/resources/skills/create-skill/references/recommended-structure.md +168 -0
  109. package/dist/resources/skills/create-skill/references/skill-structure.md +372 -0
  110. package/dist/resources/skills/create-skill/references/use-xml-tags.md +466 -0
  111. package/dist/resources/skills/create-skill/references/using-scripts.md +113 -0
  112. package/dist/resources/skills/create-skill/references/using-templates.md +112 -0
  113. package/dist/resources/skills/create-skill/references/workflows-and-validation.md +510 -0
  114. package/dist/resources/skills/create-skill/templates/router-skill.md +73 -0
  115. package/dist/resources/skills/create-skill/templates/simple-skill.md +33 -0
  116. package/dist/resources/skills/create-skill/workflows/add-reference.md +96 -0
  117. package/dist/resources/skills/create-skill/workflows/add-script.md +93 -0
  118. package/dist/resources/skills/create-skill/workflows/add-template.md +74 -0
  119. package/dist/resources/skills/create-skill/workflows/add-workflow.md +120 -0
  120. package/dist/resources/skills/create-skill/workflows/audit-skill.md +148 -0
  121. package/dist/resources/skills/create-skill/workflows/create-new-skill.md +196 -0
  122. package/dist/resources/skills/create-skill/workflows/get-guidance.md +121 -0
  123. package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +161 -0
  124. package/dist/resources/skills/create-skill/workflows/verify-skill.md +204 -0
  125. package/package.json +6 -3
  126. package/packages/native/dist/native.d.ts +2 -0
  127. package/packages/native/dist/native.js +19 -5
  128. package/packages/native/src/native.ts +23 -9
  129. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  130. package/packages/pi-coding-agent/dist/core/extensions/loader.js +13 -0
  131. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  132. package/packages/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -1
  133. package/packages/pi-coding-agent/dist/core/lsp/client.js +3 -0
  134. package/packages/pi-coding-agent/dist/core/lsp/client.js.map +1 -1
  135. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +3 -0
  136. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  137. package/packages/pi-coding-agent/dist/core/settings-manager.js +8 -0
  138. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  139. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  140. package/packages/pi-coding-agent/dist/core/system-prompt.js +10 -0
  141. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  142. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  143. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +4 -1
  144. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  145. package/packages/pi-coding-agent/package.json +1 -1
  146. package/packages/pi-coding-agent/scripts/copy-assets.cjs +39 -8
  147. package/packages/pi-coding-agent/src/core/extensions/loader.ts +13 -0
  148. package/packages/pi-coding-agent/src/core/lsp/client.ts +3 -0
  149. package/packages/pi-coding-agent/src/core/settings-manager.ts +11 -0
  150. package/packages/pi-coding-agent/src/core/system-prompt.ts +11 -0
  151. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +4 -1
  152. package/packages/pi-tui/dist/autocomplete.d.ts +3 -0
  153. package/packages/pi-tui/dist/autocomplete.d.ts.map +1 -1
  154. package/packages/pi-tui/dist/autocomplete.js +14 -0
  155. package/packages/pi-tui/dist/autocomplete.js.map +1 -1
  156. package/packages/pi-tui/src/autocomplete.ts +19 -1
  157. package/pkg/package.json +1 -1
  158. package/src/resources/extensions/bg-shell/process-manager.ts +13 -0
  159. package/src/resources/extensions/gsd/auto-dashboard.ts +217 -65
  160. package/src/resources/extensions/gsd/auto-dispatch.ts +2 -2
  161. package/src/resources/extensions/gsd/auto-post-unit.ts +53 -6
  162. package/src/resources/extensions/gsd/auto-prompts.ts +27 -14
  163. package/src/resources/extensions/gsd/auto-recovery.ts +33 -23
  164. package/src/resources/extensions/gsd/auto-start.ts +25 -10
  165. package/src/resources/extensions/gsd/auto-verification.ts +41 -7
  166. package/src/resources/extensions/gsd/auto-worktree-sync.ts +21 -6
  167. package/src/resources/extensions/gsd/auto-worktree.ts +9 -0
  168. package/src/resources/extensions/gsd/auto.ts +67 -22
  169. package/src/resources/extensions/gsd/commands-handlers.ts +3 -11
  170. package/src/resources/extensions/gsd/commands-logs.ts +536 -0
  171. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +90 -47
  172. package/src/resources/extensions/gsd/commands-workflow-templates.ts +544 -0
  173. package/src/resources/extensions/gsd/commands.ts +75 -29
  174. package/src/resources/extensions/gsd/dashboard-overlay.ts +2 -1
  175. package/src/resources/extensions/gsd/doctor-types.ts +13 -0
  176. package/src/resources/extensions/gsd/doctor.ts +2 -6
  177. package/src/resources/extensions/gsd/export.ts +28 -2
  178. package/src/resources/extensions/gsd/gsd-db.ts +19 -0
  179. package/src/resources/extensions/gsd/index.ts +2 -1
  180. package/src/resources/extensions/gsd/json-persistence.ts +67 -0
  181. package/src/resources/extensions/gsd/mechanical-completion.ts +430 -0
  182. package/src/resources/extensions/gsd/metrics.ts +17 -31
  183. package/src/resources/extensions/gsd/paths.ts +17 -8
  184. package/src/resources/extensions/gsd/preferences-models.ts +7 -1
  185. package/src/resources/extensions/gsd/preferences-validation.ts +2 -1
  186. package/src/resources/extensions/gsd/prompts/discuss-headless.md +4 -2
  187. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
  188. package/src/resources/extensions/gsd/prompts/plan-milestone.md +26 -2
  189. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -1
  190. package/src/resources/extensions/gsd/prompts/workflow-start.md +28 -0
  191. package/src/resources/extensions/gsd/queue-order.ts +10 -11
  192. package/src/resources/extensions/gsd/routing-history.ts +13 -17
  193. package/src/resources/extensions/gsd/session-lock.ts +284 -0
  194. package/src/resources/extensions/gsd/session-status-io.ts +23 -41
  195. package/src/resources/extensions/gsd/tests/auto-budget-alerts.test.ts +1 -1
  196. package/src/resources/extensions/gsd/tests/auto-skip-loop.test.ts +1 -1
  197. package/src/resources/extensions/gsd/tests/commands-logs.test.ts +241 -0
  198. package/src/resources/extensions/gsd/tests/extension-selector-separator.test.ts +60 -38
  199. package/src/resources/extensions/gsd/tests/gsd-inspect.test.ts +1 -1
  200. package/src/resources/extensions/gsd/tests/mechanical-completion.test.ts +356 -0
  201. package/src/resources/extensions/gsd/tests/parallel-workers-multi-milestone-e2e.test.ts +1 -1
  202. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +1 -0
  203. package/src/resources/extensions/gsd/tests/session-lock.test.ts +315 -0
  204. package/src/resources/extensions/gsd/tests/token-profile.test.ts +14 -16
  205. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +55 -0
  206. package/src/resources/extensions/gsd/tests/verification-evidence.test.ts +26 -24
  207. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +136 -7
  208. package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +173 -0
  209. package/src/resources/extensions/gsd/types.ts +3 -0
  210. package/src/resources/extensions/gsd/unit-runtime.ts +16 -13
  211. package/src/resources/extensions/gsd/verification-evidence.ts +2 -0
  212. package/src/resources/extensions/gsd/verification-gate.ts +13 -2
  213. package/src/resources/extensions/gsd/workflow-templates/bugfix.md +87 -0
  214. package/src/resources/extensions/gsd/workflow-templates/dep-upgrade.md +74 -0
  215. package/src/resources/extensions/gsd/workflow-templates/full-project.md +41 -0
  216. package/src/resources/extensions/gsd/workflow-templates/hotfix.md +45 -0
  217. package/src/resources/extensions/gsd/workflow-templates/refactor.md +83 -0
  218. package/src/resources/extensions/gsd/workflow-templates/registry.json +85 -0
  219. package/src/resources/extensions/gsd/workflow-templates/security-audit.md +73 -0
  220. package/src/resources/extensions/gsd/workflow-templates/small-feature.md +81 -0
  221. package/src/resources/extensions/gsd/workflow-templates/spike.md +69 -0
  222. package/src/resources/extensions/gsd/workflow-templates.ts +241 -0
  223. package/src/resources/extensions/mcp-client/index.ts +459 -0
  224. package/src/resources/extensions/remote-questions/discord-adapter.ts +9 -20
  225. package/src/resources/extensions/remote-questions/http-client.ts +76 -0
  226. package/src/resources/extensions/remote-questions/notify.ts +1 -2
  227. package/src/resources/extensions/remote-questions/slack-adapter.ts +11 -18
  228. package/src/resources/extensions/remote-questions/telegram-adapter.ts +8 -20
  229. package/src/resources/extensions/remote-questions/types.ts +3 -0
  230. package/src/resources/extensions/shared/mod.ts +3 -0
  231. package/src/resources/skills/create-gsd-extension/SKILL.md +87 -0
  232. package/src/resources/skills/create-gsd-extension/references/compaction-session-control.md +77 -0
  233. package/src/resources/skills/create-gsd-extension/references/custom-commands.md +139 -0
  234. package/src/resources/skills/create-gsd-extension/references/custom-rendering.md +108 -0
  235. package/src/resources/skills/create-gsd-extension/references/custom-tools.md +183 -0
  236. package/src/resources/skills/create-gsd-extension/references/custom-ui.md +490 -0
  237. package/src/resources/skills/create-gsd-extension/references/events-reference.md +126 -0
  238. package/src/resources/skills/create-gsd-extension/references/extension-lifecycle.md +64 -0
  239. package/src/resources/skills/create-gsd-extension/references/extensionapi-reference.md +75 -0
  240. package/src/resources/skills/create-gsd-extension/references/extensioncontext-reference.md +53 -0
  241. package/src/resources/skills/create-gsd-extension/references/key-rules-gotchas.md +36 -0
  242. package/src/resources/skills/create-gsd-extension/references/mode-behavior.md +32 -0
  243. package/src/resources/skills/create-gsd-extension/references/model-provider-management.md +89 -0
  244. package/src/resources/skills/create-gsd-extension/references/packaging-distribution.md +55 -0
  245. package/src/resources/skills/create-gsd-extension/references/remote-execution-overrides.md +90 -0
  246. package/src/resources/skills/create-gsd-extension/references/state-management.md +70 -0
  247. package/src/resources/skills/create-gsd-extension/references/system-prompt-modification.md +52 -0
  248. package/src/resources/skills/create-gsd-extension/templates/extension-skeleton.ts +51 -0
  249. package/src/resources/skills/create-gsd-extension/templates/stateful-tool-skeleton.ts +143 -0
  250. package/src/resources/skills/create-gsd-extension/workflows/add-capability.md +57 -0
  251. package/src/resources/skills/create-gsd-extension/workflows/create-extension.md +156 -0
  252. package/src/resources/skills/create-gsd-extension/workflows/debug-extension.md +74 -0
  253. package/src/resources/skills/create-skill/SKILL.md +184 -0
  254. package/src/resources/skills/create-skill/references/api-security.md +226 -0
  255. package/src/resources/skills/create-skill/references/be-clear-and-direct.md +531 -0
  256. package/src/resources/skills/create-skill/references/common-patterns.md +595 -0
  257. package/src/resources/skills/create-skill/references/core-principles.md +437 -0
  258. package/src/resources/skills/create-skill/references/executable-code.md +175 -0
  259. package/src/resources/skills/create-skill/references/gsd-skill-ecosystem.md +68 -0
  260. package/src/resources/skills/create-skill/references/iteration-and-testing.md +474 -0
  261. package/src/resources/skills/create-skill/references/recommended-structure.md +168 -0
  262. package/src/resources/skills/create-skill/references/skill-structure.md +372 -0
  263. package/src/resources/skills/create-skill/references/use-xml-tags.md +466 -0
  264. package/src/resources/skills/create-skill/references/using-scripts.md +113 -0
  265. package/src/resources/skills/create-skill/references/using-templates.md +112 -0
  266. package/src/resources/skills/create-skill/references/workflows-and-validation.md +510 -0
  267. package/src/resources/skills/create-skill/templates/router-skill.md +73 -0
  268. package/src/resources/skills/create-skill/templates/simple-skill.md +33 -0
  269. package/src/resources/skills/create-skill/workflows/add-reference.md +96 -0
  270. package/src/resources/skills/create-skill/workflows/add-script.md +93 -0
  271. package/src/resources/skills/create-skill/workflows/add-template.md +74 -0
  272. package/src/resources/skills/create-skill/workflows/add-workflow.md +120 -0
  273. package/src/resources/skills/create-skill/workflows/audit-skill.md +148 -0
  274. package/src/resources/skills/create-skill/workflows/create-new-skill.md +196 -0
  275. package/src/resources/skills/create-skill/workflows/get-guidance.md +121 -0
  276. package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +161 -0
  277. package/src/resources/skills/create-skill/workflows/verify-skill.md +204 -0
  278. package/dist/resources/extensions/gsd/preferences-hooks.ts +0 -10
  279. package/dist/resources/extensions/mcporter/index.ts +0 -525
  280. package/dist/resources/extensions/shared/progress-widget.ts +0 -282
  281. package/dist/resources/extensions/shared/thinking-widget.ts +0 -107
  282. package/src/resources/extensions/gsd/preferences-hooks.ts +0 -10
  283. package/src/resources/extensions/mcporter/index.ts +0 -525
  284. package/src/resources/extensions/shared/progress-widget.ts +0 -282
  285. package/src/resources/extensions/shared/thinking-widget.ts +0 -107
@@ -2,6 +2,9 @@
2
2
  * Remote Questions — shared types
3
3
  */
4
4
 
5
+ /** Timeout applied to every outbound HTTP request across all channel adapters. */
6
+ export const PER_REQUEST_TIMEOUT_MS = 15_000;
7
+
5
8
  export type RemoteChannel = "slack" | "discord" | "telegram";
6
9
 
7
10
  export interface RemoteQuestionOption {
@@ -28,3 +28,6 @@ export { showInterviewRound } from "./interview-ui.js";
28
28
  export type { Question, QuestionOption, RoundResult } from "./interview-ui.js";
29
29
  export { showNextAction } from "./next-action-ui.js";
30
30
  export { showConfirm } from "./confirm-ui.js";
31
+ export { sanitizeError } from "./sanitize.js";
32
+ export { formatDateShort, truncateWithEllipsis } from "./format-utils.js";
33
+ export { splitFrontmatter, parseFrontmatterMap } from "./frontmatter.js";
@@ -0,0 +1,87 @@
1
+ ---
2
+ name: create-gsd-extension
3
+ description: Create, debug, and iterate on GSD extensions (TypeScript modules that add tools, commands, event hooks, custom UI, and providers to GSD). Use when asked to build an extension, add a tool the LLM can call, register a slash command, hook into GSD events, create custom TUI components, or modify GSD behavior. Triggers on "create extension", "build extension", "add a tool", "register command", "hook into gsd", "custom tool", "gsd plugin", "gsd extension".
4
+ ---
5
+
6
+ <essential_principles>
7
+
8
+ **Extensions are TypeScript modules** that hook into GSD's runtime (built on pi). They export a default function receiving `ExtensionAPI` and use it to subscribe to events, register tools/commands/shortcuts, and interact with the session.
9
+
10
+ **GSD extension paths:**
11
+ - Global extensions: `~/.gsd/agent/extensions/*.ts` or `~/.gsd/agent/extensions/*/index.ts`
12
+ - Project-local extensions: `.gsd/extensions/*.ts` or `.gsd/extensions/*/index.ts`
13
+
14
+ **The three primitives:**
15
+ 1. **Events** — Listen and react (`pi.on("event", handler)`). Can block tool calls, modify messages, inject context.
16
+ 2. **Tools** — Give the LLM new abilities (`pi.registerTool()`). LLM calls them autonomously.
17
+ 3. **Commands** — Give users slash commands (`pi.registerCommand()`). Users type `/mycommand`.
18
+
19
+ **Non-negotiable rules:**
20
+ - Use `StringEnum` from `@mariozechner/pi-ai` for string enum params (NOT `Type.Union`/`Type.Literal` — breaks Google's API)
21
+ - Truncate tool output to 50KB / 2000 lines max (use `truncateHead`/`truncateTail` from `@mariozechner/pi-coding-agent`)
22
+ - Store stateful tool state in `details` for branching support
23
+ - Check `signal?.aborted` in long-running tool executions
24
+ - Use `pi.exec()` not `child_process` for shell commands
25
+ - Check `ctx.hasUI` before dialog methods (non-interactive modes exist)
26
+ - Session control methods (`waitForIdle`, `newSession`, `fork`, `navigateTree`, `reload`) are ONLY available in command handlers — they deadlock in event handlers
27
+ - Lines from `render()` must not exceed `width` — use `truncateToWidth()`
28
+ - Use theme from callback params, never import directly
29
+ - Strip leading `@` from path params in custom tools (some models add it)
30
+
31
+ **Available imports:**
32
+
33
+ | Package | Purpose |
34
+ |---------|---------|
35
+ | `@mariozechner/pi-coding-agent` | `ExtensionAPI`, `ExtensionContext`, `Theme`, event types, tool utilities, `DynamicBorder`, `BorderedLoader`, `CustomEditor`, `highlightCode` |
36
+ | `@sinclair/typebox` | `Type.Object`, `Type.String`, `Type.Number`, `Type.Optional`, `Type.Boolean`, `Type.Array` |
37
+ | `@mariozechner/pi-ai` | `StringEnum` (required for string enums), `Type` re-export |
38
+ | `@mariozechner/pi-tui` | `Text`, `Box`, `Container`, `Spacer`, `Markdown`, `SelectList`, `Input`, `matchesKey`, `Key`, `truncateToWidth`, `visibleWidth` |
39
+ | Node.js built-ins | `node:fs`, `node:path`, `node:child_process`, etc. |
40
+
41
+ </essential_principles>
42
+
43
+ <routing>
44
+ Based on user intent, route to the appropriate workflow:
45
+
46
+ **Building a new extension:**
47
+ - "Create an extension", "build a tool", "I want to add a command" → `workflows/create-extension.md`
48
+
49
+ **Adding capabilities to an existing extension:**
50
+ - "Add a tool to my extension", "add event hook", "add custom rendering" → `workflows/add-capability.md`
51
+
52
+ **Debugging an extension:**
53
+ - "My extension doesn't work", "tool not showing up", "event not firing" → `workflows/debug-extension.md`
54
+
55
+ **If user intent is clear from context, skip the question and go directly to the workflow.**
56
+ </routing>
57
+
58
+ <reference_index>
59
+ All domain knowledge in `references/`:
60
+
61
+ **Core architecture:** extension-lifecycle.md, events-reference.md
62
+ **API surface:** extensionapi-reference.md, extensioncontext-reference.md
63
+ **Capabilities:** custom-tools.md, custom-commands.md, custom-ui.md, custom-rendering.md
64
+ **Patterns:** state-management.md, system-prompt-modification.md, compaction-session-control.md
65
+ **Infrastructure:** model-provider-management.md, remote-execution-overrides.md, packaging-distribution.md, mode-behavior.md
66
+ **Gotchas:** key-rules-gotchas.md
67
+ </reference_index>
68
+
69
+ <workflows_index>
70
+ | Workflow | Purpose |
71
+ |----------|---------|
72
+ | create-extension.md | Build a new extension from scratch |
73
+ | add-capability.md | Add tools, commands, hooks, UI to an existing extension |
74
+ | debug-extension.md | Diagnose and fix extension issues |
75
+ </workflows_index>
76
+
77
+ <success_criteria>
78
+ Extension is complete when:
79
+ - TypeScript compiles without errors (jiti handles this at runtime)
80
+ - Extension loads on GSD startup or `/reload` without errors
81
+ - Tools appear in the LLM's system prompt and are callable
82
+ - Commands respond to `/command` input
83
+ - Event hooks fire at the expected lifecycle points
84
+ - Custom UI renders correctly within terminal width
85
+ - State persists correctly across session restarts (if stateful)
86
+ - Output is truncated to safe limits (if tools produce variable output)
87
+ </success_criteria>
@@ -0,0 +1,77 @@
1
+ <overview>
2
+ Custom compaction hooks, triggering compaction, and session control methods available only in command handlers.
3
+ </overview>
4
+
5
+ <custom_compaction>
6
+ Override default compaction behavior:
7
+
8
+ ```typescript
9
+ pi.on("session_before_compact", async (event, ctx) => {
10
+ const { preparation, branchEntries, customInstructions, signal } = event;
11
+
12
+ // Option 1: Cancel
13
+ return { cancel: true };
14
+
15
+ // Option 2: Custom summary
16
+ return {
17
+ compaction: {
18
+ summary: "Custom summary of conversation so far...",
19
+ firstKeptEntryId: preparation.firstKeptEntryId,
20
+ tokensBefore: preparation.tokensBefore,
21
+ }
22
+ };
23
+ });
24
+ ```
25
+ </custom_compaction>
26
+
27
+ <trigger_compaction>
28
+ Trigger compaction programmatically from any handler:
29
+
30
+ ```typescript
31
+ ctx.compact({
32
+ customInstructions: "Focus on the authentication changes",
33
+ onComplete: (result) => ctx.ui.notify("Compacted!", "info"),
34
+ onError: (error) => ctx.ui.notify(`Failed: ${error.message}`, "error"),
35
+ });
36
+ ```
37
+ </trigger_compaction>
38
+
39
+ <session_control>
40
+ **Only available in command handlers** (deadlocks in event handlers):
41
+
42
+ ```typescript
43
+ pi.registerCommand("handoff", {
44
+ handler: async (args, ctx) => {
45
+ await ctx.waitForIdle();
46
+
47
+ // Create new session with initial context
48
+ const result = await ctx.newSession({
49
+ parentSession: ctx.sessionManager.getSessionFile(),
50
+ setup: async (sm) => {
51
+ sm.appendMessage({
52
+ role: "user",
53
+ content: [{ type: "text", text: `Context: ${args}` }],
54
+ timestamp: Date.now(),
55
+ });
56
+ },
57
+ });
58
+
59
+ if (result.cancelled) { /* extension cancelled via session_before_switch */ }
60
+ },
61
+ });
62
+ ```
63
+
64
+ | Method | Purpose |
65
+ |--------|---------|
66
+ | `ctx.waitForIdle()` | Wait for agent to finish streaming |
67
+ | `ctx.newSession(options?)` | Create a new session |
68
+ | `ctx.fork(entryId)` | Fork from a specific entry |
69
+ | `ctx.navigateTree(targetId, options?)` | Navigate session tree (with optional summary) |
70
+ | `ctx.reload()` | Hot-reload everything (treat as terminal — code after runs pre-reload version) |
71
+
72
+ `navigateTree` options:
73
+ - `summarize: boolean` — generate summary of abandoned branch
74
+ - `customInstructions: string` — instructions for summarizer
75
+ - `replaceInstructions: boolean` — replace default prompt entirely
76
+ - `label: string` — label to attach to branch summary
77
+ </session_control>
@@ -0,0 +1,139 @@
1
+ <overview>
2
+ Custom slash commands — registration, argument completions, subcommand patterns, and the extended command context.
3
+ </overview>
4
+
5
+ <basic_registration>
6
+ ```typescript
7
+ pi.registerCommand("deploy", {
8
+ description: "Deploy to an environment",
9
+ handler: async (args, ctx) => {
10
+ // args = everything after "/deploy "
11
+ // ctx = ExtensionCommandContext (has session control methods)
12
+ ctx.ui.notify(`Deploying to ${args || "production"}`, "info");
13
+ },
14
+ });
15
+ ```
16
+ </basic_registration>
17
+
18
+ <argument_completions>
19
+ Add tab-completion for command arguments:
20
+
21
+ ```typescript
22
+ import type { AutocompleteItem } from "@mariozechner/pi-tui";
23
+
24
+ pi.registerCommand("deploy", {
25
+ description: "Deploy to an environment",
26
+ getArgumentCompletions: (prefix: string): AutocompleteItem[] | null => {
27
+ const envs = ["dev", "staging", "prod"];
28
+ const items = envs.map(e => ({ value: e, label: e }));
29
+ const filtered = items.filter(i => i.value.startsWith(prefix));
30
+ return filtered.length > 0 ? filtered : null;
31
+ },
32
+ handler: async (args, ctx) => {
33
+ ctx.ui.notify(`Deploying to ${args}`, "info");
34
+ },
35
+ });
36
+ ```
37
+ </argument_completions>
38
+
39
+ <subcommand_pattern>
40
+ Fake nested commands via first-argument parsing. Used by `/wt new|ls|switch|merge|rm`.
41
+
42
+ ```typescript
43
+ pi.registerCommand("foo", {
44
+ description: "Manage foo items: /foo new|list|delete [name]",
45
+
46
+ getArgumentCompletions: (prefix: string) => {
47
+ const parts = prefix.trim().split(/\s+/);
48
+
49
+ // First arg: subcommand
50
+ if (parts.length <= 1) {
51
+ return ["new", "list", "delete"]
52
+ .filter(cmd => cmd.startsWith(parts[0] ?? ""))
53
+ .map(cmd => ({ value: cmd, label: cmd }));
54
+ }
55
+
56
+ // Second arg: depends on subcommand
57
+ if (parts[0] === "delete") {
58
+ const items = getItemsSomehow();
59
+ return items
60
+ .filter(name => name.startsWith(parts[1] ?? ""))
61
+ .map(name => ({ value: `delete ${name}`, label: name }));
62
+ }
63
+
64
+ return [];
65
+ },
66
+
67
+ handler: async (args, ctx) => {
68
+ const parts = args.trim().split(/\s+/);
69
+ const sub = parts[0];
70
+
71
+ switch (sub) {
72
+ case "new": /* ... */ return;
73
+ case "list": /* ... */ return;
74
+ case "delete": /* handle parts[1] */ return;
75
+ default:
76
+ ctx.ui.notify("Usage: /foo <new|list|delete> [name]", "info");
77
+ }
78
+ },
79
+ });
80
+ ```
81
+
82
+ **Gotcha:** `"".trim().split(/\s+/)` produces `['']`, not `[]`. That's why `parts.length <= 1` handles both empty and partial first arg.
83
+ </subcommand_pattern>
84
+
85
+ <command_context>
86
+ Command handlers get `ExtensionCommandContext` which extends `ExtensionContext` with session control methods:
87
+
88
+ | Method | Purpose |
89
+ |--------|---------|
90
+ | `ctx.waitForIdle()` | Wait for agent to finish streaming |
91
+ | `ctx.newSession(options?)` | Create a new session |
92
+ | `ctx.fork(entryId)` | Fork from an entry |
93
+ | `ctx.navigateTree(targetId, options?)` | Navigate session tree |
94
+ | `ctx.reload()` | Hot-reload everything |
95
+
96
+ **⚠️ These methods are ONLY available in command handlers.** Calling them from event handlers causes deadlocks.
97
+
98
+ ```typescript
99
+ pi.registerCommand("handoff", {
100
+ handler: async (args, ctx) => {
101
+ await ctx.waitForIdle();
102
+ await ctx.newSession({
103
+ setup: async (sm) => {
104
+ sm.appendMessage({
105
+ role: "user",
106
+ content: [{ type: "text", text: `Context: ${args}` }],
107
+ timestamp: Date.now(),
108
+ });
109
+ },
110
+ });
111
+ },
112
+ });
113
+ ```
114
+ </command_context>
115
+
116
+ <reload_pattern>
117
+ Expose reload as both a command and a tool the LLM can call:
118
+
119
+ ```typescript
120
+ pi.registerCommand("reload-runtime", {
121
+ description: "Reload extensions, skills, prompts, and themes",
122
+ handler: async (_args, ctx) => {
123
+ await ctx.reload();
124
+ return; // Treat reload as terminal
125
+ },
126
+ });
127
+
128
+ pi.registerTool({
129
+ name: "reload_runtime",
130
+ label: "Reload Runtime",
131
+ description: "Reload extensions, skills, prompts, and themes",
132
+ parameters: Type.Object({}),
133
+ async execute() {
134
+ pi.sendUserMessage("/reload-runtime", { deliverAs: "followUp" });
135
+ return { content: [{ type: "text", text: "Queued /reload-runtime as follow-up." }] };
136
+ },
137
+ });
138
+ ```
139
+ </reload_pattern>
@@ -0,0 +1,108 @@
1
+ <overview>
2
+ Custom rendering for tools and messages — control how they appear in the TUI.
3
+ </overview>
4
+
5
+ <tool_rendering>
6
+ Tools can provide `renderCall` (how the call looks) and `renderResult` (how the result looks):
7
+
8
+ ```typescript
9
+ import { Text } from "@mariozechner/pi-tui";
10
+ import { keyHint } from "@mariozechner/pi-coding-agent";
11
+
12
+ pi.registerTool({
13
+ name: "my_tool",
14
+ // ...
15
+
16
+ renderCall(args, theme) {
17
+ let text = theme.fg("toolTitle", theme.bold("my_tool "));
18
+ text += theme.fg("muted", args.action);
19
+ if (args.text) text += " " + theme.fg("dim", `"${args.text}"`);
20
+ return new Text(text, 0, 0); // 0,0 padding — Box handles it
21
+ },
22
+
23
+ renderResult(result, { expanded, isPartial }, theme) {
24
+ // isPartial = true during streaming (onUpdate was called)
25
+ if (isPartial) {
26
+ return new Text(theme.fg("warning", "Processing..."), 0, 0);
27
+ }
28
+
29
+ // expanded = user toggled expand (Ctrl+O)
30
+ if (result.details?.error) {
31
+ return new Text(theme.fg("error", `Error: ${result.details.error}`), 0, 0);
32
+ }
33
+
34
+ let text = theme.fg("success", "✓ Done");
35
+ if (!expanded) {
36
+ text += ` (${keyHint("expandTools", "to expand")})`;
37
+ }
38
+ if (expanded && result.details?.items) {
39
+ for (const item of result.details.items) {
40
+ text += "\n " + theme.fg("dim", item);
41
+ }
42
+ }
43
+ return new Text(text, 0, 0);
44
+ },
45
+ });
46
+ ```
47
+
48
+ If you omit `renderCall`/`renderResult`, the built-in renderer is used. Useful for tool overrides where you just wrap logic without reimplementing UI.
49
+
50
+ **Fallback:** If render methods throw, `renderCall` shows tool name, `renderResult` shows raw `content` text.
51
+ </tool_rendering>
52
+
53
+ <key_hints>
54
+ Key hint helpers for showing keybinding info in render output:
55
+
56
+ ```typescript
57
+ import { keyHint, appKeyHint, editorKey, rawKeyHint } from "@mariozechner/pi-coding-agent";
58
+
59
+ // Editor action hint (respects user keybinding config)
60
+ keyHint("expandTools", "to expand") // e.g., "Ctrl+O to expand"
61
+ keyHint("selectConfirm", "to select")
62
+
63
+ // Raw key hint (always shows literal key)
64
+ rawKeyHint("Ctrl+O", "to expand")
65
+ ```
66
+ </key_hints>
67
+
68
+ <message_rendering>
69
+ Register a renderer for custom message types:
70
+
71
+ ```typescript
72
+ import { Text } from "@mariozechner/pi-tui";
73
+
74
+ pi.registerMessageRenderer("my-extension", (message, options, theme) => {
75
+ const { expanded } = options;
76
+ let text = theme.fg("accent", `[${message.customType}] `) + message.content;
77
+ if (expanded && message.details) {
78
+ text += "\n" + theme.fg("dim", JSON.stringify(message.details, null, 2));
79
+ }
80
+ return new Text(text, 0, 0);
81
+ });
82
+
83
+ // Send messages that use this renderer:
84
+ pi.sendMessage({
85
+ customType: "my-extension", // Matches renderer name
86
+ content: "Status update",
87
+ display: true,
88
+ details: { foo: "bar" },
89
+ });
90
+ ```
91
+ </message_rendering>
92
+
93
+ <syntax_highlighting>
94
+ ```typescript
95
+ import { highlightCode, getLanguageFromPath } from "@mariozechner/pi-coding-agent";
96
+
97
+ const lang = getLanguageFromPath("/path/to/file.rs"); // "rust"
98
+ const highlighted = highlightCode(code, lang, theme);
99
+ ```
100
+ </syntax_highlighting>
101
+
102
+ <best_practices>
103
+ - Return `Text` with padding `(0, 0)` — the wrapping `Box` handles padding
104
+ - Support `expanded` for detail on demand
105
+ - Handle `isPartial` for streaming progress
106
+ - Keep collapsed view compact
107
+ - Use `\n` for multi-line content within a single `Text`
108
+ </best_practices>
@@ -0,0 +1,183 @@
1
+ <overview>
2
+ Complete custom tools reference — registration, parameters, execution, output truncation, overrides, rendering, and dynamic registration.
3
+ </overview>
4
+
5
+ <registration>
6
+ ```typescript
7
+ import { Type } from "@sinclair/typebox";
8
+ import { StringEnum } from "@mariozechner/pi-ai";
9
+
10
+ pi.registerTool({
11
+ name: "my_tool", // Unique identifier (snake_case)
12
+ label: "My Tool", // Display name in TUI
13
+ description: "What this does", // Full description shown to LLM
14
+
15
+ // Optional: one-liner for system prompt "Available tools" section
16
+ promptSnippet: "Manage project todo items",
17
+
18
+ // Optional: bullets added to system prompt "Guidelines" when tool is active
19
+ promptGuidelines: [
20
+ "Use my_tool for task management instead of file edits."
21
+ ],
22
+
23
+ // Parameter schema (MUST use TypeBox)
24
+ parameters: Type.Object({
25
+ action: StringEnum(["list", "add", "remove"] as const),
26
+ text: Type.Optional(Type.String({ description: "Item text" })),
27
+ id: Type.Optional(Type.Number({ description: "Item ID" })),
28
+ }),
29
+
30
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
31
+ // 1. Check cancellation
32
+ if (signal?.aborted) {
33
+ return { content: [{ type: "text", text: "Cancelled" }] };
34
+ }
35
+
36
+ // 2. Stream progress (optional)
37
+ onUpdate?.({
38
+ content: [{ type: "text", text: "Working..." }],
39
+ details: { progress: 50 },
40
+ });
41
+
42
+ // 3. Do the work
43
+ const result = await doWork(params);
44
+
45
+ // 4. Return result
46
+ return {
47
+ content: [{ type: "text", text: "Result text for LLM" }], // Sent to LLM context
48
+ details: { data: result }, // For rendering & state
49
+ };
50
+ },
51
+
52
+ // Optional: custom TUI rendering
53
+ renderCall(args, theme) { ... },
54
+ renderResult(result, { expanded, isPartial }, theme) { ... },
55
+ });
56
+ ```
57
+ </registration>
58
+
59
+ <critical_stringenum>
60
+ **⚠️ MUST use `StringEnum` for string enum parameters:**
61
+
62
+ ```typescript
63
+ import { StringEnum } from "@mariozechner/pi-ai";
64
+
65
+ // ✅ Correct — works with all providers including Google
66
+ action: StringEnum(["list", "add", "remove"] as const)
67
+
68
+ // ❌ BROKEN with Google's API
69
+ action: Type.Union([Type.Literal("list"), Type.Literal("add")])
70
+ ```
71
+ </critical_stringenum>
72
+
73
+ <output_truncation>
74
+ Tools MUST truncate output to avoid context overflow. Built-in limit: 50KB / 2000 lines.
75
+
76
+ ```typescript
77
+ import {
78
+ truncateHead, truncateTail, formatSize,
79
+ DEFAULT_MAX_BYTES, DEFAULT_MAX_LINES,
80
+ } from "@mariozechner/pi-coding-agent";
81
+
82
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
83
+ const output = await runCommand();
84
+ const truncation = truncateHead(output, {
85
+ maxLines: DEFAULT_MAX_LINES,
86
+ maxBytes: DEFAULT_MAX_BYTES,
87
+ });
88
+
89
+ let result = truncation.content;
90
+ if (truncation.truncated) {
91
+ const tempFile = writeTempFile(output);
92
+ result += `\n\n[Output truncated: ${truncation.outputLines}/${truncation.totalLines} lines`;
93
+ result += ` (${formatSize(truncation.outputBytes)}/${formatSize(truncation.totalBytes)}).`;
94
+ result += ` Full output: ${tempFile}]`;
95
+ }
96
+ return { content: [{ type: "text", text: result }] };
97
+ }
98
+ ```
99
+
100
+ Use `truncateHead` when beginning matters (search results, file reads). Use `truncateTail` when end matters (logs, command output).
101
+ </output_truncation>
102
+
103
+ <signaling_errors>
104
+ Throw to signal an error (sets `isError: true`). Returning a value never sets error flag.
105
+
106
+ ```typescript
107
+ async execute(toolCallId, params) {
108
+ if (!isValid(params.input)) {
109
+ throw new Error(`Invalid input: ${params.input}`);
110
+ }
111
+ return { content: [{ type: "text", text: "OK" }], details: {} };
112
+ }
113
+ ```
114
+ </signaling_errors>
115
+
116
+ <dynamic_registration>
117
+ Tools can be registered at any time — during load, in `session_start`, in command handlers. Available immediately without `/reload`.
118
+
119
+ ```typescript
120
+ pi.on("session_start", async (_event, ctx) => {
121
+ pi.registerTool({ name: "dynamic_tool", ... });
122
+ });
123
+ ```
124
+
125
+ Use `pi.setActiveTools(names)` to enable/disable tools at runtime.
126
+ </dynamic_registration>
127
+
128
+ <overriding_builtins>
129
+ Register a tool with the same name as a built-in (`read`, `bash`, `edit`, `write`, `grep`, `find`, `ls`) to override it. **Must match exact result shape including `details` type.**
130
+
131
+ ```typescript
132
+ import { createReadTool } from "@mariozechner/pi-coding-agent";
133
+
134
+ pi.registerTool({
135
+ name: "read",
136
+ label: "Read (Logged)",
137
+ description: "Read file contents with logging",
138
+ parameters: Type.Object({
139
+ path: Type.String(),
140
+ offset: Type.Optional(Type.Number()),
141
+ limit: Type.Optional(Type.Number()),
142
+ }),
143
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
144
+ console.log(`[AUDIT] Reading: ${params.path}`);
145
+ const builtIn = createReadTool(ctx.cwd);
146
+ return builtIn.execute(toolCallId, params, signal, onUpdate);
147
+ },
148
+ // Omit renderCall/renderResult to use built-in renderer
149
+ });
150
+ ```
151
+
152
+ Start with no built-in tools: `gsd --no-tools -e ./my-extension.ts`
153
+ </overriding_builtins>
154
+
155
+ <multiple_tools>
156
+ One extension can register multiple tools with shared state:
157
+
158
+ ```typescript
159
+ export default function (pi: ExtensionAPI) {
160
+ let connection = null;
161
+
162
+ pi.registerTool({ name: "db_connect", ... });
163
+ pi.registerTool({ name: "db_query", ... });
164
+ pi.registerTool({ name: "db_close", ... });
165
+
166
+ pi.on("session_shutdown", async () => {
167
+ connection?.close();
168
+ });
169
+ }
170
+ ```
171
+ </multiple_tools>
172
+
173
+ <path_normalization>
174
+ Some models add `@` prefix to path arguments. Strip it:
175
+
176
+ ```typescript
177
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
178
+ let path = params.path;
179
+ if (path.startsWith("@")) path = path.slice(1);
180
+ // ...
181
+ }
182
+ ```
183
+ </path_normalization>