@mrquake/quakecode-cli 0.64.0

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 (651) hide show
  1. package/CHANGELOG.md +3492 -0
  2. package/README.md +129 -0
  3. package/dist/bun/cli.d.ts +3 -0
  4. package/dist/bun/cli.d.ts.map +1 -0
  5. package/dist/bun/cli.js +7 -0
  6. package/dist/bun/cli.js.map +1 -0
  7. package/dist/bun/register-bedrock.d.ts +2 -0
  8. package/dist/bun/register-bedrock.d.ts.map +1 -0
  9. package/dist/bun/register-bedrock.js +4 -0
  10. package/dist/bun/register-bedrock.js.map +1 -0
  11. package/dist/cli/args.d.ts +49 -0
  12. package/dist/cli/args.d.ts.map +1 -0
  13. package/dist/cli/args.js +304 -0
  14. package/dist/cli/args.js.map +1 -0
  15. package/dist/cli/config-selector.d.ts +14 -0
  16. package/dist/cli/config-selector.d.ts.map +1 -0
  17. package/dist/cli/config-selector.js +31 -0
  18. package/dist/cli/config-selector.js.map +1 -0
  19. package/dist/cli/file-processor.d.ts +15 -0
  20. package/dist/cli/file-processor.d.ts.map +1 -0
  21. package/dist/cli/file-processor.js +83 -0
  22. package/dist/cli/file-processor.js.map +1 -0
  23. package/dist/cli/initial-message.d.ts +18 -0
  24. package/dist/cli/initial-message.d.ts.map +1 -0
  25. package/dist/cli/initial-message.js +22 -0
  26. package/dist/cli/initial-message.js.map +1 -0
  27. package/dist/cli/list-models.d.ts +9 -0
  28. package/dist/cli/list-models.d.ts.map +1 -0
  29. package/dist/cli/list-models.js +92 -0
  30. package/dist/cli/list-models.js.map +1 -0
  31. package/dist/cli/session-picker.d.ts +9 -0
  32. package/dist/cli/session-picker.d.ts.map +1 -0
  33. package/dist/cli/session-picker.js +35 -0
  34. package/dist/cli/session-picker.js.map +1 -0
  35. package/dist/cli.d.ts +3 -0
  36. package/dist/cli.d.ts.map +1 -0
  37. package/dist/cli.js +15 -0
  38. package/dist/cli.js.map +1 -0
  39. package/dist/config.d.ts +73 -0
  40. package/dist/config.d.ts.map +1 -0
  41. package/dist/config.js +212 -0
  42. package/dist/config.js.map +1 -0
  43. package/dist/core/agent-session-runtime.d.ts +136 -0
  44. package/dist/core/agent-session-runtime.d.ts.map +1 -0
  45. package/dist/core/agent-session-runtime.js +267 -0
  46. package/dist/core/agent-session-runtime.js.map +1 -0
  47. package/dist/core/agent-session.d.ts +585 -0
  48. package/dist/core/agent-session.d.ts.map +1 -0
  49. package/dist/core/agent-session.js +2497 -0
  50. package/dist/core/agent-session.js.map +1 -0
  51. package/dist/core/auth-storage.d.ts +132 -0
  52. package/dist/core/auth-storage.d.ts.map +1 -0
  53. package/dist/core/auth-storage.js +422 -0
  54. package/dist/core/auth-storage.js.map +1 -0
  55. package/dist/core/bash-executor.d.ts +46 -0
  56. package/dist/core/bash-executor.d.ts.map +1 -0
  57. package/dist/core/bash-executor.js +113 -0
  58. package/dist/core/bash-executor.js.map +1 -0
  59. package/dist/core/compaction/branch-summarization.d.ts +88 -0
  60. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  61. package/dist/core/compaction/branch-summarization.js +243 -0
  62. package/dist/core/compaction/branch-summarization.js.map +1 -0
  63. package/dist/core/compaction/compaction.d.ts +121 -0
  64. package/dist/core/compaction/compaction.d.ts.map +1 -0
  65. package/dist/core/compaction/compaction.js +613 -0
  66. package/dist/core/compaction/compaction.js.map +1 -0
  67. package/dist/core/compaction/index.d.ts +7 -0
  68. package/dist/core/compaction/index.d.ts.map +1 -0
  69. package/dist/core/compaction/index.js +7 -0
  70. package/dist/core/compaction/index.js.map +1 -0
  71. package/dist/core/compaction/utils.d.ts +38 -0
  72. package/dist/core/compaction/utils.d.ts.map +1 -0
  73. package/dist/core/compaction/utils.js +153 -0
  74. package/dist/core/compaction/utils.js.map +1 -0
  75. package/dist/core/defaults.d.ts +3 -0
  76. package/dist/core/defaults.d.ts.map +1 -0
  77. package/dist/core/defaults.js +2 -0
  78. package/dist/core/defaults.js.map +1 -0
  79. package/dist/core/diagnostics.d.ts +15 -0
  80. package/dist/core/diagnostics.d.ts.map +1 -0
  81. package/dist/core/diagnostics.js +2 -0
  82. package/dist/core/diagnostics.js.map +1 -0
  83. package/dist/core/event-bus.d.ts +9 -0
  84. package/dist/core/event-bus.d.ts.map +1 -0
  85. package/dist/core/event-bus.js +25 -0
  86. package/dist/core/event-bus.js.map +1 -0
  87. package/dist/core/exec.d.ts +29 -0
  88. package/dist/core/exec.d.ts.map +1 -0
  89. package/dist/core/exec.js +75 -0
  90. package/dist/core/exec.js.map +1 -0
  91. package/dist/core/export-html/ansi-to-html.d.ts +22 -0
  92. package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
  93. package/dist/core/export-html/ansi-to-html.js +249 -0
  94. package/dist/core/export-html/ansi-to-html.js.map +1 -0
  95. package/dist/core/export-html/index.d.ts +37 -0
  96. package/dist/core/export-html/index.d.ts.map +1 -0
  97. package/dist/core/export-html/index.js +224 -0
  98. package/dist/core/export-html/index.js.map +1 -0
  99. package/dist/core/export-html/template.css +1001 -0
  100. package/dist/core/export-html/template.html +55 -0
  101. package/dist/core/export-html/template.js +1690 -0
  102. package/dist/core/export-html/tool-renderer.d.ts +40 -0
  103. package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
  104. package/dist/core/export-html/tool-renderer.js +95 -0
  105. package/dist/core/export-html/tool-renderer.js.map +1 -0
  106. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  107. package/dist/core/export-html/vendor/marked.min.js +6 -0
  108. package/dist/core/extensions/index.d.ts +12 -0
  109. package/dist/core/extensions/index.d.ts.map +1 -0
  110. package/dist/core/extensions/index.js +9 -0
  111. package/dist/core/extensions/index.js.map +1 -0
  112. package/dist/core/extensions/loader.d.ts +25 -0
  113. package/dist/core/extensions/loader.d.ts.map +1 -0
  114. package/dist/core/extensions/loader.js +436 -0
  115. package/dist/core/extensions/loader.js.map +1 -0
  116. package/dist/core/extensions/runner.d.ts +148 -0
  117. package/dist/core/extensions/runner.d.ts.map +1 -0
  118. package/dist/core/extensions/runner.js +700 -0
  119. package/dist/core/extensions/runner.js.map +1 -0
  120. package/dist/core/extensions/types.d.ts +1085 -0
  121. package/dist/core/extensions/types.d.ts.map +1 -0
  122. package/dist/core/extensions/types.js +45 -0
  123. package/dist/core/extensions/types.js.map +1 -0
  124. package/dist/core/extensions/wrapper.d.ts +20 -0
  125. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  126. package/dist/core/extensions/wrapper.js +22 -0
  127. package/dist/core/extensions/wrapper.js.map +1 -0
  128. package/dist/core/footer-data-provider.d.ts +48 -0
  129. package/dist/core/footer-data-provider.d.ts.map +1 -0
  130. package/dist/core/footer-data-provider.js +314 -0
  131. package/dist/core/footer-data-provider.js.map +1 -0
  132. package/dist/core/index.d.ts +11 -0
  133. package/dist/core/index.d.ts.map +1 -0
  134. package/dist/core/index.js +11 -0
  135. package/dist/core/index.js.map +1 -0
  136. package/dist/core/keybindings.d.ts +285 -0
  137. package/dist/core/keybindings.d.ts.map +1 -0
  138. package/dist/core/keybindings.js +251 -0
  139. package/dist/core/keybindings.js.map +1 -0
  140. package/dist/core/messages.d.ts +77 -0
  141. package/dist/core/messages.d.ts.map +1 -0
  142. package/dist/core/messages.js +123 -0
  143. package/dist/core/messages.js.map +1 -0
  144. package/dist/core/model-registry.d.ts +132 -0
  145. package/dist/core/model-registry.d.ts.map +1 -0
  146. package/dist/core/model-registry.js +583 -0
  147. package/dist/core/model-registry.js.map +1 -0
  148. package/dist/core/model-resolver.d.ts +110 -0
  149. package/dist/core/model-resolver.d.ts.map +1 -0
  150. package/dist/core/model-resolver.js +486 -0
  151. package/dist/core/model-resolver.js.map +1 -0
  152. package/dist/core/output-guard.d.ts +6 -0
  153. package/dist/core/output-guard.d.ts.map +1 -0
  154. package/dist/core/output-guard.js +59 -0
  155. package/dist/core/output-guard.js.map +1 -0
  156. package/dist/core/package-manager.d.ts +172 -0
  157. package/dist/core/package-manager.d.ts.map +1 -0
  158. package/dist/core/package-manager.js +1783 -0
  159. package/dist/core/package-manager.js.map +1 -0
  160. package/dist/core/prompt-templates.d.ts +51 -0
  161. package/dist/core/prompt-templates.d.ts.map +1 -0
  162. package/dist/core/prompt-templates.js +249 -0
  163. package/dist/core/prompt-templates.js.map +1 -0
  164. package/dist/core/resolve-config-value.d.ts +23 -0
  165. package/dist/core/resolve-config-value.d.ts.map +1 -0
  166. package/dist/core/resolve-config-value.js +126 -0
  167. package/dist/core/resolve-config-value.js.map +1 -0
  168. package/dist/core/resource-loader.d.ts +185 -0
  169. package/dist/core/resource-loader.d.ts.map +1 -0
  170. package/dist/core/resource-loader.js +698 -0
  171. package/dist/core/resource-loader.js.map +1 -0
  172. package/dist/core/sdk.d.ts +93 -0
  173. package/dist/core/sdk.d.ts.map +1 -0
  174. package/dist/core/sdk.js +236 -0
  175. package/dist/core/sdk.js.map +1 -0
  176. package/dist/core/session-manager.d.ts +332 -0
  177. package/dist/core/session-manager.d.ts.map +1 -0
  178. package/dist/core/session-manager.js +1104 -0
  179. package/dist/core/session-manager.js.map +1 -0
  180. package/dist/core/settings-manager.d.ts +237 -0
  181. package/dist/core/settings-manager.d.ts.map +1 -0
  182. package/dist/core/settings-manager.js +702 -0
  183. package/dist/core/settings-manager.js.map +1 -0
  184. package/dist/core/skills.d.ts +60 -0
  185. package/dist/core/skills.d.ts.map +1 -0
  186. package/dist/core/skills.js +409 -0
  187. package/dist/core/skills.js.map +1 -0
  188. package/dist/core/slash-commands.d.ts +14 -0
  189. package/dist/core/slash-commands.d.ts.map +1 -0
  190. package/dist/core/slash-commands.js +23 -0
  191. package/dist/core/slash-commands.js.map +1 -0
  192. package/dist/core/source-info.d.ts +18 -0
  193. package/dist/core/source-info.d.ts.map +1 -0
  194. package/dist/core/source-info.js +19 -0
  195. package/dist/core/source-info.js.map +1 -0
  196. package/dist/core/system-prompt.d.ts +28 -0
  197. package/dist/core/system-prompt.d.ts.map +1 -0
  198. package/dist/core/system-prompt.js +116 -0
  199. package/dist/core/system-prompt.js.map +1 -0
  200. package/dist/core/timings.d.ts +8 -0
  201. package/dist/core/timings.d.ts.map +1 -0
  202. package/dist/core/timings.js +31 -0
  203. package/dist/core/timings.js.map +1 -0
  204. package/dist/core/tools/bash.d.ts +73 -0
  205. package/dist/core/tools/bash.d.ts.map +1 -0
  206. package/dist/core/tools/bash.js +342 -0
  207. package/dist/core/tools/bash.js.map +1 -0
  208. package/dist/core/tools/edit-diff.d.ts +85 -0
  209. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  210. package/dist/core/tools/edit-diff.js +337 -0
  211. package/dist/core/tools/edit-diff.js.map +1 -0
  212. package/dist/core/tools/edit.d.ts +53 -0
  213. package/dist/core/tools/edit.d.ts.map +1 -0
  214. package/dist/core/tools/edit.js +196 -0
  215. package/dist/core/tools/edit.js.map +1 -0
  216. package/dist/core/tools/file-mutation-queue.d.ts +6 -0
  217. package/dist/core/tools/file-mutation-queue.d.ts.map +1 -0
  218. package/dist/core/tools/file-mutation-queue.js +37 -0
  219. package/dist/core/tools/file-mutation-queue.js.map +1 -0
  220. package/dist/core/tools/find.d.ts +46 -0
  221. package/dist/core/tools/find.d.ts.map +1 -0
  222. package/dist/core/tools/find.js +258 -0
  223. package/dist/core/tools/find.js.map +1 -0
  224. package/dist/core/tools/grep.d.ts +56 -0
  225. package/dist/core/tools/grep.d.ts.map +1 -0
  226. package/dist/core/tools/grep.js +293 -0
  227. package/dist/core/tools/grep.js.map +1 -0
  228. package/dist/core/tools/index.d.ts +115 -0
  229. package/dist/core/tools/index.d.ts.map +1 -0
  230. package/dist/core/tools/index.js +86 -0
  231. package/dist/core/tools/index.js.map +1 -0
  232. package/dist/core/tools/ls.d.ts +46 -0
  233. package/dist/core/tools/ls.d.ts.map +1 -0
  234. package/dist/core/tools/ls.js +172 -0
  235. package/dist/core/tools/ls.js.map +1 -0
  236. package/dist/core/tools/path-utils.d.ts +8 -0
  237. package/dist/core/tools/path-utils.d.ts.map +1 -0
  238. package/dist/core/tools/path-utils.js +81 -0
  239. package/dist/core/tools/path-utils.js.map +1 -0
  240. package/dist/core/tools/read.d.ts +46 -0
  241. package/dist/core/tools/read.d.ts.map +1 -0
  242. package/dist/core/tools/read.js +225 -0
  243. package/dist/core/tools/read.js.map +1 -0
  244. package/dist/core/tools/render-utils.d.ts +21 -0
  245. package/dist/core/tools/render-utils.d.ts.map +1 -0
  246. package/dist/core/tools/render-utils.js +49 -0
  247. package/dist/core/tools/render-utils.js.map +1 -0
  248. package/dist/core/tools/tool-definition-wrapper.d.ts +14 -0
  249. package/dist/core/tools/tool-definition-wrapper.d.ts.map +1 -0
  250. package/dist/core/tools/tool-definition-wrapper.js +32 -0
  251. package/dist/core/tools/tool-definition-wrapper.js.map +1 -0
  252. package/dist/core/tools/truncate.d.ts +70 -0
  253. package/dist/core/tools/truncate.d.ts.map +1 -0
  254. package/dist/core/tools/truncate.js +205 -0
  255. package/dist/core/tools/truncate.js.map +1 -0
  256. package/dist/core/tools/write.d.ts +35 -0
  257. package/dist/core/tools/write.d.ts.map +1 -0
  258. package/dist/core/tools/write.js +216 -0
  259. package/dist/core/tools/write.js.map +1 -0
  260. package/dist/index.d.ts +28 -0
  261. package/dist/index.d.ts.map +1 -0
  262. package/dist/index.js +43 -0
  263. package/dist/index.js.map +1 -0
  264. package/dist/main.d.ts +8 -0
  265. package/dist/main.d.ts.map +1 -0
  266. package/dist/main.js +807 -0
  267. package/dist/main.js.map +1 -0
  268. package/dist/migrations.d.ts +33 -0
  269. package/dist/migrations.d.ts.map +1 -0
  270. package/dist/migrations.js +261 -0
  271. package/dist/migrations.js.map +1 -0
  272. package/dist/modes/index.d.ts +9 -0
  273. package/dist/modes/index.d.ts.map +1 -0
  274. package/dist/modes/index.js +8 -0
  275. package/dist/modes/index.js.map +1 -0
  276. package/dist/modes/interactive/components/armin.d.ts +34 -0
  277. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  278. package/dist/modes/interactive/components/armin.js +333 -0
  279. package/dist/modes/interactive/components/armin.js.map +1 -0
  280. package/dist/modes/interactive/components/assistant-message.d.ts +18 -0
  281. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  282. package/dist/modes/interactive/components/assistant-message.js +107 -0
  283. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  284. package/dist/modes/interactive/components/bash-execution.d.ts +34 -0
  285. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  286. package/dist/modes/interactive/components/bash-execution.js +175 -0
  287. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  288. package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
  289. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  290. package/dist/modes/interactive/components/bordered-loader.js +51 -0
  291. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  292. package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
  293. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  294. package/dist/modes/interactive/components/branch-summary-message.js +44 -0
  295. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  296. package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
  297. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  298. package/dist/modes/interactive/components/compaction-summary-message.js +45 -0
  299. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  300. package/dist/modes/interactive/components/config-selector.d.ts +71 -0
  301. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  302. package/dist/modes/interactive/components/config-selector.js +479 -0
  303. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  304. package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
  305. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  306. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  307. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  308. package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
  309. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  310. package/dist/modes/interactive/components/custom-editor.js +70 -0
  311. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  312. package/dist/modes/interactive/components/custom-message.d.ts +20 -0
  313. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  314. package/dist/modes/interactive/components/custom-message.js +79 -0
  315. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  316. package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
  317. package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
  318. package/dist/modes/interactive/components/daxnuts.js +140 -0
  319. package/dist/modes/interactive/components/daxnuts.js.map +1 -0
  320. package/dist/modes/interactive/components/diff.d.ts +12 -0
  321. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  322. package/dist/modes/interactive/components/diff.js +133 -0
  323. package/dist/modes/interactive/components/diff.js.map +1 -0
  324. package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
  325. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  326. package/dist/modes/interactive/components/dynamic-border.js +21 -0
  327. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  328. package/dist/modes/interactive/components/extension-editor.d.ts +20 -0
  329. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  330. package/dist/modes/interactive/components/extension-editor.js +111 -0
  331. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  332. package/dist/modes/interactive/components/extension-input.d.ts +23 -0
  333. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  334. package/dist/modes/interactive/components/extension-input.js +61 -0
  335. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  336. package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
  337. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  338. package/dist/modes/interactive/components/extension-selector.js +78 -0
  339. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  340. package/dist/modes/interactive/components/footer.d.ts +27 -0
  341. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  342. package/dist/modes/interactive/components/footer.js +201 -0
  343. package/dist/modes/interactive/components/footer.js.map +1 -0
  344. package/dist/modes/interactive/components/index.d.ts +32 -0
  345. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  346. package/dist/modes/interactive/components/index.js +33 -0
  347. package/dist/modes/interactive/components/index.js.map +1 -0
  348. package/dist/modes/interactive/components/keybinding-hints.d.ts +8 -0
  349. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
  350. package/dist/modes/interactive/components/keybinding-hints.js +22 -0
  351. package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
  352. package/dist/modes/interactive/components/login-dialog.d.ts +42 -0
  353. package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
  354. package/dist/modes/interactive/components/login-dialog.js +145 -0
  355. package/dist/modes/interactive/components/login-dialog.js.map +1 -0
  356. package/dist/modes/interactive/components/model-selector.d.ts +47 -0
  357. package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
  358. package/dist/modes/interactive/components/model-selector.js +275 -0
  359. package/dist/modes/interactive/components/model-selector.js.map +1 -0
  360. package/dist/modes/interactive/components/oauth-selector.d.ts +19 -0
  361. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
  362. package/dist/modes/interactive/components/oauth-selector.js +97 -0
  363. package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
  364. package/dist/modes/interactive/components/scoped-models-selector.d.ts +49 -0
  365. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
  366. package/dist/modes/interactive/components/scoped-models-selector.js +275 -0
  367. package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
  368. package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
  369. package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
  370. package/dist/modes/interactive/components/session-selector-search.js +155 -0
  371. package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
  372. package/dist/modes/interactive/components/session-selector.d.ts +95 -0
  373. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  374. package/dist/modes/interactive/components/session-selector.js +848 -0
  375. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  376. package/dist/modes/interactive/components/settings-selector.d.ts +58 -0
  377. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  378. package/dist/modes/interactive/components/settings-selector.js +301 -0
  379. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  380. package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
  381. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  382. package/dist/modes/interactive/components/show-images-selector.js +39 -0
  383. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  384. package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
  385. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  386. package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
  387. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  388. package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
  389. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  390. package/dist/modes/interactive/components/theme-selector.js +50 -0
  391. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  392. package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
  393. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  394. package/dist/modes/interactive/components/thinking-selector.js +51 -0
  395. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  396. package/dist/modes/interactive/components/tool-execution.d.ts +58 -0
  397. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  398. package/dist/modes/interactive/components/tool-execution.js +274 -0
  399. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  400. package/dist/modes/interactive/components/tree-selector.d.ts +89 -0
  401. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  402. package/dist/modes/interactive/components/tree-selector.js +1084 -0
  403. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  404. package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
  405. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  406. package/dist/modes/interactive/components/user-message-selector.js +113 -0
  407. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  408. package/dist/modes/interactive/components/user-message.d.ts +9 -0
  409. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  410. package/dist/modes/interactive/components/user-message.js +28 -0
  411. package/dist/modes/interactive/components/user-message.js.map +1 -0
  412. package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
  413. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  414. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  415. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  416. package/dist/modes/interactive/interactive-mode.d.ts +318 -0
  417. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  418. package/dist/modes/interactive/interactive-mode.js +3921 -0
  419. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  420. package/dist/modes/interactive/theme/dark.json +85 -0
  421. package/dist/modes/interactive/theme/light.json +84 -0
  422. package/dist/modes/interactive/theme/quake.json +84 -0
  423. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  424. package/dist/modes/interactive/theme/theme.d.ts +81 -0
  425. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  426. package/dist/modes/interactive/theme/theme.js +973 -0
  427. package/dist/modes/interactive/theme/theme.js.map +1 -0
  428. package/dist/modes/print-mode.d.ts +28 -0
  429. package/dist/modes/print-mode.d.ts.map +1 -0
  430. package/dist/modes/print-mode.js +108 -0
  431. package/dist/modes/print-mode.js.map +1 -0
  432. package/dist/modes/rpc/jsonl.d.ts +17 -0
  433. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  434. package/dist/modes/rpc/jsonl.js +49 -0
  435. package/dist/modes/rpc/jsonl.js.map +1 -0
  436. package/dist/modes/rpc/rpc-client.d.ts +217 -0
  437. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  438. package/dist/modes/rpc/rpc-client.js +401 -0
  439. package/dist/modes/rpc/rpc-client.js.map +1 -0
  440. package/dist/modes/rpc/rpc-mode.d.ts +20 -0
  441. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  442. package/dist/modes/rpc/rpc-mode.js +542 -0
  443. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  444. package/dist/modes/rpc/rpc-types.d.ts +408 -0
  445. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  446. package/dist/modes/rpc/rpc-types.js +8 -0
  447. package/dist/modes/rpc/rpc-types.js.map +1 -0
  448. package/dist/utils/changelog.d.ts +21 -0
  449. package/dist/utils/changelog.d.ts.map +1 -0
  450. package/dist/utils/changelog.js +87 -0
  451. package/dist/utils/changelog.js.map +1 -0
  452. package/dist/utils/child-process.d.ts +11 -0
  453. package/dist/utils/child-process.d.ts.map +1 -0
  454. package/dist/utils/child-process.js +78 -0
  455. package/dist/utils/child-process.js.map +1 -0
  456. package/dist/utils/clipboard-image.d.ts +11 -0
  457. package/dist/utils/clipboard-image.d.ts.map +1 -0
  458. package/dist/utils/clipboard-image.js +245 -0
  459. package/dist/utils/clipboard-image.js.map +1 -0
  460. package/dist/utils/clipboard-native.d.ts +8 -0
  461. package/dist/utils/clipboard-native.d.ts.map +1 -0
  462. package/dist/utils/clipboard-native.js +14 -0
  463. package/dist/utils/clipboard-native.js.map +1 -0
  464. package/dist/utils/clipboard.d.ts +2 -0
  465. package/dist/utils/clipboard.d.ts.map +1 -0
  466. package/dist/utils/clipboard.js +78 -0
  467. package/dist/utils/clipboard.js.map +1 -0
  468. package/dist/utils/exif-orientation.d.ts +5 -0
  469. package/dist/utils/exif-orientation.d.ts.map +1 -0
  470. package/dist/utils/exif-orientation.js +158 -0
  471. package/dist/utils/exif-orientation.js.map +1 -0
  472. package/dist/utils/frontmatter.d.ts +8 -0
  473. package/dist/utils/frontmatter.d.ts.map +1 -0
  474. package/dist/utils/frontmatter.js +26 -0
  475. package/dist/utils/frontmatter.js.map +1 -0
  476. package/dist/utils/git.d.ts +26 -0
  477. package/dist/utils/git.d.ts.map +1 -0
  478. package/dist/utils/git.js +163 -0
  479. package/dist/utils/git.js.map +1 -0
  480. package/dist/utils/image-convert.d.ts +9 -0
  481. package/dist/utils/image-convert.d.ts.map +1 -0
  482. package/dist/utils/image-convert.js +39 -0
  483. package/dist/utils/image-convert.js.map +1 -0
  484. package/dist/utils/image-resize.d.ts +36 -0
  485. package/dist/utils/image-resize.d.ts.map +1 -0
  486. package/dist/utils/image-resize.js +137 -0
  487. package/dist/utils/image-resize.js.map +1 -0
  488. package/dist/utils/mime.d.ts +2 -0
  489. package/dist/utils/mime.d.ts.map +1 -0
  490. package/dist/utils/mime.js +26 -0
  491. package/dist/utils/mime.js.map +1 -0
  492. package/dist/utils/photon.d.ts +21 -0
  493. package/dist/utils/photon.d.ts.map +1 -0
  494. package/dist/utils/photon.js +121 -0
  495. package/dist/utils/photon.js.map +1 -0
  496. package/dist/utils/shell.d.ts +26 -0
  497. package/dist/utils/shell.d.ts.map +1 -0
  498. package/dist/utils/shell.js +186 -0
  499. package/dist/utils/shell.js.map +1 -0
  500. package/dist/utils/sleep.d.ts +5 -0
  501. package/dist/utils/sleep.d.ts.map +1 -0
  502. package/dist/utils/sleep.js +17 -0
  503. package/dist/utils/sleep.js.map +1 -0
  504. package/dist/utils/tools-manager.d.ts +3 -0
  505. package/dist/utils/tools-manager.d.ts.map +1 -0
  506. package/dist/utils/tools-manager.js +252 -0
  507. package/dist/utils/tools-manager.js.map +1 -0
  508. package/docs/compaction.md +394 -0
  509. package/docs/custom-provider.md +596 -0
  510. package/docs/development.md +71 -0
  511. package/docs/extensions.md +2286 -0
  512. package/docs/images/doom-extension.png +0 -0
  513. package/docs/images/exy.png +0 -0
  514. package/docs/images/interactive-mode.png +0 -0
  515. package/docs/images/tree-view.png +0 -0
  516. package/docs/json.md +82 -0
  517. package/docs/keybindings.md +175 -0
  518. package/docs/models.md +341 -0
  519. package/docs/packages.md +218 -0
  520. package/docs/prompt-templates.md +67 -0
  521. package/docs/providers.md +195 -0
  522. package/docs/rpc.md +1377 -0
  523. package/docs/sdk.md +1064 -0
  524. package/docs/session.md +412 -0
  525. package/docs/settings.md +246 -0
  526. package/docs/shell-aliases.md +13 -0
  527. package/docs/skills.md +232 -0
  528. package/docs/terminal-setup.md +106 -0
  529. package/docs/termux.md +127 -0
  530. package/docs/themes.md +295 -0
  531. package/docs/tmux.md +61 -0
  532. package/docs/tree.md +231 -0
  533. package/docs/tui.md +887 -0
  534. package/docs/windows.md +17 -0
  535. package/examples/README.md +25 -0
  536. package/examples/extensions/README.md +206 -0
  537. package/examples/extensions/antigravity-image-gen.ts +418 -0
  538. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  539. package/examples/extensions/bash-spawn-hook.ts +30 -0
  540. package/examples/extensions/bookmark.ts +50 -0
  541. package/examples/extensions/built-in-tool-renderer.ts +246 -0
  542. package/examples/extensions/claude-rules.ts +86 -0
  543. package/examples/extensions/commands.ts +72 -0
  544. package/examples/extensions/confirm-destructive.ts +59 -0
  545. package/examples/extensions/custom-compaction.ts +127 -0
  546. package/examples/extensions/custom-footer.ts +64 -0
  547. package/examples/extensions/custom-header.ts +73 -0
  548. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  549. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  550. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  551. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  552. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  553. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  554. package/examples/extensions/custom-provider-qwen-cli/index.ts +345 -0
  555. package/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
  556. package/examples/extensions/dirty-repo-guard.ts +56 -0
  557. package/examples/extensions/doom-overlay/README.md +46 -0
  558. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  559. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  560. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  561. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  562. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  563. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  564. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  565. package/examples/extensions/doom-overlay/index.ts +74 -0
  566. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  567. package/examples/extensions/dynamic-resources/SKILL.md +8 -0
  568. package/examples/extensions/dynamic-resources/dynamic.json +79 -0
  569. package/examples/extensions/dynamic-resources/dynamic.md +5 -0
  570. package/examples/extensions/dynamic-resources/index.ts +15 -0
  571. package/examples/extensions/dynamic-tools.ts +74 -0
  572. package/examples/extensions/event-bus.ts +43 -0
  573. package/examples/extensions/file-trigger.ts +41 -0
  574. package/examples/extensions/git-checkpoint.ts +53 -0
  575. package/examples/extensions/handoff.ts +153 -0
  576. package/examples/extensions/hello.ts +26 -0
  577. package/examples/extensions/hidden-thinking-label.ts +53 -0
  578. package/examples/extensions/inline-bash.ts +94 -0
  579. package/examples/extensions/input-transform.ts +43 -0
  580. package/examples/extensions/interactive-shell.ts +196 -0
  581. package/examples/extensions/mac-system-theme.ts +47 -0
  582. package/examples/extensions/message-renderer.ts +59 -0
  583. package/examples/extensions/minimal-mode.ts +426 -0
  584. package/examples/extensions/modal-editor.ts +85 -0
  585. package/examples/extensions/model-status.ts +31 -0
  586. package/examples/extensions/notify.ts +55 -0
  587. package/examples/extensions/overlay-qa-tests.ts +1348 -0
  588. package/examples/extensions/overlay-test.ts +150 -0
  589. package/examples/extensions/permission-gate.ts +34 -0
  590. package/examples/extensions/pirate.ts +47 -0
  591. package/examples/extensions/plan-mode/README.md +65 -0
  592. package/examples/extensions/plan-mode/index.ts +340 -0
  593. package/examples/extensions/plan-mode/utils.ts +168 -0
  594. package/examples/extensions/preset.ts +397 -0
  595. package/examples/extensions/protected-paths.ts +30 -0
  596. package/examples/extensions/provider-payload.ts +14 -0
  597. package/examples/extensions/qna.ts +122 -0
  598. package/examples/extensions/question.ts +264 -0
  599. package/examples/extensions/questionnaire.ts +427 -0
  600. package/examples/extensions/rainbow-editor.ts +88 -0
  601. package/examples/extensions/reload-runtime.ts +37 -0
  602. package/examples/extensions/rpc-demo.ts +118 -0
  603. package/examples/extensions/sandbox/index.ts +321 -0
  604. package/examples/extensions/sandbox/package-lock.json +92 -0
  605. package/examples/extensions/sandbox/package.json +19 -0
  606. package/examples/extensions/send-user-message.ts +97 -0
  607. package/examples/extensions/session-name.ts +27 -0
  608. package/examples/extensions/shutdown-command.ts +63 -0
  609. package/examples/extensions/snake.ts +343 -0
  610. package/examples/extensions/space-invaders.ts +560 -0
  611. package/examples/extensions/ssh.ts +220 -0
  612. package/examples/extensions/status-line.ts +32 -0
  613. package/examples/extensions/subagent/README.md +172 -0
  614. package/examples/extensions/subagent/agents/planner.md +37 -0
  615. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  616. package/examples/extensions/subagent/agents/scout.md +50 -0
  617. package/examples/extensions/subagent/agents/worker.md +24 -0
  618. package/examples/extensions/subagent/agents.ts +126 -0
  619. package/examples/extensions/subagent/index.ts +986 -0
  620. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  621. package/examples/extensions/subagent/prompts/implement.md +10 -0
  622. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  623. package/examples/extensions/summarize.ts +206 -0
  624. package/examples/extensions/system-prompt-header.ts +17 -0
  625. package/examples/extensions/timed-confirm.ts +70 -0
  626. package/examples/extensions/titlebar-spinner.ts +58 -0
  627. package/examples/extensions/todo.ts +297 -0
  628. package/examples/extensions/tool-override.ts +144 -0
  629. package/examples/extensions/tools.ts +141 -0
  630. package/examples/extensions/trigger-compact.ts +50 -0
  631. package/examples/extensions/truncated-tool.ts +195 -0
  632. package/examples/extensions/widget-placement.ts +9 -0
  633. package/examples/extensions/with-deps/index.ts +32 -0
  634. package/examples/extensions/with-deps/package-lock.json +31 -0
  635. package/examples/extensions/with-deps/package.json +22 -0
  636. package/examples/rpc-extension-ui.ts +632 -0
  637. package/examples/sdk/01-minimal.ts +22 -0
  638. package/examples/sdk/02-custom-model.ts +49 -0
  639. package/examples/sdk/03-custom-prompt.ts +55 -0
  640. package/examples/sdk/04-skills.ts +52 -0
  641. package/examples/sdk/05-tools.ts +56 -0
  642. package/examples/sdk/06-extensions.ts +88 -0
  643. package/examples/sdk/07-context-files.ts +40 -0
  644. package/examples/sdk/08-prompt-templates.ts +48 -0
  645. package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  646. package/examples/sdk/10-settings.ts +51 -0
  647. package/examples/sdk/11-sessions.ts +48 -0
  648. package/examples/sdk/12-full-control.ts +81 -0
  649. package/examples/sdk/13-session-runtime.ts +49 -0
  650. package/examples/sdk/README.md +145 -0
  651. package/package.json +96 -0
@@ -0,0 +1,2286 @@
1
+ > pi can create extensions. Ask it to build one for your use case.
2
+
3
+ # Extensions
4
+
5
+ Extensions are TypeScript modules that extend pi's behavior. They can subscribe to lifecycle events, register custom tools callable by the LLM, add commands, and more.
6
+
7
+ > **Placement for /reload:** Put extensions in `~/.pi/agent/extensions/` (global) or `.pi/extensions/` (project-local) for auto-discovery. Use `pi -e ./path.ts` only for quick tests. Extensions in auto-discovered locations can be hot-reloaded with `/reload`.
8
+
9
+ **Key capabilities:**
10
+ - **Custom tools** - Register tools the LLM can call via `pi.registerTool()`
11
+ - **Event interception** - Block or modify tool calls, inject context, customize compaction
12
+ - **User interaction** - Prompt users via `ctx.ui` (select, confirm, input, notify)
13
+ - **Custom UI components** - Full TUI components with keyboard input via `ctx.ui.custom()` for complex interactions
14
+ - **Custom commands** - Register commands like `/mycommand` via `pi.registerCommand()`
15
+ - **Session persistence** - Store state that survives restarts via `pi.appendEntry()`
16
+ - **Custom rendering** - Control how tool calls/results and messages appear in TUI
17
+
18
+ **Example use cases:**
19
+ - Permission gates (confirm before `rm -rf`, `sudo`, etc.)
20
+ - Git checkpointing (stash at each turn, restore on branch)
21
+ - Path protection (block writes to `.env`, `node_modules/`)
22
+ - Custom compaction (summarize conversation your way)
23
+ - Conversation summaries (see `summarize.ts` example)
24
+ - Interactive tools (questions, wizards, custom dialogs)
25
+ - Stateful tools (todo lists, connection pools)
26
+ - External integrations (file watchers, webhooks, CI triggers)
27
+ - Games while you wait (see `snake.ts` example)
28
+
29
+ See [examples/extensions/](../examples/extensions/) for working implementations.
30
+
31
+ ## Table of Contents
32
+
33
+ - [Quick Start](#quick-start)
34
+ - [Extension Locations](#extension-locations)
35
+ - [Available Imports](#available-imports)
36
+ - [Writing an Extension](#writing-an-extension)
37
+ - [Extension Styles](#extension-styles)
38
+ - [Events](#events)
39
+ - [Lifecycle Overview](#lifecycle-overview)
40
+ - [Resource Events](#resource-events)
41
+ - [Session Events](#session-events)
42
+ - [Agent Events](#agent-events)
43
+ - [Tool Events](#tool-events)
44
+ - [ExtensionContext](#extensioncontext)
45
+ - [ExtensionCommandContext](#extensioncommandcontext)
46
+ - [ExtensionAPI Methods](#extensionapi-methods)
47
+ - [State Management](#state-management)
48
+ - [Custom Tools](#custom-tools)
49
+ - [Custom UI](#custom-ui)
50
+ - [Error Handling](#error-handling)
51
+ - [Mode Behavior](#mode-behavior)
52
+ - [Examples Reference](#examples-reference)
53
+
54
+ ## Quick Start
55
+
56
+ Create `~/.pi/agent/extensions/my-extension.ts`:
57
+
58
+ ```typescript
59
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
60
+ import { Type } from "@sinclair/typebox";
61
+
62
+ export default function (pi: ExtensionAPI) {
63
+ // React to events
64
+ pi.on("session_start", async (_event, ctx) => {
65
+ ctx.ui.notify("Extension loaded!", "info");
66
+ });
67
+
68
+ pi.on("tool_call", async (event, ctx) => {
69
+ if (event.toolName === "bash" && event.input.command?.includes("rm -rf")) {
70
+ const ok = await ctx.ui.confirm("Dangerous!", "Allow rm -rf?");
71
+ if (!ok) return { block: true, reason: "Blocked by user" };
72
+ }
73
+ });
74
+
75
+ // Register a custom tool
76
+ pi.registerTool({
77
+ name: "greet",
78
+ label: "Greet",
79
+ description: "Greet someone by name",
80
+ parameters: Type.Object({
81
+ name: Type.String({ description: "Name to greet" }),
82
+ }),
83
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
84
+ return {
85
+ content: [{ type: "text", text: `Hello, ${params.name}!` }],
86
+ details: {},
87
+ };
88
+ },
89
+ });
90
+
91
+ // Register a command
92
+ pi.registerCommand("hello", {
93
+ description: "Say hello",
94
+ handler: async (args, ctx) => {
95
+ ctx.ui.notify(`Hello ${args || "world"}!`, "info");
96
+ },
97
+ });
98
+ }
99
+ ```
100
+
101
+ Test with `--extension` (or `-e`) flag:
102
+
103
+ ```bash
104
+ pi -e ./my-extension.ts
105
+ ```
106
+
107
+ ## Extension Locations
108
+
109
+ > **Security:** Extensions run with your full system permissions and can execute arbitrary code. Only install from sources you trust.
110
+
111
+ Extensions are auto-discovered from:
112
+
113
+ | Location | Scope |
114
+ |----------|-------|
115
+ | `~/.pi/agent/extensions/*.ts` | Global (all projects) |
116
+ | `~/.pi/agent/extensions/*/index.ts` | Global (subdirectory) |
117
+ | `.pi/extensions/*.ts` | Project-local |
118
+ | `.pi/extensions/*/index.ts` | Project-local (subdirectory) |
119
+
120
+ Additional paths via `settings.json`:
121
+
122
+ ```json
123
+ {
124
+ "packages": [
125
+ "npm:@foo/bar@1.0.0",
126
+ "git:github.com/user/repo@v1"
127
+ ],
128
+ "extensions": [
129
+ "/path/to/local/extension.ts",
130
+ "/path/to/local/extension/dir"
131
+ ]
132
+ }
133
+ ```
134
+
135
+ To share extensions via npm or git as pi packages, see [packages.md](packages.md).
136
+
137
+ ## Available Imports
138
+
139
+ | Package | Purpose |
140
+ |---------|---------|
141
+ | `@mariozechner/pi-coding-agent` | Extension types (`ExtensionAPI`, `ExtensionContext`, events) |
142
+ | `@sinclair/typebox` | Schema definitions for tool parameters |
143
+ | `@mariozechner/pi-ai` | AI utilities (`StringEnum` for Google-compatible enums) |
144
+ | `@mariozechner/pi-tui` | TUI components for custom rendering |
145
+
146
+ npm dependencies work too. Add a `package.json` next to your extension (or in a parent directory), run `npm install`, and imports from `node_modules/` are resolved automatically.
147
+
148
+ Node.js built-ins (`node:fs`, `node:path`, etc.) are also available.
149
+
150
+ ## Writing an Extension
151
+
152
+ An extension exports a default function that receives `ExtensionAPI`:
153
+
154
+ ```typescript
155
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
156
+
157
+ export default function (pi: ExtensionAPI) {
158
+ // Subscribe to events
159
+ pi.on("event_name", async (event, ctx) => {
160
+ // ctx.ui for user interaction
161
+ const ok = await ctx.ui.confirm("Title", "Are you sure?");
162
+ ctx.ui.notify("Done!", "success");
163
+ ctx.ui.setStatus("my-ext", "Processing..."); // Footer status
164
+ ctx.ui.setWidget("my-ext", ["Line 1", "Line 2"]); // Widget above editor (default)
165
+ });
166
+
167
+ // Register tools, commands, shortcuts, flags
168
+ pi.registerTool({ ... });
169
+ pi.registerCommand("name", { ... });
170
+ pi.registerShortcut("ctrl+x", { ... });
171
+ pi.registerFlag("my-flag", { ... });
172
+ }
173
+ ```
174
+
175
+ Extensions are loaded via [jiti](https://github.com/unjs/jiti), so TypeScript works without compilation.
176
+
177
+ ### Extension Styles
178
+
179
+ **Single file** - simplest, for small extensions:
180
+
181
+ ```
182
+ ~/.pi/agent/extensions/
183
+ └── my-extension.ts
184
+ ```
185
+
186
+ **Directory with index.ts** - for multi-file extensions:
187
+
188
+ ```
189
+ ~/.pi/agent/extensions/
190
+ └── my-extension/
191
+ ├── index.ts # Entry point (exports default function)
192
+ ├── tools.ts # Helper module
193
+ └── utils.ts # Helper module
194
+ ```
195
+
196
+ **Package with dependencies** - for extensions that need npm packages:
197
+
198
+ ```
199
+ ~/.pi/agent/extensions/
200
+ └── my-extension/
201
+ ├── package.json # Declares dependencies and entry points
202
+ ├── package-lock.json
203
+ ├── node_modules/ # After npm install
204
+ └── src/
205
+ └── index.ts
206
+ ```
207
+
208
+ ```json
209
+ // package.json
210
+ {
211
+ "name": "my-extension",
212
+ "dependencies": {
213
+ "zod": "^3.0.0",
214
+ "chalk": "^5.0.0"
215
+ },
216
+ "pi": {
217
+ "extensions": ["./src/index.ts"]
218
+ }
219
+ }
220
+ ```
221
+
222
+ Run `npm install` in the extension directory, then imports from `node_modules/` work automatically.
223
+
224
+ ## Events
225
+
226
+ ### Lifecycle Overview
227
+
228
+ ```
229
+ pi starts (CLI only)
230
+
231
+ ├─► session_directory (CLI startup only, no ctx)
232
+ ├─► session_start { reason: "startup" }
233
+ └─► resources_discover { reason: "startup" }
234
+
235
+
236
+ user sends prompt ─────────────────────────────────────────┐
237
+ │ │
238
+ ├─► (extension commands checked first, bypass if found) │
239
+ ├─► input (can intercept, transform, or handle) │
240
+ ├─► (skill/template expansion if not handled) │
241
+ ├─► before_agent_start (can inject message, modify system prompt)
242
+ ├─► agent_start │
243
+ ├─► message_start / message_update / message_end │
244
+ │ │
245
+ │ ┌─── turn (repeats while LLM calls tools) ───┐ │
246
+ │ │ │ │
247
+ │ ├─► turn_start │ │
248
+ │ ├─► context (can modify messages) │ │
249
+ │ ├─► before_provider_request (can inspect or replace payload)
250
+ │ │ │ │
251
+ │ │ LLM responds, may call tools: │ │
252
+ │ │ ├─► tool_execution_start │ │
253
+ │ │ ├─► tool_call (can block) │ │
254
+ │ │ ├─► tool_execution_update │ │
255
+ │ │ ├─► tool_result (can modify) │ │
256
+ │ │ └─► tool_execution_end │ │
257
+ │ │ │ │
258
+ │ └─► turn_end │ │
259
+ │ │
260
+ └─► agent_end │
261
+
262
+ user sends another prompt ◄────────────────────────────────┘
263
+
264
+ /new (new session) or /resume (switch session)
265
+ ├─► session_before_switch (can cancel)
266
+ ├─► session_shutdown
267
+ ├─► session_start { reason: "new" | "resume", previousSessionFile? }
268
+ └─► resources_discover { reason: "startup" }
269
+
270
+ /fork
271
+ ├─► session_before_fork (can cancel)
272
+ ├─► session_shutdown
273
+ ├─► session_start { reason: "fork", previousSessionFile }
274
+ └─► resources_discover { reason: "startup" }
275
+
276
+ /compact or auto-compaction
277
+ ├─► session_before_compact (can cancel or customize)
278
+ └─► session_compact
279
+
280
+ /tree navigation
281
+ ├─► session_before_tree (can cancel or customize)
282
+ └─► session_tree
283
+
284
+ /model or Ctrl+P (model selection/cycling)
285
+ └─► model_select
286
+
287
+ exit (Ctrl+C, Ctrl+D)
288
+ └─► session_shutdown
289
+ ```
290
+
291
+ ### Resource Events
292
+
293
+ #### resources_discover
294
+
295
+ Fired after `session_start` so extensions can contribute additional skill, prompt, and theme paths.
296
+ The startup path uses `reason: "startup"`. Reload uses `reason: "reload"`.
297
+
298
+ ```typescript
299
+ pi.on("resources_discover", async (event, _ctx) => {
300
+ // event.cwd - current working directory
301
+ // event.reason - "startup" | "reload"
302
+ return {
303
+ skillPaths: ["/path/to/skills"],
304
+ promptPaths: ["/path/to/prompts"],
305
+ themePaths: ["/path/to/themes"],
306
+ };
307
+ });
308
+ ```
309
+
310
+ ### Session Events
311
+
312
+ See [session.md](session.md) for session storage internals and the SessionManager API.
313
+
314
+ #### session_directory
315
+
316
+ Fired by the `pi` CLI during startup session resolution, before the initial session manager is created.
317
+
318
+ This event is:
319
+ - CLI-only. It is not emitted in SDK mode.
320
+ - Startup-only. It is not emitted for later interactive `/new` or `/resume` actions.
321
+ - Lower priority than `--session-dir` and `sessionDir` in `settings.json`.
322
+ - Special-cased to receive no `ctx` argument.
323
+
324
+ If multiple extensions return `sessionDir`, the last one wins.
325
+ Combined precedence is: `--session-dir` CLI flag, then `sessionDir` in settings, then extension `session_directory` hooks.
326
+
327
+ ```typescript
328
+ pi.on("session_directory", async (event) => {
329
+ return {
330
+ sessionDir: `/tmp/pi-sessions/${encodeURIComponent(event.cwd)}`,
331
+ };
332
+ });
333
+ ```
334
+
335
+ #### session_start
336
+
337
+ Fired when a session is started, loaded, or reloaded.
338
+
339
+ ```typescript
340
+ pi.on("session_start", async (event, ctx) => {
341
+ // event.reason - "startup" | "reload" | "new" | "resume" | "fork"
342
+ // event.previousSessionFile - present for "new", "resume", and "fork"
343
+ ctx.ui.notify(`Session: ${ctx.sessionManager.getSessionFile() ?? "ephemeral"}`, "info");
344
+ });
345
+ ```
346
+
347
+ #### session_before_switch
348
+
349
+ Fired before starting a new session (`/new`) or switching sessions (`/resume`).
350
+
351
+ ```typescript
352
+ pi.on("session_before_switch", async (event, ctx) => {
353
+ // event.reason - "new" or "resume"
354
+ // event.targetSessionFile - session we're switching to (only for "resume")
355
+
356
+ if (event.reason === "new") {
357
+ const ok = await ctx.ui.confirm("Clear?", "Delete all messages?");
358
+ if (!ok) return { cancel: true };
359
+ }
360
+ });
361
+ ```
362
+
363
+ After a successful switch or new-session action, pi emits `session_shutdown` for the old extension instance, reloads and rebinds extensions for the new session, then emits `session_start` with `reason: "new" | "resume"` and `previousSessionFile`.
364
+ Do cleanup work in `session_shutdown`, then reestablish any in-memory state in `session_start`.
365
+
366
+ #### session_before_fork
367
+
368
+ Fired when forking via `/fork`.
369
+
370
+ ```typescript
371
+ pi.on("session_before_fork", async (event, ctx) => {
372
+ // event.entryId - ID of the entry being forked from
373
+ return { cancel: true }; // Cancel fork
374
+ // OR
375
+ return { skipConversationRestore: true }; // Fork but don't rewind messages
376
+ });
377
+ ```
378
+
379
+ After a successful fork, pi emits `session_shutdown` for the old extension instance, reloads and rebinds extensions for the new session, then emits `session_start` with `reason: "fork"` and `previousSessionFile`.
380
+ Do cleanup work in `session_shutdown`, then reestablish any in-memory state in `session_start`.
381
+
382
+ #### session_before_compact / session_compact
383
+
384
+ Fired on compaction. See [compaction.md](compaction.md) for details.
385
+
386
+ ```typescript
387
+ pi.on("session_before_compact", async (event, ctx) => {
388
+ const { preparation, branchEntries, customInstructions, signal } = event;
389
+
390
+ // Cancel:
391
+ return { cancel: true };
392
+
393
+ // Custom summary:
394
+ return {
395
+ compaction: {
396
+ summary: "...",
397
+ firstKeptEntryId: preparation.firstKeptEntryId,
398
+ tokensBefore: preparation.tokensBefore,
399
+ }
400
+ };
401
+ });
402
+
403
+ pi.on("session_compact", async (event, ctx) => {
404
+ // event.compactionEntry - the saved compaction
405
+ // event.fromExtension - whether extension provided it
406
+ });
407
+ ```
408
+
409
+ #### session_before_tree / session_tree
410
+
411
+ Fired on `/tree` navigation. See [tree.md](tree.md) for tree navigation concepts.
412
+
413
+ ```typescript
414
+ pi.on("session_before_tree", async (event, ctx) => {
415
+ const { preparation, signal } = event;
416
+ return { cancel: true };
417
+ // OR provide custom summary:
418
+ return { summary: { summary: "...", details: {} } };
419
+ });
420
+
421
+ pi.on("session_tree", async (event, ctx) => {
422
+ // event.newLeafId, oldLeafId, summaryEntry, fromExtension
423
+ });
424
+ ```
425
+
426
+ #### session_shutdown
427
+
428
+ Fired on exit (Ctrl+C, Ctrl+D, SIGTERM).
429
+
430
+ ```typescript
431
+ pi.on("session_shutdown", async (_event, ctx) => {
432
+ // Cleanup, save state, etc.
433
+ });
434
+ ```
435
+
436
+ ### Agent Events
437
+
438
+ #### before_agent_start
439
+
440
+ Fired after user submits prompt, before agent loop. Can inject a message and/or modify the system prompt.
441
+
442
+ ```typescript
443
+ pi.on("before_agent_start", async (event, ctx) => {
444
+ // event.prompt - user's prompt text
445
+ // event.images - attached images (if any)
446
+ // event.systemPrompt - current system prompt
447
+
448
+ return {
449
+ // Inject a persistent message (stored in session, sent to LLM)
450
+ message: {
451
+ customType: "my-extension",
452
+ content: "Additional context for the LLM",
453
+ display: true,
454
+ },
455
+ // Replace the system prompt for this turn (chained across extensions)
456
+ systemPrompt: event.systemPrompt + "\n\nExtra instructions for this turn...",
457
+ };
458
+ });
459
+ ```
460
+
461
+ #### agent_start / agent_end
462
+
463
+ Fired once per user prompt.
464
+
465
+ ```typescript
466
+ pi.on("agent_start", async (_event, ctx) => {});
467
+
468
+ pi.on("agent_end", async (event, ctx) => {
469
+ // event.messages - messages from this prompt
470
+ });
471
+ ```
472
+
473
+ #### turn_start / turn_end
474
+
475
+ Fired for each turn (one LLM response + tool calls).
476
+
477
+ ```typescript
478
+ pi.on("turn_start", async (event, ctx) => {
479
+ // event.turnIndex, event.timestamp
480
+ });
481
+
482
+ pi.on("turn_end", async (event, ctx) => {
483
+ // event.turnIndex, event.message, event.toolResults
484
+ });
485
+ ```
486
+
487
+ #### message_start / message_update / message_end
488
+
489
+ Fired for message lifecycle updates.
490
+
491
+ - `message_start` and `message_end` fire for user, assistant, and toolResult messages.
492
+ - `message_update` fires for assistant streaming updates.
493
+
494
+ ```typescript
495
+ pi.on("message_start", async (event, ctx) => {
496
+ // event.message
497
+ });
498
+
499
+ pi.on("message_update", async (event, ctx) => {
500
+ // event.message
501
+ // event.assistantMessageEvent (token-by-token stream event)
502
+ });
503
+
504
+ pi.on("message_end", async (event, ctx) => {
505
+ // event.message
506
+ });
507
+ ```
508
+
509
+ #### tool_execution_start / tool_execution_update / tool_execution_end
510
+
511
+ Fired for tool execution lifecycle updates.
512
+
513
+ In parallel tool mode:
514
+ - `tool_execution_start` is emitted in assistant source order during the preflight phase
515
+ - `tool_execution_update` events may interleave across tools
516
+ - `tool_execution_end` is emitted in assistant source order, matching final tool result message order
517
+
518
+ ```typescript
519
+ pi.on("tool_execution_start", async (event, ctx) => {
520
+ // event.toolCallId, event.toolName, event.args
521
+ });
522
+
523
+ pi.on("tool_execution_update", async (event, ctx) => {
524
+ // event.toolCallId, event.toolName, event.args, event.partialResult
525
+ });
526
+
527
+ pi.on("tool_execution_end", async (event, ctx) => {
528
+ // event.toolCallId, event.toolName, event.result, event.isError
529
+ });
530
+ ```
531
+
532
+ #### context
533
+
534
+ Fired before each LLM call. Modify messages non-destructively. See [session.md](session.md) for message types.
535
+
536
+ ```typescript
537
+ pi.on("context", async (event, ctx) => {
538
+ // event.messages - deep copy, safe to modify
539
+ const filtered = event.messages.filter(m => !shouldPrune(m));
540
+ return { messages: filtered };
541
+ });
542
+ ```
543
+
544
+ #### before_provider_request
545
+
546
+ Fired after the provider-specific payload is built, right before the request is sent. Handlers run in extension load order. Returning `undefined` keeps the payload unchanged. Returning any other value replaces the payload for later handlers and for the actual request.
547
+
548
+ ```typescript
549
+ pi.on("before_provider_request", (event, ctx) => {
550
+ console.log(JSON.stringify(event.payload, null, 2));
551
+
552
+ // Optional: replace payload
553
+ // return { ...event.payload, temperature: 0 };
554
+ });
555
+ ```
556
+
557
+ This is mainly useful for debugging provider serialization and cache behavior.
558
+
559
+ ### Model Events
560
+
561
+ #### model_select
562
+
563
+ Fired when the model changes via `/model` command, model cycling (`Ctrl+P`), or session restore.
564
+
565
+ ```typescript
566
+ pi.on("model_select", async (event, ctx) => {
567
+ // event.model - newly selected model
568
+ // event.previousModel - previous model (undefined if first selection)
569
+ // event.source - "set" | "cycle" | "restore"
570
+
571
+ const prev = event.previousModel
572
+ ? `${event.previousModel.provider}/${event.previousModel.id}`
573
+ : "none";
574
+ const next = `${event.model.provider}/${event.model.id}`;
575
+
576
+ ctx.ui.notify(`Model changed (${event.source}): ${prev} -> ${next}`, "info");
577
+ });
578
+ ```
579
+
580
+ Use this to update UI elements (status bars, footers) or perform model-specific initialization when the active model changes.
581
+
582
+ ### Tool Events
583
+
584
+ #### tool_call
585
+
586
+ Fired after `tool_execution_start`, before the tool executes. **Can block.** Use `isToolCallEventType` to narrow and get typed inputs.
587
+
588
+ Before `tool_call` runs, pi waits for previously emitted Agent events to finish draining through `AgentSession`. This means `ctx.sessionManager` is up to date through the current assistant tool-calling message.
589
+
590
+ In the default parallel tool execution mode, sibling tool calls from the same assistant message are preflighted sequentially, then executed concurrently. `tool_call` is not guaranteed to see sibling tool results from that same assistant message in `ctx.sessionManager`.
591
+
592
+ `event.input` is mutable. Mutate it in place to patch tool arguments before execution.
593
+
594
+ Behavior guarantees:
595
+ - Mutations to `event.input` affect the actual tool execution
596
+ - Later `tool_call` handlers see mutations made by earlier handlers
597
+ - No re-validation is performed after your mutation
598
+ - Return values from `tool_call` only control blocking via `{ block: true, reason?: string }`
599
+
600
+ ```typescript
601
+ import { isToolCallEventType } from "@mariozechner/pi-coding-agent";
602
+
603
+ pi.on("tool_call", async (event, ctx) => {
604
+ // event.toolName - "bash", "read", "write", "edit", etc.
605
+ // event.toolCallId
606
+ // event.input - tool parameters (mutable)
607
+
608
+ // Built-in tools: no type params needed
609
+ if (isToolCallEventType("bash", event)) {
610
+ // event.input is { command: string; timeout?: number }
611
+ event.input.command = `source ~/.profile\n${event.input.command}`;
612
+
613
+ if (event.input.command.includes("rm -rf")) {
614
+ return { block: true, reason: "Dangerous command" };
615
+ }
616
+ }
617
+
618
+ if (isToolCallEventType("read", event)) {
619
+ // event.input is { path: string; offset?: number; limit?: number }
620
+ console.log(`Reading: ${event.input.path}`);
621
+ }
622
+ });
623
+ ```
624
+
625
+ #### Typing custom tool input
626
+
627
+ Custom tools should export their input type:
628
+
629
+ ```typescript
630
+ // my-extension.ts
631
+ export type MyToolInput = Static<typeof myToolSchema>;
632
+ ```
633
+
634
+ Use `isToolCallEventType` with explicit type parameters:
635
+
636
+ ```typescript
637
+ import { isToolCallEventType } from "@mariozechner/pi-coding-agent";
638
+ import type { MyToolInput } from "my-extension";
639
+
640
+ pi.on("tool_call", (event) => {
641
+ if (isToolCallEventType<"my_tool", MyToolInput>("my_tool", event)) {
642
+ event.input.action; // typed
643
+ }
644
+ });
645
+ ```
646
+
647
+ #### tool_result
648
+
649
+ Fired after tool execution finishes and before `tool_execution_end` plus the final tool result message events are emitted. **Can modify result.**
650
+
651
+ `tool_result` handlers chain like middleware:
652
+ - Handlers run in extension load order
653
+ - Each handler sees the latest result after previous handler changes
654
+ - Handlers can return partial patches (`content`, `details`, or `isError`); omitted fields keep their current values
655
+
656
+ Use `ctx.signal` for nested async work inside the handler. This lets Esc cancel model calls, `fetch()`, and other abort-aware operations started by the extension.
657
+
658
+ ```typescript
659
+ import { isBashToolResult } from "@mariozechner/pi-coding-agent";
660
+
661
+ pi.on("tool_result", async (event, ctx) => {
662
+ // event.toolName, event.toolCallId, event.input
663
+ // event.content, event.details, event.isError
664
+
665
+ if (isBashToolResult(event)) {
666
+ // event.details is typed as BashToolDetails
667
+ }
668
+
669
+ const response = await fetch("https://example.com/summarize", {
670
+ method: "POST",
671
+ body: JSON.stringify({ content: event.content }),
672
+ signal: ctx.signal,
673
+ });
674
+
675
+ // Modify result:
676
+ return { content: [...], details: {...}, isError: false };
677
+ });
678
+ ```
679
+
680
+ ### User Bash Events
681
+
682
+ #### user_bash
683
+
684
+ Fired when user executes `!` or `!!` commands. **Can intercept.**
685
+
686
+ ```typescript
687
+ import { createLocalBashOperations } from "@mariozechner/pi-coding-agent";
688
+
689
+ pi.on("user_bash", (event, ctx) => {
690
+ // event.command - the bash command
691
+ // event.excludeFromContext - true if !! prefix
692
+ // event.cwd - working directory
693
+
694
+ // Option 1: Provide custom operations (e.g., SSH)
695
+ return { operations: remoteBashOps };
696
+
697
+ // Option 2: Wrap pi's built-in local bash backend
698
+ const local = createLocalBashOperations();
699
+ return {
700
+ operations: {
701
+ exec(command, cwd, options) {
702
+ return local.exec(`source ~/.profile\n${command}`, cwd, options);
703
+ }
704
+ }
705
+ };
706
+
707
+ // Option 3: Full replacement - return result directly
708
+ return { result: { output: "...", exitCode: 0, cancelled: false, truncated: false } };
709
+ });
710
+ ```
711
+
712
+ ### Input Events
713
+
714
+ #### input
715
+
716
+ Fired when user input is received, after extension commands are checked but before skill and template expansion. The event sees the raw input text, so `/skill:foo` and `/template` are not yet expanded.
717
+
718
+ **Processing order:**
719
+ 1. Extension commands (`/cmd`) checked first - if found, handler runs and input event is skipped
720
+ 2. `input` event fires - can intercept, transform, or handle
721
+ 3. If not handled: skill commands (`/skill:name`) expanded to skill content
722
+ 4. If not handled: prompt templates (`/template`) expanded to template content
723
+ 5. Agent processing begins (`before_agent_start`, etc.)
724
+
725
+ ```typescript
726
+ pi.on("input", async (event, ctx) => {
727
+ // event.text - raw input (before skill/template expansion)
728
+ // event.images - attached images, if any
729
+ // event.source - "interactive" (typed), "rpc" (API), or "extension" (via sendUserMessage)
730
+
731
+ // Transform: rewrite input before expansion
732
+ if (event.text.startsWith("?quick "))
733
+ return { action: "transform", text: `Respond briefly: ${event.text.slice(7)}` };
734
+
735
+ // Handle: respond without LLM (extension shows its own feedback)
736
+ if (event.text === "ping") {
737
+ ctx.ui.notify("pong", "info");
738
+ return { action: "handled" };
739
+ }
740
+
741
+ // Route by source: skip processing for extension-injected messages
742
+ if (event.source === "extension") return { action: "continue" };
743
+
744
+ // Intercept skill commands before expansion
745
+ if (event.text.startsWith("/skill:")) {
746
+ // Could transform, block, or let pass through
747
+ }
748
+
749
+ return { action: "continue" }; // Default: pass through to expansion
750
+ });
751
+ ```
752
+
753
+ **Results:**
754
+ - `continue` - pass through unchanged (default if handler returns nothing)
755
+ - `transform` - modify text/images, then continue to expansion
756
+ - `handled` - skip agent entirely (first handler to return this wins)
757
+
758
+ Transforms chain across handlers. See [input-transform.ts](../examples/extensions/input-transform.ts).
759
+
760
+ ## ExtensionContext
761
+
762
+ All handlers except `session_directory` receive `ctx: ExtensionContext`.
763
+
764
+ `session_directory` is a CLI startup hook and receives only the event.
765
+
766
+ ### ctx.ui
767
+
768
+ UI methods for user interaction. See [Custom UI](#custom-ui) for full details.
769
+
770
+ ### ctx.hasUI
771
+
772
+ `false` in print mode (`-p`) and JSON mode. `true` in interactive and RPC mode. In RPC mode, dialog methods (`select`, `confirm`, `input`, `editor`) work via the extension UI sub-protocol, and fire-and-forget methods (`notify`, `setStatus`, `setWidget`, `setTitle`, `setEditorText`) emit requests to the client. Some TUI-specific methods are no-ops or return defaults (see [rpc.md](rpc.md#extension-ui-protocol)).
773
+
774
+ ### ctx.cwd
775
+
776
+ Current working directory.
777
+
778
+ ### ctx.sessionManager
779
+
780
+ Read-only access to session state. See [session.md](session.md) for the full SessionManager API and entry types.
781
+
782
+ For `tool_call`, this state is synchronized through the current assistant message before handlers run. In parallel tool execution mode it is still not guaranteed to include sibling tool results from the same assistant message.
783
+
784
+ ```typescript
785
+ ctx.sessionManager.getEntries() // All entries
786
+ ctx.sessionManager.getBranch() // Current branch
787
+ ctx.sessionManager.getLeafId() // Current leaf entry ID
788
+ ```
789
+
790
+ ### ctx.modelRegistry / ctx.model
791
+
792
+ Access to models and API keys.
793
+
794
+ ### ctx.signal
795
+
796
+ The current agent abort signal, or `undefined` when no agent turn is active.
797
+
798
+ Use this for abort-aware nested work started by extension handlers, for example:
799
+ - `fetch(..., { signal: ctx.signal })`
800
+ - model calls that accept `signal`
801
+ - file or process helpers that accept `AbortSignal`
802
+
803
+ `ctx.signal` is typically defined during active turn events such as `tool_call`, `tool_result`, `message_update`, and `turn_end`.
804
+ It is usually `undefined` in idle or non-turn contexts such as session events, extension commands, and shortcuts fired while pi is idle.
805
+
806
+ ```typescript
807
+ pi.on("tool_result", async (event, ctx) => {
808
+ const response = await fetch("https://example.com/api", {
809
+ method: "POST",
810
+ body: JSON.stringify(event),
811
+ signal: ctx.signal,
812
+ });
813
+
814
+ const data = await response.json();
815
+ return { details: data };
816
+ });
817
+ ```
818
+
819
+ ### ctx.isIdle() / ctx.abort() / ctx.hasPendingMessages()
820
+
821
+ Control flow helpers.
822
+
823
+ ### ctx.shutdown()
824
+
825
+ Request a graceful shutdown of pi.
826
+
827
+ - **Interactive mode:** Deferred until the agent becomes idle (after processing all queued steering and follow-up messages).
828
+ - **RPC mode:** Deferred until the next idle state (after completing the current command response, when waiting for the next command).
829
+ - **Print mode:** No-op. The process exits automatically when all prompts are processed.
830
+
831
+ Emits `session_shutdown` event to all extensions before exiting. Available in all contexts (event handlers, tools, commands, shortcuts).
832
+
833
+ ```typescript
834
+ pi.on("tool_call", (event, ctx) => {
835
+ if (isFatal(event.input)) {
836
+ ctx.shutdown();
837
+ }
838
+ });
839
+ ```
840
+
841
+ ### ctx.getContextUsage()
842
+
843
+ Returns current context usage for the active model. Uses last assistant usage when available, then estimates tokens for trailing messages.
844
+
845
+ ```typescript
846
+ const usage = ctx.getContextUsage();
847
+ if (usage && usage.tokens > 100_000) {
848
+ // ...
849
+ }
850
+ ```
851
+
852
+ ### ctx.compact()
853
+
854
+ Trigger compaction without awaiting completion. Use `onComplete` and `onError` for follow-up actions.
855
+
856
+ ```typescript
857
+ ctx.compact({
858
+ customInstructions: "Focus on recent changes",
859
+ onComplete: (result) => {
860
+ ctx.ui.notify("Compaction completed", "info");
861
+ },
862
+ onError: (error) => {
863
+ ctx.ui.notify(`Compaction failed: ${error.message}`, "error");
864
+ },
865
+ });
866
+ ```
867
+
868
+ ### ctx.getSystemPrompt()
869
+
870
+ Returns the current effective system prompt. This includes any modifications made by `before_agent_start` handlers for the current turn.
871
+
872
+ ```typescript
873
+ pi.on("before_agent_start", (event, ctx) => {
874
+ const prompt = ctx.getSystemPrompt();
875
+ console.log(`System prompt length: ${prompt.length}`);
876
+ });
877
+ ```
878
+
879
+ ## ExtensionCommandContext
880
+
881
+ Command handlers receive `ExtensionCommandContext`, which extends `ExtensionContext` with session control methods. These are only available in commands because they can deadlock if called from event handlers.
882
+
883
+ ### ctx.waitForIdle()
884
+
885
+ Wait for the agent to finish streaming:
886
+
887
+ ```typescript
888
+ pi.registerCommand("my-cmd", {
889
+ handler: async (args, ctx) => {
890
+ await ctx.waitForIdle();
891
+ // Agent is now idle, safe to modify session
892
+ },
893
+ });
894
+ ```
895
+
896
+ ### ctx.newSession(options?)
897
+
898
+ Create a new session:
899
+
900
+ ```typescript
901
+ const result = await ctx.newSession({
902
+ parentSession: ctx.sessionManager.getSessionFile(),
903
+ setup: async (sm) => {
904
+ sm.appendMessage({
905
+ role: "user",
906
+ content: [{ type: "text", text: "Context from previous session..." }],
907
+ timestamp: Date.now(),
908
+ });
909
+ },
910
+ });
911
+
912
+ if (result.cancelled) {
913
+ // An extension cancelled the new session
914
+ }
915
+ ```
916
+
917
+ ### ctx.fork(entryId)
918
+
919
+ Fork from a specific entry, creating a new session file:
920
+
921
+ ```typescript
922
+ const result = await ctx.fork("entry-id-123");
923
+ if (!result.cancelled) {
924
+ // Now in the forked session
925
+ }
926
+ ```
927
+
928
+ ### ctx.navigateTree(targetId, options?)
929
+
930
+ Navigate to a different point in the session tree:
931
+
932
+ ```typescript
933
+ const result = await ctx.navigateTree("entry-id-456", {
934
+ summarize: true,
935
+ customInstructions: "Focus on error handling changes",
936
+ replaceInstructions: false, // true = replace default prompt entirely
937
+ label: "review-checkpoint",
938
+ });
939
+ ```
940
+
941
+ Options:
942
+ - `summarize`: Whether to generate a summary of the abandoned branch
943
+ - `customInstructions`: Custom instructions for the summarizer
944
+ - `replaceInstructions`: If true, `customInstructions` replaces the default prompt instead of being appended
945
+ - `label`: Label to attach to the branch summary entry (or target entry if not summarizing)
946
+
947
+ ### ctx.switchSession(sessionPath)
948
+
949
+ Switch to a different session file:
950
+
951
+ ```typescript
952
+ const result = await ctx.switchSession("/path/to/session.jsonl");
953
+ if (result.cancelled) {
954
+ // An extension cancelled the switch via session_before_switch
955
+ }
956
+ ```
957
+
958
+ To discover available sessions, use the static `SessionManager.list()` or `SessionManager.listAll()` methods:
959
+
960
+ ```typescript
961
+ import { SessionManager } from "@mariozechner/pi-coding-agent";
962
+
963
+ pi.registerCommand("switch", {
964
+ description: "Switch to another session",
965
+ handler: async (args, ctx) => {
966
+ const sessions = await SessionManager.list(ctx.cwd);
967
+ if (sessions.length === 0) return;
968
+ const choice = await ctx.ui.select(
969
+ "Pick session:",
970
+ sessions.map(s => s.file),
971
+ );
972
+ if (choice) {
973
+ await ctx.switchSession(choice);
974
+ }
975
+ },
976
+ });
977
+ ```
978
+
979
+ ### ctx.reload()
980
+
981
+ Run the same reload flow as `/reload`.
982
+
983
+ ```typescript
984
+ pi.registerCommand("reload-runtime", {
985
+ description: "Reload extensions, skills, prompts, and themes",
986
+ handler: async (_args, ctx) => {
987
+ await ctx.reload();
988
+ return;
989
+ },
990
+ });
991
+ ```
992
+
993
+ Important behavior:
994
+ - `await ctx.reload()` emits `session_shutdown` for the current extension runtime
995
+ - It then reloads resources and emits `session_start` with `reason: "reload"` and `resources_discover` with reason `"reload"`
996
+ - The currently running command handler still continues in the old call frame
997
+ - Code after `await ctx.reload()` still runs from the pre-reload version
998
+ - Code after `await ctx.reload()` must not assume old in-memory extension state is still valid
999
+ - After the handler returns, future commands/events/tool calls use the new extension version
1000
+
1001
+ For predictable behavior, treat reload as terminal for that handler (`await ctx.reload(); return;`).
1002
+
1003
+ Tools run with `ExtensionContext`, so they cannot call `ctx.reload()` directly. Use a command as the reload entrypoint, then expose a tool that queues that command as a follow-up user message.
1004
+
1005
+ Example tool the LLM can call to trigger reload:
1006
+
1007
+ ```typescript
1008
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
1009
+ import { Type } from "@sinclair/typebox";
1010
+
1011
+ export default function (pi: ExtensionAPI) {
1012
+ pi.registerCommand("reload-runtime", {
1013
+ description: "Reload extensions, skills, prompts, and themes",
1014
+ handler: async (_args, ctx) => {
1015
+ await ctx.reload();
1016
+ return;
1017
+ },
1018
+ });
1019
+
1020
+ pi.registerTool({
1021
+ name: "reload_runtime",
1022
+ label: "Reload Runtime",
1023
+ description: "Reload extensions, skills, prompts, and themes",
1024
+ parameters: Type.Object({}),
1025
+ async execute() {
1026
+ pi.sendUserMessage("/reload-runtime", { deliverAs: "followUp" });
1027
+ return {
1028
+ content: [{ type: "text", text: "Queued /reload-runtime as a follow-up command." }],
1029
+ };
1030
+ },
1031
+ });
1032
+ }
1033
+ ```
1034
+
1035
+ ## ExtensionAPI Methods
1036
+
1037
+ ### pi.on(event, handler)
1038
+
1039
+ Subscribe to events. See [Events](#events) for event types and return values.
1040
+
1041
+ ### pi.registerTool(definition)
1042
+
1043
+ Register a custom tool callable by the LLM. See [Custom Tools](#custom-tools) for full details.
1044
+
1045
+ `pi.registerTool()` works both during extension load and after startup. You can call it inside `session_start`, command handlers, or other event handlers. New tools are refreshed immediately in the same session, so they appear in `pi.getAllTools()` and are callable by the LLM without `/reload`.
1046
+
1047
+ Use `pi.setActiveTools()` to enable or disable tools (including dynamically added tools) at runtime.
1048
+
1049
+ Use `promptSnippet` to opt a custom tool into a one-line entry in `Available tools`, and `promptGuidelines` to append tool-specific bullets to the default `Guidelines` section when the tool is active.
1050
+
1051
+ See [dynamic-tools.ts](../examples/extensions/dynamic-tools.ts) for a full example.
1052
+
1053
+ ```typescript
1054
+ import { Type } from "@sinclair/typebox";
1055
+ import { StringEnum } from "@mariozechner/pi-ai";
1056
+
1057
+ pi.registerTool({
1058
+ name: "my_tool",
1059
+ label: "My Tool",
1060
+ description: "What this tool does",
1061
+ promptSnippet: "Summarize or transform text according to action",
1062
+ promptGuidelines: ["Use this tool when the user asks to summarize previously generated text."],
1063
+ parameters: Type.Object({
1064
+ action: StringEnum(["list", "add"] as const),
1065
+ text: Type.Optional(Type.String()),
1066
+ }),
1067
+ prepareArguments(args) {
1068
+ // Optional compatibility shim. Runs before schema validation.
1069
+ // Return the current schema shape, for example to fold legacy fields
1070
+ // into the modern parameter object.
1071
+ return args;
1072
+ },
1073
+
1074
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
1075
+ // Stream progress
1076
+ onUpdate?.({ content: [{ type: "text", text: "Working..." }] });
1077
+
1078
+ return {
1079
+ content: [{ type: "text", text: "Done" }],
1080
+ details: { result: "..." },
1081
+ };
1082
+ },
1083
+
1084
+ // Optional: Custom rendering
1085
+ renderCall(args, theme, context) { ... },
1086
+ renderResult(result, options, theme, context) { ... },
1087
+ });
1088
+ ```
1089
+
1090
+ ### pi.sendMessage(message, options?)
1091
+
1092
+ Inject a custom message into the session.
1093
+
1094
+ ```typescript
1095
+ pi.sendMessage({
1096
+ customType: "my-extension",
1097
+ content: "Message text",
1098
+ display: true,
1099
+ details: { ... },
1100
+ }, {
1101
+ triggerTurn: true,
1102
+ deliverAs: "steer",
1103
+ });
1104
+ ```
1105
+
1106
+ **Options:**
1107
+ - `deliverAs` - Delivery mode:
1108
+ - `"steer"` (default) - Queues the message while streaming. Delivered after the current assistant turn finishes executing its tool calls, before the next LLM call.
1109
+ - `"followUp"` - Waits for agent to finish. Delivered only when agent has no more tool calls.
1110
+ - `"nextTurn"` - Queued for next user prompt. Does not interrupt or trigger anything.
1111
+ - `triggerTurn: true` - If agent is idle, trigger an LLM response immediately. Only applies to `"steer"` and `"followUp"` modes (ignored for `"nextTurn"`).
1112
+
1113
+ ### pi.sendUserMessage(content, options?)
1114
+
1115
+ Send a user message to the agent. Unlike `sendMessage()` which sends custom messages, this sends an actual user message that appears as if typed by the user. Always triggers a turn.
1116
+
1117
+ ```typescript
1118
+ // Simple text message
1119
+ pi.sendUserMessage("What is 2+2?");
1120
+
1121
+ // With content array (text + images)
1122
+ pi.sendUserMessage([
1123
+ { type: "text", text: "Describe this image:" },
1124
+ { type: "image", source: { type: "base64", mediaType: "image/png", data: "..." } },
1125
+ ]);
1126
+
1127
+ // During streaming - must specify delivery mode
1128
+ pi.sendUserMessage("Focus on error handling", { deliverAs: "steer" });
1129
+ pi.sendUserMessage("And then summarize", { deliverAs: "followUp" });
1130
+ ```
1131
+
1132
+ **Options:**
1133
+ - `deliverAs` - Required when agent is streaming:
1134
+ - `"steer"` - Queues the message for delivery after the current assistant turn finishes executing its tool calls
1135
+ - `"followUp"` - Waits for agent to finish all tools
1136
+
1137
+ When not streaming, the message is sent immediately and triggers a new turn. When streaming without `deliverAs`, throws an error.
1138
+
1139
+ See [send-user-message.ts](../examples/extensions/send-user-message.ts) for a complete example.
1140
+
1141
+ ### pi.appendEntry(customType, data?)
1142
+
1143
+ Persist extension state (does NOT participate in LLM context).
1144
+
1145
+ ```typescript
1146
+ pi.appendEntry("my-state", { count: 42 });
1147
+
1148
+ // Restore on reload
1149
+ pi.on("session_start", async (_event, ctx) => {
1150
+ for (const entry of ctx.sessionManager.getEntries()) {
1151
+ if (entry.type === "custom" && entry.customType === "my-state") {
1152
+ // Reconstruct from entry.data
1153
+ }
1154
+ }
1155
+ });
1156
+ ```
1157
+
1158
+ ### pi.setSessionName(name)
1159
+
1160
+ Set the session display name (shown in session selector instead of first message).
1161
+
1162
+ ```typescript
1163
+ pi.setSessionName("Refactor auth module");
1164
+ ```
1165
+
1166
+ ### pi.getSessionName()
1167
+
1168
+ Get the current session name, if set.
1169
+
1170
+ ```typescript
1171
+ const name = pi.getSessionName();
1172
+ if (name) {
1173
+ console.log(`Session: ${name}`);
1174
+ }
1175
+ ```
1176
+
1177
+ ### pi.setLabel(entryId, label)
1178
+
1179
+ Set or clear a label on an entry. Labels are user-defined markers for bookmarking and navigation (shown in `/tree` selector).
1180
+
1181
+ ```typescript
1182
+ // Set a label
1183
+ pi.setLabel(entryId, "checkpoint-before-refactor");
1184
+
1185
+ // Clear a label
1186
+ pi.setLabel(entryId, undefined);
1187
+
1188
+ // Read labels via sessionManager
1189
+ const label = ctx.sessionManager.getLabel(entryId);
1190
+ ```
1191
+
1192
+ Labels persist in the session and survive restarts. Use them to mark important points (turns, checkpoints) in the conversation tree.
1193
+
1194
+ ### pi.registerCommand(name, options)
1195
+
1196
+ Register a command.
1197
+
1198
+ If multiple extensions register the same command name, pi keeps them all and assigns numeric invocation suffixes in load order, for example `/review:1` and `/review:2`.
1199
+
1200
+ ```typescript
1201
+ pi.registerCommand("stats", {
1202
+ description: "Show session statistics",
1203
+ handler: async (args, ctx) => {
1204
+ const count = ctx.sessionManager.getEntries().length;
1205
+ ctx.ui.notify(`${count} entries`, "info");
1206
+ }
1207
+ });
1208
+ ```
1209
+
1210
+ Optional: add argument auto-completion for `/command ...`:
1211
+
1212
+ ```typescript
1213
+ import type { AutocompleteItem } from "@mariozechner/pi-tui";
1214
+
1215
+ pi.registerCommand("deploy", {
1216
+ description: "Deploy to an environment",
1217
+ getArgumentCompletions: (prefix: string): AutocompleteItem[] | null => {
1218
+ const envs = ["dev", "staging", "prod"];
1219
+ const items = envs.map((e) => ({ value: e, label: e }));
1220
+ const filtered = items.filter((i) => i.value.startsWith(prefix));
1221
+ return filtered.length > 0 ? filtered : null;
1222
+ },
1223
+ handler: async (args, ctx) => {
1224
+ ctx.ui.notify(`Deploying: ${args}`, "info");
1225
+ },
1226
+ });
1227
+ ```
1228
+
1229
+ ### pi.getCommands()
1230
+
1231
+ Get the slash commands available for invocation via `prompt` in the current session. Includes extension commands, prompt templates, and skill commands.
1232
+ The list matches the RPC `get_commands` ordering: extensions first, then templates, then skills.
1233
+
1234
+ ```typescript
1235
+ const commands = pi.getCommands();
1236
+ const bySource = commands.filter((command) => command.source === "extension");
1237
+ const userScoped = commands.filter((command) => command.sourceInfo.scope === "user");
1238
+ ```
1239
+
1240
+ Each entry has this shape:
1241
+
1242
+ ```typescript
1243
+ {
1244
+ name: string; // Invokable command name without the leading slash. May be suffixed like "review:1"
1245
+ description?: string;
1246
+ source: "extension" | "prompt" | "skill";
1247
+ sourceInfo: {
1248
+ path: string;
1249
+ source: string;
1250
+ scope: "user" | "project" | "temporary";
1251
+ origin: "package" | "top-level";
1252
+ baseDir?: string;
1253
+ };
1254
+ }
1255
+ ```
1256
+
1257
+ Use `sourceInfo` as the canonical provenance field. Do not infer ownership from command names or from ad hoc path parsing.
1258
+
1259
+ Built-in interactive commands (like `/model` and `/settings`) are not included here. They are handled only in interactive
1260
+ mode and would not execute if sent via `prompt`.
1261
+
1262
+ ### pi.registerMessageRenderer(customType, renderer)
1263
+
1264
+ Register a custom TUI renderer for messages with your `customType`. See [Custom UI](#custom-ui).
1265
+
1266
+ ### pi.registerShortcut(shortcut, options)
1267
+
1268
+ Register a keyboard shortcut. See [keybindings.md](keybindings.md) for the shortcut format and built-in keybindings.
1269
+
1270
+ ```typescript
1271
+ pi.registerShortcut("ctrl+shift+p", {
1272
+ description: "Toggle plan mode",
1273
+ handler: async (ctx) => {
1274
+ ctx.ui.notify("Toggled!");
1275
+ },
1276
+ });
1277
+ ```
1278
+
1279
+ ### pi.registerFlag(name, options)
1280
+
1281
+ Register a CLI flag.
1282
+
1283
+ ```typescript
1284
+ pi.registerFlag("plan", {
1285
+ description: "Start in plan mode",
1286
+ type: "boolean",
1287
+ default: false,
1288
+ });
1289
+
1290
+ // Check value
1291
+ if (pi.getFlag("--plan")) {
1292
+ // Plan mode enabled
1293
+ }
1294
+ ```
1295
+
1296
+ ### pi.exec(command, args, options?)
1297
+
1298
+ Execute a shell command.
1299
+
1300
+ ```typescript
1301
+ const result = await pi.exec("git", ["status"], { signal, timeout: 5000 });
1302
+ // result.stdout, result.stderr, result.code, result.killed
1303
+ ```
1304
+
1305
+ ### pi.getActiveTools() / pi.getAllTools() / pi.setActiveTools(names)
1306
+
1307
+ Manage active tools. This works for both built-in tools and dynamically registered tools.
1308
+
1309
+ ```typescript
1310
+ const active = pi.getActiveTools();
1311
+ const all = pi.getAllTools();
1312
+ // [{
1313
+ // name: "read",
1314
+ // description: "Read file contents...",
1315
+ // parameters: ...,
1316
+ // sourceInfo: { path: "<builtin:read>", source: "builtin", scope: "temporary", origin: "top-level" }
1317
+ // }, ...]
1318
+ const names = all.map(t => t.name);
1319
+ const builtinTools = all.filter((t) => t.sourceInfo.source === "builtin");
1320
+ const extensionTools = all.filter((t) => t.sourceInfo.source !== "builtin" && t.sourceInfo.source !== "sdk");
1321
+ pi.setActiveTools(["read", "bash"]); // Switch to read-only
1322
+ ```
1323
+
1324
+ `pi.getAllTools()` returns `name`, `description`, `parameters`, and `sourceInfo`.
1325
+
1326
+ Typical `sourceInfo.source` values:
1327
+ - `builtin` for built-in tools
1328
+ - `sdk` for tools passed via `createAgentSession({ customTools })`
1329
+ - extension source metadata for tools registered by extensions
1330
+
1331
+ ### pi.setModel(model)
1332
+
1333
+ Set the current model. Returns `false` if no API key is available for the model. See [models.md](models.md) for configuring custom models.
1334
+
1335
+ ```typescript
1336
+ const model = ctx.modelRegistry.find("anthropic", "claude-sonnet-4-5");
1337
+ if (model) {
1338
+ const success = await pi.setModel(model);
1339
+ if (!success) {
1340
+ ctx.ui.notify("No API key for this model", "error");
1341
+ }
1342
+ }
1343
+ ```
1344
+
1345
+ ### pi.getThinkingLevel() / pi.setThinkingLevel(level)
1346
+
1347
+ Get or set the thinking level. Level is clamped to model capabilities (non-reasoning models always use "off").
1348
+
1349
+ ```typescript
1350
+ const current = pi.getThinkingLevel(); // "off" | "minimal" | "low" | "medium" | "high" | "xhigh"
1351
+ pi.setThinkingLevel("high");
1352
+ ```
1353
+
1354
+ ### pi.events
1355
+
1356
+ Shared event bus for communication between extensions:
1357
+
1358
+ ```typescript
1359
+ pi.events.on("my:event", (data) => { ... });
1360
+ pi.events.emit("my:event", { ... });
1361
+ ```
1362
+
1363
+ ### pi.registerProvider(name, config)
1364
+
1365
+ Register or override a model provider dynamically. Useful for proxies, custom endpoints, or team-wide model configurations.
1366
+
1367
+ Calls made during the extension factory function are queued and applied once the runner initialises. Calls made after that — for example from a command handler following a user setup flow — take effect immediately without requiring a `/reload`.
1368
+
1369
+ ```typescript
1370
+ // Register a new provider with custom models
1371
+ pi.registerProvider("my-proxy", {
1372
+ baseUrl: "https://proxy.example.com",
1373
+ apiKey: "PROXY_API_KEY", // env var name or literal
1374
+ api: "anthropic-messages",
1375
+ models: [
1376
+ {
1377
+ id: "claude-sonnet-4-20250514",
1378
+ name: "Claude 4 Sonnet (proxy)",
1379
+ reasoning: false,
1380
+ input: ["text", "image"],
1381
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
1382
+ contextWindow: 200000,
1383
+ maxTokens: 16384
1384
+ }
1385
+ ]
1386
+ });
1387
+
1388
+ // Override baseUrl for an existing provider (keeps all models)
1389
+ pi.registerProvider("anthropic", {
1390
+ baseUrl: "https://proxy.example.com"
1391
+ });
1392
+
1393
+ // Register provider with OAuth support for /login
1394
+ pi.registerProvider("corporate-ai", {
1395
+ baseUrl: "https://ai.corp.com",
1396
+ api: "openai-responses",
1397
+ models: [...],
1398
+ oauth: {
1399
+ name: "Corporate AI (SSO)",
1400
+ async login(callbacks) {
1401
+ // Custom OAuth flow
1402
+ callbacks.onAuth({ url: "https://sso.corp.com/..." });
1403
+ const code = await callbacks.onPrompt({ message: "Enter code:" });
1404
+ return { refresh: code, access: code, expires: Date.now() + 3600000 };
1405
+ },
1406
+ async refreshToken(credentials) {
1407
+ // Refresh logic
1408
+ return credentials;
1409
+ },
1410
+ getApiKey(credentials) {
1411
+ return credentials.access;
1412
+ }
1413
+ }
1414
+ });
1415
+ ```
1416
+
1417
+ **Config options:**
1418
+ - `baseUrl` - API endpoint URL. Required when defining models.
1419
+ - `apiKey` - API key or environment variable name. Required when defining models (unless `oauth` provided).
1420
+ - `api` - API type: `"anthropic-messages"`, `"openai-completions"`, `"openai-responses"`, etc.
1421
+ - `headers` - Custom headers to include in requests.
1422
+ - `authHeader` - If true, adds `Authorization: Bearer` header automatically.
1423
+ - `models` - Array of model definitions. If provided, replaces all existing models for this provider.
1424
+ - `oauth` - OAuth provider config for `/login` support. When provided, the provider appears in the login menu.
1425
+ - `streamSimple` - Custom streaming implementation for non-standard APIs.
1426
+
1427
+ See [custom-provider.md](custom-provider.md) for advanced topics: custom streaming APIs, OAuth details, model definition reference.
1428
+
1429
+ ### pi.unregisterProvider(name)
1430
+
1431
+ Remove a previously registered provider and its models. Built-in models that were overridden by the provider are restored. Has no effect if the provider was not registered.
1432
+
1433
+ Like `registerProvider`, this takes effect immediately when called after the initial load phase, so a `/reload` is not required.
1434
+
1435
+ ```typescript
1436
+ pi.registerCommand("my-setup-teardown", {
1437
+ description: "Remove the custom proxy provider",
1438
+ handler: async (_args, _ctx) => {
1439
+ pi.unregisterProvider("my-proxy");
1440
+ },
1441
+ });
1442
+ ```
1443
+
1444
+ ## State Management
1445
+
1446
+ Extensions with state should store it in tool result `details` for proper branching support:
1447
+
1448
+ ```typescript
1449
+ export default function (pi: ExtensionAPI) {
1450
+ let items: string[] = [];
1451
+
1452
+ // Reconstruct state from session
1453
+ pi.on("session_start", async (_event, ctx) => {
1454
+ items = [];
1455
+ for (const entry of ctx.sessionManager.getBranch()) {
1456
+ if (entry.type === "message" && entry.message.role === "toolResult") {
1457
+ if (entry.message.toolName === "my_tool") {
1458
+ items = entry.message.details?.items ?? [];
1459
+ }
1460
+ }
1461
+ }
1462
+ });
1463
+
1464
+ pi.registerTool({
1465
+ name: "my_tool",
1466
+ // ...
1467
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
1468
+ items.push("new item");
1469
+ return {
1470
+ content: [{ type: "text", text: "Added" }],
1471
+ details: { items: [...items] }, // Store for reconstruction
1472
+ };
1473
+ },
1474
+ });
1475
+ }
1476
+ ```
1477
+
1478
+ ## Custom Tools
1479
+
1480
+ Register tools the LLM can call via `pi.registerTool()`. Tools appear in the system prompt and can have custom rendering.
1481
+
1482
+ Use `promptSnippet` for a short one-line entry in the `Available tools` section in the default system prompt. If omitted, custom tools are left out of that section.
1483
+
1484
+ Use `promptGuidelines` to add tool-specific bullets to the default system prompt `Guidelines` section. These bullets are included only while the tool is active (for example, after `pi.setActiveTools([...])`).
1485
+
1486
+ Note: Some models are idiots and include the @ prefix in tool path arguments. Built-in tools strip a leading @ before resolving paths. If your custom tool accepts a path, normalize a leading @ as well.
1487
+
1488
+ If your custom tool mutates files, use `withFileMutationQueue()` so it participates in the same per-file queue as built-in `edit` and `write`. This matters because tool calls run in parallel by default. Without the queue, two tools can read the same old file contents, compute different updates, and then whichever write lands last overwrites the other.
1489
+
1490
+ Example failure case: your custom tool edits `foo.ts` while built-in `edit` also changes `foo.ts` in the same assistant turn. If your tool does not participate in the queue, both can read the original `foo.ts`, apply separate changes, and one of those changes is lost.
1491
+
1492
+ Pass the real target file path to `withFileMutationQueue()`, not the raw user argument. Resolve it to an absolute path first, relative to `ctx.cwd` or your tool's working directory. For existing files, the helper canonicalizes through `realpath()`, so symlink aliases for the same file share one queue. For new files, it falls back to the resolved absolute path because there is nothing to `realpath()` yet.
1493
+
1494
+ Queue the entire mutation window on that target path. That includes read-modify-write logic, not just the final write.
1495
+
1496
+ ```typescript
1497
+ import { withFileMutationQueue } from "@mariozechner/pi-coding-agent";
1498
+ import { mkdir, readFile, writeFile } from "node:fs/promises";
1499
+ import { dirname, resolve } from "node:path";
1500
+
1501
+ async execute(_toolCallId, params, _signal, _onUpdate, ctx) {
1502
+ const absolutePath = resolve(ctx.cwd, params.path);
1503
+
1504
+ return withFileMutationQueue(absolutePath, async () => {
1505
+ await mkdir(dirname(absolutePath), { recursive: true });
1506
+ const current = await readFile(absolutePath, "utf8");
1507
+ const next = current.replace(params.oldText, params.newText);
1508
+ await writeFile(absolutePath, next, "utf8");
1509
+
1510
+ return {
1511
+ content: [{ type: "text", text: `Updated ${params.path}` }],
1512
+ details: {},
1513
+ };
1514
+ });
1515
+ }
1516
+ ```
1517
+
1518
+ ### Tool Definition
1519
+
1520
+ ```typescript
1521
+ import { Type } from "@sinclair/typebox";
1522
+ import { StringEnum } from "@mariozechner/pi-ai";
1523
+ import { Text } from "@mariozechner/pi-tui";
1524
+
1525
+ pi.registerTool({
1526
+ name: "my_tool",
1527
+ label: "My Tool",
1528
+ description: "What this tool does (shown to LLM)",
1529
+ promptSnippet: "List or add items in the project todo list",
1530
+ promptGuidelines: [
1531
+ "Use this tool for todo planning instead of direct file edits when the user asks for a task list."
1532
+ ],
1533
+ parameters: Type.Object({
1534
+ action: StringEnum(["list", "add"] as const), // Use StringEnum for Google compatibility
1535
+ text: Type.Optional(Type.String()),
1536
+ }),
1537
+ prepareArguments(args) {
1538
+ if (!args || typeof args !== "object") return args;
1539
+ const input = args as { action?: string; oldAction?: string };
1540
+ if (typeof input.oldAction === "string" && input.action === undefined) {
1541
+ return { ...input, action: input.oldAction };
1542
+ }
1543
+ return args;
1544
+ },
1545
+
1546
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
1547
+ // Check for cancellation
1548
+ if (signal?.aborted) {
1549
+ return { content: [{ type: "text", text: "Cancelled" }] };
1550
+ }
1551
+
1552
+ // Stream progress updates
1553
+ onUpdate?.({
1554
+ content: [{ type: "text", text: "Working..." }],
1555
+ details: { progress: 50 },
1556
+ });
1557
+
1558
+ // Run commands via pi.exec (captured from extension closure)
1559
+ const result = await pi.exec("some-command", [], { signal });
1560
+
1561
+ // Return result
1562
+ return {
1563
+ content: [{ type: "text", text: "Done" }], // Sent to LLM
1564
+ details: { data: result }, // For rendering & state
1565
+ };
1566
+ },
1567
+
1568
+ // Optional: Custom rendering
1569
+ renderCall(args, theme, context) { ... },
1570
+ renderResult(result, options, theme, context) { ... },
1571
+ });
1572
+ ```
1573
+
1574
+ **Signaling errors:** To mark a tool execution as failed (sets `isError: true` on the result and reports it to the LLM), throw an error from `execute`. Returning a value never sets the error flag regardless of what properties you include in the return object.
1575
+
1576
+ ```typescript
1577
+ // Correct: throw to signal an error
1578
+ async execute(toolCallId, params) {
1579
+ if (!isValid(params.input)) {
1580
+ throw new Error(`Invalid input: ${params.input}`);
1581
+ }
1582
+ return { content: [{ type: "text", text: "OK" }], details: {} };
1583
+ }
1584
+ ```
1585
+
1586
+ **Important:** Use `StringEnum` from `@mariozechner/pi-ai` for string enums. `Type.Union`/`Type.Literal` doesn't work with Google's API.
1587
+
1588
+ **Argument preparation:** `prepareArguments(args)` is optional. If defined, it runs before schema validation and before `execute()`. Use it to mimic an older accepted input shape when pi resumes an older session whose stored tool call arguments no longer match the current schema. Return the object you want validated against `parameters`. Keep the public schema strict. Do not add deprecated compatibility fields to `parameters` just to keep old resumed sessions working.
1589
+
1590
+ Example: an older session may contain an `edit` tool call with top-level `oldText` and `newText`, while the current schema only accepts `edits: [{ oldText, newText }]`.
1591
+
1592
+ ```typescript
1593
+ pi.registerTool({
1594
+ name: "edit",
1595
+ label: "Edit",
1596
+ description: "Edit a single file using exact text replacement",
1597
+ parameters: Type.Object({
1598
+ path: Type.String(),
1599
+ edits: Type.Array(
1600
+ Type.Object({
1601
+ oldText: Type.String(),
1602
+ newText: Type.String(),
1603
+ }),
1604
+ ),
1605
+ }),
1606
+ prepareArguments(args) {
1607
+ if (!args || typeof args !== "object") return args;
1608
+
1609
+ const input = args as {
1610
+ path?: string;
1611
+ edits?: Array<{ oldText: string; newText: string }>;
1612
+ oldText?: unknown;
1613
+ newText?: unknown;
1614
+ };
1615
+
1616
+ if (typeof input.oldText !== "string" || typeof input.newText !== "string") {
1617
+ return args;
1618
+ }
1619
+
1620
+ return {
1621
+ ...input,
1622
+ edits: [...(input.edits ?? []), { oldText: input.oldText, newText: input.newText }],
1623
+ };
1624
+ },
1625
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
1626
+ // params now matches the current schema
1627
+ return {
1628
+ content: [{ type: "text", text: `Applying ${params.edits.length} edit block(s)` }],
1629
+ details: {},
1630
+ };
1631
+ },
1632
+ });
1633
+ ```
1634
+
1635
+ ### Overriding Built-in Tools
1636
+
1637
+ Extensions can override built-in tools (`read`, `bash`, `edit`, `write`, `grep`, `find`, `ls`) by registering a tool with the same name. Interactive mode displays a warning when this happens.
1638
+
1639
+ ```bash
1640
+ # Extension's read tool replaces built-in read
1641
+ pi -e ./tool-override.ts
1642
+ ```
1643
+
1644
+ Alternatively, use `--no-tools` to start without any built-in tools:
1645
+ ```bash
1646
+ # No built-in tools, only extension tools
1647
+ pi --no-tools -e ./my-extension.ts
1648
+ ```
1649
+
1650
+ See [examples/extensions/tool-override.ts](../examples/extensions/tool-override.ts) for a complete example that overrides `read` with logging and access control.
1651
+
1652
+ **Rendering:** Built-in renderer inheritance is resolved per slot. Execution override and rendering override are independent. If your override omits `renderCall`, the built-in `renderCall` is used. If your override omits `renderResult`, the built-in `renderResult` is used. If your override omits both, the built-in renderer is used automatically (syntax highlighting, diffs, etc.). This lets you wrap built-in tools for logging or access control without reimplementing the UI.
1653
+
1654
+ **Prompt metadata:** `promptSnippet` and `promptGuidelines` are not inherited from the built-in tool. If your override should keep those prompt instructions, define them on the override explicitly.
1655
+
1656
+ **Your implementation must match the exact result shape**, including the `details` type. The UI and session logic depend on these shapes for rendering and state tracking.
1657
+
1658
+ Built-in tool implementations:
1659
+ - [read.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/read.ts) - `ReadToolDetails`
1660
+ - [bash.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/bash.ts) - `BashToolDetails`
1661
+ - [edit.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/edit.ts)
1662
+ - [write.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/write.ts)
1663
+ - [grep.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/grep.ts) - `GrepToolDetails`
1664
+ - [find.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/find.ts) - `FindToolDetails`
1665
+ - [ls.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/ls.ts) - `LsToolDetails`
1666
+
1667
+ ### Remote Execution
1668
+
1669
+ Built-in tools support pluggable operations for delegating to remote systems (SSH, containers, etc.):
1670
+
1671
+ ```typescript
1672
+ import { createReadTool, createBashTool, type ReadOperations } from "@mariozechner/pi-coding-agent";
1673
+
1674
+ // Create tool with custom operations
1675
+ const remoteRead = createReadTool(cwd, {
1676
+ operations: {
1677
+ readFile: (path) => sshExec(remote, `cat ${path}`),
1678
+ access: (path) => sshExec(remote, `test -r ${path}`).then(() => {}),
1679
+ }
1680
+ });
1681
+
1682
+ // Register, checking flag at execution time
1683
+ pi.registerTool({
1684
+ ...remoteRead,
1685
+ async execute(id, params, signal, onUpdate, _ctx) {
1686
+ const ssh = getSshConfig();
1687
+ if (ssh) {
1688
+ const tool = createReadTool(cwd, { operations: createRemoteOps(ssh) });
1689
+ return tool.execute(id, params, signal, onUpdate);
1690
+ }
1691
+ return localRead.execute(id, params, signal, onUpdate);
1692
+ },
1693
+ });
1694
+ ```
1695
+
1696
+ **Operations interfaces:** `ReadOperations`, `WriteOperations`, `EditOperations`, `BashOperations`, `LsOperations`, `GrepOperations`, `FindOperations`
1697
+
1698
+ For `user_bash`, extensions can reuse pi's local shell backend via `createLocalBashOperations()` instead of reimplementing local process spawning, shell resolution, and process-tree termination.
1699
+
1700
+ The bash tool also supports a spawn hook to adjust the command, cwd, or env before execution:
1701
+
1702
+ ```typescript
1703
+ import { createBashTool } from "@mariozechner/pi-coding-agent";
1704
+
1705
+ const bashTool = createBashTool(cwd, {
1706
+ spawnHook: ({ command, cwd, env }) => ({
1707
+ command: `source ~/.profile\n${command}`,
1708
+ cwd: `/mnt/sandbox${cwd}`,
1709
+ env: { ...env, CI: "1" },
1710
+ }),
1711
+ });
1712
+ ```
1713
+
1714
+ See [examples/extensions/ssh.ts](../examples/extensions/ssh.ts) for a complete SSH example with `--ssh` flag.
1715
+
1716
+ ### Output Truncation
1717
+
1718
+ **Tools MUST truncate their output** to avoid overwhelming the LLM context. Large outputs can cause:
1719
+ - Context overflow errors (prompt too long)
1720
+ - Compaction failures
1721
+ - Degraded model performance
1722
+
1723
+ The built-in limit is **50KB** (~10k tokens) and **2000 lines**, whichever is hit first. Use the exported truncation utilities:
1724
+
1725
+ ```typescript
1726
+ import {
1727
+ truncateHead, // Keep first N lines/bytes (good for file reads, search results)
1728
+ truncateTail, // Keep last N lines/bytes (good for logs, command output)
1729
+ truncateLine, // Truncate a single line to maxBytes with ellipsis
1730
+ formatSize, // Human-readable size (e.g., "50KB", "1.5MB")
1731
+ DEFAULT_MAX_BYTES, // 50KB
1732
+ DEFAULT_MAX_LINES, // 2000
1733
+ } from "@mariozechner/pi-coding-agent";
1734
+
1735
+ async execute(toolCallId, params, signal, onUpdate, ctx) {
1736
+ const output = await runCommand();
1737
+
1738
+ // Apply truncation
1739
+ const truncation = truncateHead(output, {
1740
+ maxLines: DEFAULT_MAX_LINES,
1741
+ maxBytes: DEFAULT_MAX_BYTES,
1742
+ });
1743
+
1744
+ let result = truncation.content;
1745
+
1746
+ if (truncation.truncated) {
1747
+ // Write full output to temp file
1748
+ const tempFile = writeTempFile(output);
1749
+
1750
+ // Inform the LLM where to find complete output
1751
+ result += `\n\n[Output truncated: ${truncation.outputLines} of ${truncation.totalLines} lines`;
1752
+ result += ` (${formatSize(truncation.outputBytes)} of ${formatSize(truncation.totalBytes)}).`;
1753
+ result += ` Full output saved to: ${tempFile}]`;
1754
+ }
1755
+
1756
+ return { content: [{ type: "text", text: result }] };
1757
+ }
1758
+ ```
1759
+
1760
+ **Key points:**
1761
+ - Use `truncateHead` for content where the beginning matters (search results, file reads)
1762
+ - Use `truncateTail` for content where the end matters (logs, command output)
1763
+ - Always inform the LLM when output is truncated and where to find the full version
1764
+ - Document the truncation limits in your tool's description
1765
+
1766
+ See [examples/extensions/truncated-tool.ts](../examples/extensions/truncated-tool.ts) for a complete example wrapping `rg` (ripgrep) with proper truncation.
1767
+
1768
+ ### Multiple Tools
1769
+
1770
+ One extension can register multiple tools with shared state:
1771
+
1772
+ ```typescript
1773
+ export default function (pi: ExtensionAPI) {
1774
+ let connection = null;
1775
+
1776
+ pi.registerTool({ name: "db_connect", ... });
1777
+ pi.registerTool({ name: "db_query", ... });
1778
+ pi.registerTool({ name: "db_close", ... });
1779
+
1780
+ pi.on("session_shutdown", async () => {
1781
+ connection?.close();
1782
+ });
1783
+ }
1784
+ ```
1785
+
1786
+ ### Custom Rendering
1787
+
1788
+ Tools can provide `renderCall` and `renderResult` for custom TUI display. See [tui.md](tui.md) for the full component API and [tool-execution.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/modes/interactive/components/tool-execution.ts) for how tool rows are composed.
1789
+
1790
+ Tool output is wrapped in a `Box` that handles padding and background. A defined `renderCall` or `renderResult` must return a `Component`. If a slot renderer is not defined, `tool-execution.ts` uses fallback rendering for that slot.
1791
+
1792
+ `renderCall` and `renderResult` each receive a `context` object with:
1793
+ - `args` - the current tool call arguments
1794
+ - `state` - shared row-local state across `renderCall` and `renderResult`
1795
+ - `lastComponent` - the previously returned component for that slot, if any
1796
+ - `invalidate()` - request a rerender of this tool row
1797
+ - `toolCallId`, `cwd`, `executionStarted`, `argsComplete`, `isPartial`, `expanded`, `showImages`, `isError`
1798
+
1799
+ Use `context.state` for cross-slot shared state. Keep slot-local caches on the returned component instance when you want to reuse and mutate the same component across renders.
1800
+
1801
+ #### renderCall
1802
+
1803
+ Renders the tool call or header:
1804
+
1805
+ ```typescript
1806
+ import { Text } from "@mariozechner/pi-tui";
1807
+
1808
+ renderCall(args, theme, context) {
1809
+ const text = (context.lastComponent as Text | undefined) ?? new Text("", 0, 0);
1810
+ let content = theme.fg("toolTitle", theme.bold("my_tool "));
1811
+ content += theme.fg("muted", args.action);
1812
+ if (args.text) {
1813
+ content += " " + theme.fg("dim", `"${args.text}"`);
1814
+ }
1815
+ text.setText(content);
1816
+ return text;
1817
+ }
1818
+ ```
1819
+
1820
+ #### renderResult
1821
+
1822
+ Renders the tool result or output:
1823
+
1824
+ ```typescript
1825
+ renderResult(result, { expanded, isPartial }, theme, context) {
1826
+ if (isPartial) {
1827
+ return new Text(theme.fg("warning", "Processing..."), 0, 0);
1828
+ }
1829
+
1830
+ if (result.details?.error) {
1831
+ return new Text(theme.fg("error", `Error: ${result.details.error}`), 0, 0);
1832
+ }
1833
+
1834
+ let text = theme.fg("success", "✓ Done");
1835
+ if (expanded && result.details?.items) {
1836
+ for (const item of result.details.items) {
1837
+ text += "\n " + theme.fg("dim", item);
1838
+ }
1839
+ }
1840
+ return new Text(text, 0, 0);
1841
+ }
1842
+ ```
1843
+
1844
+ If a slot intentionally has no visible content, return an empty `Component` such as an empty `Container`.
1845
+
1846
+ #### Keybinding Hints
1847
+
1848
+ Use `keyHint()` to display keybinding hints that respect the active keybinding configuration:
1849
+
1850
+ ```typescript
1851
+ import { keyHint } from "@mariozechner/pi-coding-agent";
1852
+
1853
+ renderResult(result, { expanded }, theme, context) {
1854
+ let text = theme.fg("success", "✓ Done");
1855
+ if (!expanded) {
1856
+ text += ` (${keyHint("app.tools.expand", "to expand")})`;
1857
+ }
1858
+ return new Text(text, 0, 0);
1859
+ }
1860
+ ```
1861
+
1862
+ Available functions:
1863
+ - `keyHint(keybinding, description)` - Formats a configured keybinding id such as `"app.tools.expand"` or `"tui.select.confirm"`
1864
+ - `keyText(keybinding)` - Returns the raw configured key text for a keybinding id
1865
+ - `rawKeyHint(key, description)` - Format a raw key string
1866
+
1867
+ Use namespaced keybinding ids:
1868
+ - Coding-agent ids use the `app.*` namespace, for example `app.tools.expand`, `app.editor.external`, `app.session.rename`
1869
+ - Shared TUI ids use the `tui.*` namespace, for example `tui.select.confirm`, `tui.select.cancel`, `tui.input.tab`
1870
+
1871
+ For the exhaustive list of keybinding ids and defaults, see [keybindings.md](keybindings.md). `keybindings.json` uses those same namespaced ids.
1872
+
1873
+ Custom editors and `ctx.ui.custom()` components receive `keybindings: KeybindingsManager` as an injected argument. They should use that injected manager directly instead of calling `getKeybindings()` or `setKeybindings()`.
1874
+
1875
+ #### Best Practices
1876
+
1877
+ - Use `Text` with padding `(0, 0)`. The Box handles padding.
1878
+ - Use `\n` for multi-line content.
1879
+ - Handle `isPartial` for streaming progress.
1880
+ - Support `expanded` for detail on demand.
1881
+ - Keep default view compact.
1882
+ - Read `context.args` in `renderResult` instead of copying args into `context.state`.
1883
+ - Use `context.state` only for data that must be shared across call and result slots.
1884
+ - Reuse `context.lastComponent` when the same component instance can be updated in place.
1885
+
1886
+ #### Fallback
1887
+
1888
+ If a slot renderer is not defined or throws:
1889
+ - `renderCall`: Shows the tool name
1890
+ - `renderResult`: Shows raw text from `content`
1891
+
1892
+ ## Custom UI
1893
+
1894
+ Extensions can interact with users via `ctx.ui` methods and customize how messages/tools render.
1895
+
1896
+ **For custom components, see [tui.md](tui.md)** which has copy-paste patterns for:
1897
+ - Selection dialogs (SelectList)
1898
+ - Async operations with cancel (BorderedLoader)
1899
+ - Settings toggles (SettingsList)
1900
+ - Status indicators (setStatus)
1901
+ - Working message during streaming (setWorkingMessage)
1902
+ - Widgets above/below editor (setWidget)
1903
+ - Custom footers (setFooter)
1904
+
1905
+ ### Dialogs
1906
+
1907
+ ```typescript
1908
+ // Select from options
1909
+ const choice = await ctx.ui.select("Pick one:", ["A", "B", "C"]);
1910
+
1911
+ // Confirm dialog
1912
+ const ok = await ctx.ui.confirm("Delete?", "This cannot be undone");
1913
+
1914
+ // Text input
1915
+ const name = await ctx.ui.input("Name:", "placeholder");
1916
+
1917
+ // Multi-line editor
1918
+ const text = await ctx.ui.editor("Edit:", "prefilled text");
1919
+
1920
+ // Notification (non-blocking)
1921
+ ctx.ui.notify("Done!", "info"); // "info" | "warning" | "error"
1922
+ ```
1923
+
1924
+ #### Timed Dialogs with Countdown
1925
+
1926
+ Dialogs support a `timeout` option that auto-dismisses with a live countdown display:
1927
+
1928
+ ```typescript
1929
+ // Dialog shows "Title (5s)" → "Title (4s)" → ... → auto-dismisses at 0
1930
+ const confirmed = await ctx.ui.confirm(
1931
+ "Timed Confirmation",
1932
+ "This dialog will auto-cancel in 5 seconds. Confirm?",
1933
+ { timeout: 5000 }
1934
+ );
1935
+
1936
+ if (confirmed) {
1937
+ // User confirmed
1938
+ } else {
1939
+ // User cancelled or timed out
1940
+ }
1941
+ ```
1942
+
1943
+ **Return values on timeout:**
1944
+ - `select()` returns `undefined`
1945
+ - `confirm()` returns `false`
1946
+ - `input()` returns `undefined`
1947
+
1948
+ #### Manual Dismissal with AbortSignal
1949
+
1950
+ For more control (e.g., to distinguish timeout from user cancel), use `AbortSignal`:
1951
+
1952
+ ```typescript
1953
+ const controller = new AbortController();
1954
+ const timeoutId = setTimeout(() => controller.abort(), 5000);
1955
+
1956
+ const confirmed = await ctx.ui.confirm(
1957
+ "Timed Confirmation",
1958
+ "This dialog will auto-cancel in 5 seconds. Confirm?",
1959
+ { signal: controller.signal }
1960
+ );
1961
+
1962
+ clearTimeout(timeoutId);
1963
+
1964
+ if (confirmed) {
1965
+ // User confirmed
1966
+ } else if (controller.signal.aborted) {
1967
+ // Dialog timed out
1968
+ } else {
1969
+ // User cancelled (pressed Escape or selected "No")
1970
+ }
1971
+ ```
1972
+
1973
+ See [examples/extensions/timed-confirm.ts](../examples/extensions/timed-confirm.ts) for complete examples.
1974
+
1975
+ ### Widgets, Status, and Footer
1976
+
1977
+ ```typescript
1978
+ // Status in footer (persistent until cleared)
1979
+ ctx.ui.setStatus("my-ext", "Processing...");
1980
+ ctx.ui.setStatus("my-ext", undefined); // Clear
1981
+
1982
+ // Working message (shown during streaming)
1983
+ ctx.ui.setWorkingMessage("Thinking deeply...");
1984
+ ctx.ui.setWorkingMessage(); // Restore default
1985
+
1986
+ // Widget above editor (default)
1987
+ ctx.ui.setWidget("my-widget", ["Line 1", "Line 2"]);
1988
+ // Widget below editor
1989
+ ctx.ui.setWidget("my-widget", ["Line 1", "Line 2"], { placement: "belowEditor" });
1990
+ ctx.ui.setWidget("my-widget", (tui, theme) => new Text(theme.fg("accent", "Custom"), 0, 0));
1991
+ ctx.ui.setWidget("my-widget", undefined); // Clear
1992
+
1993
+ // Custom footer (replaces built-in footer entirely)
1994
+ ctx.ui.setFooter((tui, theme) => ({
1995
+ render(width) { return [theme.fg("dim", "Custom footer")]; },
1996
+ invalidate() {},
1997
+ }));
1998
+ ctx.ui.setFooter(undefined); // Restore built-in footer
1999
+
2000
+ // Terminal title
2001
+ ctx.ui.setTitle("pi - my-project");
2002
+
2003
+ // Editor text
2004
+ ctx.ui.setEditorText("Prefill text");
2005
+ const current = ctx.ui.getEditorText();
2006
+
2007
+ // Paste into editor (triggers paste handling, including collapse for large content)
2008
+ ctx.ui.pasteToEditor("pasted content");
2009
+
2010
+ // Tool output expansion
2011
+ const wasExpanded = ctx.ui.getToolsExpanded();
2012
+ ctx.ui.setToolsExpanded(true);
2013
+ ctx.ui.setToolsExpanded(wasExpanded);
2014
+
2015
+ // Custom editor (vim mode, emacs mode, etc.)
2016
+ ctx.ui.setEditorComponent((tui, theme, keybindings) => new VimEditor(tui, theme, keybindings));
2017
+ ctx.ui.setEditorComponent(undefined); // Restore default editor
2018
+
2019
+ // Theme management (see themes.md for creating themes)
2020
+ const themes = ctx.ui.getAllThemes(); // [{ name: "dark", path: "/..." | undefined }, ...]
2021
+ const lightTheme = ctx.ui.getTheme("light"); // Load without switching
2022
+ const result = ctx.ui.setTheme("light"); // Switch by name
2023
+ if (!result.success) {
2024
+ ctx.ui.notify(`Failed: ${result.error}`, "error");
2025
+ }
2026
+ ctx.ui.setTheme(lightTheme!); // Or switch by Theme object
2027
+ ctx.ui.theme.fg("accent", "styled text"); // Access current theme
2028
+ ```
2029
+
2030
+ ### Custom Components
2031
+
2032
+ For complex UI, use `ctx.ui.custom()`. This temporarily replaces the editor with your component until `done()` is called:
2033
+
2034
+ ```typescript
2035
+ import { Text, Component } from "@mariozechner/pi-tui";
2036
+
2037
+ const result = await ctx.ui.custom<boolean>((tui, theme, keybindings, done) => {
2038
+ const text = new Text("Press Enter to confirm, Escape to cancel", 1, 1);
2039
+
2040
+ text.onKey = (key) => {
2041
+ if (key === "return") done(true);
2042
+ if (key === "escape") done(false);
2043
+ return true;
2044
+ };
2045
+
2046
+ return text;
2047
+ });
2048
+
2049
+ if (result) {
2050
+ // User pressed Enter
2051
+ }
2052
+ ```
2053
+
2054
+ The callback receives:
2055
+ - `tui` - TUI instance (for screen dimensions, focus management)
2056
+ - `theme` - Current theme for styling
2057
+ - `keybindings` - App keybinding manager (for checking shortcuts)
2058
+ - `done(value)` - Call to close component and return value
2059
+
2060
+ See [tui.md](tui.md) for the full component API.
2061
+
2062
+ #### Overlay Mode (Experimental)
2063
+
2064
+ Pass `{ overlay: true }` to render the component as a floating modal on top of existing content, without clearing the screen:
2065
+
2066
+ ```typescript
2067
+ const result = await ctx.ui.custom<string | null>(
2068
+ (tui, theme, keybindings, done) => new MyOverlayComponent({ onClose: done }),
2069
+ { overlay: true }
2070
+ );
2071
+ ```
2072
+
2073
+ For advanced positioning (anchors, margins, percentages, responsive visibility), pass `overlayOptions`. Use `onHandle` to control visibility programmatically:
2074
+
2075
+ ```typescript
2076
+ const result = await ctx.ui.custom<string | null>(
2077
+ (tui, theme, keybindings, done) => new MyOverlayComponent({ onClose: done }),
2078
+ {
2079
+ overlay: true,
2080
+ overlayOptions: { anchor: "top-right", width: "50%", margin: 2 },
2081
+ onHandle: (handle) => { /* handle.setHidden(true/false) */ }
2082
+ }
2083
+ );
2084
+ ```
2085
+
2086
+ See [tui.md](tui.md) for the full `OverlayOptions` API and [overlay-qa-tests.ts](../examples/extensions/overlay-qa-tests.ts) for examples.
2087
+
2088
+ ### Custom Editor
2089
+
2090
+ Replace the main input editor with a custom implementation (vim mode, emacs mode, etc.):
2091
+
2092
+ ```typescript
2093
+ import { CustomEditor, type ExtensionAPI } from "@mariozechner/pi-coding-agent";
2094
+ import { matchesKey } from "@mariozechner/pi-tui";
2095
+
2096
+ class VimEditor extends CustomEditor {
2097
+ private mode: "normal" | "insert" = "insert";
2098
+
2099
+ handleInput(data: string): void {
2100
+ if (matchesKey(data, "escape") && this.mode === "insert") {
2101
+ this.mode = "normal";
2102
+ return;
2103
+ }
2104
+ if (this.mode === "normal" && data === "i") {
2105
+ this.mode = "insert";
2106
+ return;
2107
+ }
2108
+ super.handleInput(data); // App keybindings + text editing
2109
+ }
2110
+ }
2111
+
2112
+ export default function (pi: ExtensionAPI) {
2113
+ pi.on("session_start", (_event, ctx) => {
2114
+ ctx.ui.setEditorComponent((_tui, theme, keybindings) =>
2115
+ new VimEditor(theme, keybindings)
2116
+ );
2117
+ });
2118
+ }
2119
+ ```
2120
+
2121
+ **Key points:**
2122
+ - Extend `CustomEditor` (not base `Editor`) to get app keybindings (escape to abort, ctrl+d, model switching)
2123
+ - Call `super.handleInput(data)` for keys you don't handle
2124
+ - Factory receives `theme` and `keybindings` from the app
2125
+ - Pass `undefined` to restore default: `ctx.ui.setEditorComponent(undefined)`
2126
+
2127
+ See [tui.md](tui.md) Pattern 7 for a complete example with mode indicator.
2128
+
2129
+ ### Message Rendering
2130
+
2131
+ Register a custom renderer for messages with your `customType`:
2132
+
2133
+ ```typescript
2134
+ import { Text } from "@mariozechner/pi-tui";
2135
+
2136
+ pi.registerMessageRenderer("my-extension", (message, options, theme) => {
2137
+ const { expanded } = options;
2138
+ let text = theme.fg("accent", `[${message.customType}] `);
2139
+ text += message.content;
2140
+
2141
+ if (expanded && message.details) {
2142
+ text += "\n" + theme.fg("dim", JSON.stringify(message.details, null, 2));
2143
+ }
2144
+
2145
+ return new Text(text, 0, 0);
2146
+ });
2147
+ ```
2148
+
2149
+ Messages are sent via `pi.sendMessage()`:
2150
+
2151
+ ```typescript
2152
+ pi.sendMessage({
2153
+ customType: "my-extension", // Matches registerMessageRenderer
2154
+ content: "Status update",
2155
+ display: true, // Show in TUI
2156
+ details: { ... }, // Available in renderer
2157
+ });
2158
+ ```
2159
+
2160
+ ### Theme Colors
2161
+
2162
+ All render functions receive a `theme` object. See [themes.md](themes.md) for creating custom themes and the full color palette.
2163
+
2164
+ ```typescript
2165
+ // Foreground colors
2166
+ theme.fg("toolTitle", text) // Tool names
2167
+ theme.fg("accent", text) // Highlights
2168
+ theme.fg("success", text) // Success (green)
2169
+ theme.fg("error", text) // Errors (red)
2170
+ theme.fg("warning", text) // Warnings (yellow)
2171
+ theme.fg("muted", text) // Secondary text
2172
+ theme.fg("dim", text) // Tertiary text
2173
+
2174
+ // Text styles
2175
+ theme.bold(text)
2176
+ theme.italic(text)
2177
+ theme.strikethrough(text)
2178
+ ```
2179
+
2180
+ For syntax highlighting in custom tool renderers:
2181
+
2182
+ ```typescript
2183
+ import { highlightCode, getLanguageFromPath } from "@mariozechner/pi-coding-agent";
2184
+
2185
+ // Highlight code with explicit language
2186
+ const highlighted = highlightCode("const x = 1;", "typescript", theme);
2187
+
2188
+ // Auto-detect language from file path
2189
+ const lang = getLanguageFromPath("/path/to/file.rs"); // "rust"
2190
+ const highlighted = highlightCode(code, lang, theme);
2191
+ ```
2192
+
2193
+ ## Error Handling
2194
+
2195
+ - Extension errors are logged, agent continues
2196
+ - `tool_call` errors block the tool (fail-safe)
2197
+ - Tool `execute` errors must be signaled by throwing; the thrown error is caught, reported to the LLM with `isError: true`, and execution continues
2198
+
2199
+ ## Mode Behavior
2200
+
2201
+ | Mode | UI Methods | Notes |
2202
+ |------|-----------|-------|
2203
+ | Interactive | Full TUI | Normal operation |
2204
+ | RPC (`--mode rpc`) | JSON protocol | Host handles UI, see [rpc.md](rpc.md) |
2205
+ | JSON (`--mode json`) | No-op | Event stream to stdout, see [json.md](json.md) |
2206
+ | Print (`-p`) | No-op | Extensions run but can't prompt |
2207
+
2208
+ In non-interactive modes, check `ctx.hasUI` before using UI methods.
2209
+
2210
+ ## Examples Reference
2211
+
2212
+ All examples in [examples/extensions/](../examples/extensions/).
2213
+
2214
+ | Example | Description | Key APIs |
2215
+ |---------|-------------|----------|
2216
+ | **Tools** |||
2217
+ | `hello.ts` | Minimal tool registration | `registerTool` |
2218
+ | `question.ts` | Tool with user interaction | `registerTool`, `ui.select` |
2219
+ | `questionnaire.ts` | Multi-step wizard tool | `registerTool`, `ui.custom` |
2220
+ | `todo.ts` | Stateful tool with persistence | `registerTool`, `appendEntry`, `renderResult`, session events |
2221
+ | `dynamic-tools.ts` | Register tools after startup and during commands | `registerTool`, `session_start`, `registerCommand` |
2222
+ | `truncated-tool.ts` | Output truncation example | `registerTool`, `truncateHead` |
2223
+ | `tool-override.ts` | Override built-in read tool | `registerTool` (same name as built-in) |
2224
+ | **Commands** |||
2225
+ | `pirate.ts` | Modify system prompt per-turn | `registerCommand`, `before_agent_start` |
2226
+ | `summarize.ts` | Conversation summary command | `registerCommand`, `ui.custom` |
2227
+ | `handoff.ts` | Cross-provider model handoff | `registerCommand`, `ui.editor`, `ui.custom` |
2228
+ | `qna.ts` | Q&A with custom UI | `registerCommand`, `ui.custom`, `setEditorText` |
2229
+ | `send-user-message.ts` | Inject user messages | `registerCommand`, `sendUserMessage` |
2230
+ | `reload-runtime.ts` | Reload command and LLM tool handoff | `registerCommand`, `ctx.reload()`, `sendUserMessage` |
2231
+ | `shutdown-command.ts` | Graceful shutdown command | `registerCommand`, `shutdown()` |
2232
+ | **Events & Gates** |||
2233
+ | `permission-gate.ts` | Block dangerous commands | `on("tool_call")`, `ui.confirm` |
2234
+ | `protected-paths.ts` | Block writes to specific paths | `on("tool_call")` |
2235
+ | `confirm-destructive.ts` | Confirm session changes | `on("session_before_switch")`, `on("session_before_fork")` |
2236
+ | `dirty-repo-guard.ts` | Warn on dirty git repo | `on("session_before_*")`, `exec` |
2237
+ | `input-transform.ts` | Transform user input | `on("input")` |
2238
+ | `model-status.ts` | React to model changes | `on("model_select")`, `setStatus` |
2239
+ | `provider-payload.ts` | Inspect or patch provider payloads | `on("before_provider_request")` |
2240
+ | `system-prompt-header.ts` | Display system prompt info | `on("agent_start")`, `getSystemPrompt` |
2241
+ | `claude-rules.ts` | Load rules from files | `on("session_start")`, `on("before_agent_start")` |
2242
+ | `file-trigger.ts` | File watcher triggers messages | `sendMessage` |
2243
+ | **Compaction & Sessions** |||
2244
+ | `custom-compaction.ts` | Custom compaction summary | `on("session_before_compact")` |
2245
+ | `trigger-compact.ts` | Trigger compaction manually | `compact()` |
2246
+ | `git-checkpoint.ts` | Git stash on turns | `on("turn_start")`, `on("session_before_fork")`, `exec` |
2247
+ | `auto-commit-on-exit.ts` | Commit on shutdown | `on("session_shutdown")`, `exec` |
2248
+ | **UI Components** |||
2249
+ | `status-line.ts` | Footer status indicator | `setStatus`, session events |
2250
+ | `custom-footer.ts` | Replace footer entirely | `registerCommand`, `setFooter` |
2251
+ | `custom-header.ts` | Replace startup header | `on("session_start")`, `setHeader` |
2252
+ | `modal-editor.ts` | Vim-style modal editor | `setEditorComponent`, `CustomEditor` |
2253
+ | `rainbow-editor.ts` | Custom editor styling | `setEditorComponent` |
2254
+ | `widget-placement.ts` | Widget above/below editor | `setWidget` |
2255
+ | `overlay-test.ts` | Overlay components | `ui.custom` with overlay options |
2256
+ | `overlay-qa-tests.ts` | Comprehensive overlay tests | `ui.custom`, all overlay options |
2257
+ | `notify.ts` | Simple notifications | `ui.notify` |
2258
+ | `timed-confirm.ts` | Dialogs with timeout | `ui.confirm` with timeout/signal |
2259
+ | `mac-system-theme.ts` | Auto-switch theme | `setTheme`, `exec` |
2260
+ | **Complex Extensions** |||
2261
+ | `plan-mode/` | Full plan mode implementation | All event types, `registerCommand`, `registerShortcut`, `registerFlag`, `setStatus`, `setWidget`, `sendMessage`, `setActiveTools` |
2262
+ | `preset.ts` | Saveable presets (model, tools, thinking) | `registerCommand`, `registerShortcut`, `registerFlag`, `setModel`, `setActiveTools`, `setThinkingLevel`, `appendEntry` |
2263
+ | `tools.ts` | Toggle tools on/off UI | `registerCommand`, `setActiveTools`, `SettingsList`, session events |
2264
+ | **Remote & Sandbox** |||
2265
+ | `ssh.ts` | SSH remote execution | `registerFlag`, `on("user_bash")`, `on("before_agent_start")`, tool operations |
2266
+ | `interactive-shell.ts` | Persistent shell session | `on("user_bash")` |
2267
+ | `sandbox/` | Sandboxed tool execution | Tool operations |
2268
+ | `subagent/` | Spawn sub-agents | `registerTool`, `exec` |
2269
+ | **Games** |||
2270
+ | `snake.ts` | Snake game | `registerCommand`, `ui.custom`, keyboard handling |
2271
+ | `space-invaders.ts` | Space Invaders game | `registerCommand`, `ui.custom` |
2272
+ | `doom-overlay/` | Doom in overlay | `ui.custom` with overlay |
2273
+ | **Providers** |||
2274
+ | `custom-provider-anthropic/` | Custom Anthropic proxy | `registerProvider` |
2275
+ | `custom-provider-gitlab-duo/` | GitLab Duo integration | `registerProvider` with OAuth |
2276
+ | **Messages & Communication** |||
2277
+ | `message-renderer.ts` | Custom message rendering | `registerMessageRenderer`, `sendMessage` |
2278
+ | `event-bus.ts` | Inter-extension events | `pi.events` |
2279
+ | **Session Metadata** |||
2280
+ | `session-name.ts` | Name sessions for selector | `setSessionName`, `getSessionName` |
2281
+ | `bookmark.ts` | Bookmark entries for /tree | `setLabel` |
2282
+ | **Misc** |||
2283
+ | `antigravity-image-gen.ts` | Image generation tool | `registerTool`, Google Antigravity |
2284
+ | `inline-bash.ts` | Inline bash in tool calls | `on("tool_call")` |
2285
+ | `bash-spawn-hook.ts` | Adjust bash command, cwd, and env before execution | `createBashTool`, `spawnHook` |
2286
+ | `with-deps/` | Extension with npm dependencies | Package structure with `package.json` |