@vaclav-synacek/pi-coding-agent-termux 0.45.7

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 (478) hide show
  1. package/CHANGELOG.md +1961 -0
  2. package/README.md +1392 -0
  3. package/dist/cli/args.d.ts +42 -0
  4. package/dist/cli/args.d.ts.map +1 -0
  5. package/dist/cli/args.js +248 -0
  6. package/dist/cli/args.js.map +1 -0
  7. package/dist/cli/file-processor.d.ts +15 -0
  8. package/dist/cli/file-processor.d.ts.map +1 -0
  9. package/dist/cli/file-processor.js +79 -0
  10. package/dist/cli/file-processor.js.map +1 -0
  11. package/dist/cli/list-models.d.ts +9 -0
  12. package/dist/cli/list-models.d.ts.map +1 -0
  13. package/dist/cli/list-models.js +92 -0
  14. package/dist/cli/list-models.js.map +1 -0
  15. package/dist/cli/session-picker.d.ts +9 -0
  16. package/dist/cli/session-picker.d.ts.map +1 -0
  17. package/dist/cli/session-picker.js +32 -0
  18. package/dist/cli/session-picker.js.map +1 -0
  19. package/dist/cli.d.ts +3 -0
  20. package/dist/cli.d.ts.map +1 -0
  21. package/dist/cli.js +10 -0
  22. package/dist/cli.js.map +1 -0
  23. package/dist/config.d.ts +61 -0
  24. package/dist/config.d.ts.map +1 -0
  25. package/dist/config.js +141 -0
  26. package/dist/config.js.map +1 -0
  27. package/dist/core/agent-session.d.ts +523 -0
  28. package/dist/core/agent-session.d.ts.map +1 -0
  29. package/dist/core/agent-session.js +1795 -0
  30. package/dist/core/agent-session.js.map +1 -0
  31. package/dist/core/auth-storage.d.ts +112 -0
  32. package/dist/core/auth-storage.d.ts.map +1 -0
  33. package/dist/core/auth-storage.js +297 -0
  34. package/dist/core/auth-storage.js.map +1 -0
  35. package/dist/core/bash-executor.d.ts +47 -0
  36. package/dist/core/bash-executor.d.ts.map +1 -0
  37. package/dist/core/bash-executor.js +211 -0
  38. package/dist/core/bash-executor.js.map +1 -0
  39. package/dist/core/compaction/branch-summarization.d.ts +84 -0
  40. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  41. package/dist/core/compaction/branch-summarization.js +235 -0
  42. package/dist/core/compaction/branch-summarization.js.map +1 -0
  43. package/dist/core/compaction/compaction.d.ts +110 -0
  44. package/dist/core/compaction/compaction.d.ts.map +1 -0
  45. package/dist/core/compaction/compaction.js +559 -0
  46. package/dist/core/compaction/compaction.js.map +1 -0
  47. package/dist/core/compaction/index.d.ts +7 -0
  48. package/dist/core/compaction/index.d.ts.map +1 -0
  49. package/dist/core/compaction/index.js +7 -0
  50. package/dist/core/compaction/index.js.map +1 -0
  51. package/dist/core/compaction/utils.d.ts +35 -0
  52. package/dist/core/compaction/utils.d.ts.map +1 -0
  53. package/dist/core/compaction/utils.js +138 -0
  54. package/dist/core/compaction/utils.js.map +1 -0
  55. package/dist/core/event-bus.d.ts +9 -0
  56. package/dist/core/event-bus.d.ts.map +1 -0
  57. package/dist/core/event-bus.js +25 -0
  58. package/dist/core/event-bus.js.map +1 -0
  59. package/dist/core/exec.d.ts +29 -0
  60. package/dist/core/exec.d.ts.map +1 -0
  61. package/dist/core/exec.js +71 -0
  62. package/dist/core/exec.js.map +1 -0
  63. package/dist/core/export-html/index.d.ts +17 -0
  64. package/dist/core/export-html/index.d.ts.map +1 -0
  65. package/dist/core/export-html/index.js +193 -0
  66. package/dist/core/export-html/index.js.map +1 -0
  67. package/dist/core/export-html/template.css +910 -0
  68. package/dist/core/export-html/template.html +54 -0
  69. package/dist/core/export-html/template.js +1329 -0
  70. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  71. package/dist/core/export-html/vendor/marked.min.js +6 -0
  72. package/dist/core/extensions/index.d.ts +10 -0
  73. package/dist/core/extensions/index.d.ts.map +1 -0
  74. package/dist/core/extensions/index.js +9 -0
  75. package/dist/core/extensions/index.js.map +1 -0
  76. package/dist/core/extensions/loader.d.ts +25 -0
  77. package/dist/core/extensions/loader.d.ts.map +1 -0
  78. package/dist/core/extensions/loader.js +383 -0
  79. package/dist/core/extensions/loader.js.map +1 -0
  80. package/dist/core/extensions/runner.d.ts +89 -0
  81. package/dist/core/extensions/runner.d.ts.map +1 -0
  82. package/dist/core/extensions/runner.js +406 -0
  83. package/dist/core/extensions/runner.js.map +1 -0
  84. package/dist/core/extensions/types.d.ts +654 -0
  85. package/dist/core/extensions/types.d.ts.map +1 -0
  86. package/dist/core/extensions/types.js +32 -0
  87. package/dist/core/extensions/types.js.map +1 -0
  88. package/dist/core/extensions/wrapper.d.ts +27 -0
  89. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  90. package/dist/core/extensions/wrapper.js +102 -0
  91. package/dist/core/extensions/wrapper.js.map +1 -0
  92. package/dist/core/footer-data-provider.d.ts +25 -0
  93. package/dist/core/footer-data-provider.d.ts.map +1 -0
  94. package/dist/core/footer-data-provider.js +121 -0
  95. package/dist/core/footer-data-provider.js.map +1 -0
  96. package/dist/core/index.d.ts +9 -0
  97. package/dist/core/index.d.ts.map +1 -0
  98. package/dist/core/index.js +9 -0
  99. package/dist/core/index.js.map +1 -0
  100. package/dist/core/keybindings.d.ts +59 -0
  101. package/dist/core/keybindings.d.ts.map +1 -0
  102. package/dist/core/keybindings.js +151 -0
  103. package/dist/core/keybindings.js.map +1 -0
  104. package/dist/core/messages.d.ts +77 -0
  105. package/dist/core/messages.d.ts.map +1 -0
  106. package/dist/core/messages.js +123 -0
  107. package/dist/core/messages.js.map +1 -0
  108. package/dist/core/model-registry.d.ts +57 -0
  109. package/dist/core/model-registry.d.ts.map +1 -0
  110. package/dist/core/model-registry.js +314 -0
  111. package/dist/core/model-registry.js.map +1 -0
  112. package/dist/core/model-resolver.d.ts +76 -0
  113. package/dist/core/model-resolver.d.ts.map +1 -0
  114. package/dist/core/model-resolver.js +308 -0
  115. package/dist/core/model-resolver.js.map +1 -0
  116. package/dist/core/prompt-templates.d.ts +40 -0
  117. package/dist/core/prompt-templates.d.ts.map +1 -0
  118. package/dist/core/prompt-templates.js +197 -0
  119. package/dist/core/prompt-templates.js.map +1 -0
  120. package/dist/core/sdk.d.ts +181 -0
  121. package/dist/core/sdk.d.ts.map +1 -0
  122. package/dist/core/sdk.js +466 -0
  123. package/dist/core/sdk.js.map +1 -0
  124. package/dist/core/session-manager.d.ts +313 -0
  125. package/dist/core/session-manager.d.ts.map +1 -0
  126. package/dist/core/session-manager.js +996 -0
  127. package/dist/core/session-manager.js.map +1 -0
  128. package/dist/core/settings-manager.d.ts +138 -0
  129. package/dist/core/settings-manager.d.ts.map +1 -0
  130. package/dist/core/settings-manager.js +327 -0
  131. package/dist/core/settings-manager.js.map +1 -0
  132. package/dist/core/skills.d.ts +50 -0
  133. package/dist/core/skills.d.ts.map +1 -0
  134. package/dist/core/skills.js +338 -0
  135. package/dist/core/skills.js.map +1 -0
  136. package/dist/core/system-prompt.d.ts +48 -0
  137. package/dist/core/system-prompt.d.ts.map +1 -0
  138. package/dist/core/system-prompt.js +224 -0
  139. package/dist/core/system-prompt.js.map +1 -0
  140. package/dist/core/timings.d.ts +7 -0
  141. package/dist/core/timings.d.ts.map +1 -0
  142. package/dist/core/timings.js +25 -0
  143. package/dist/core/timings.js.map +1 -0
  144. package/dist/core/tools/bash.d.ts +42 -0
  145. package/dist/core/tools/bash.d.ts.map +1 -0
  146. package/dist/core/tools/bash.js +223 -0
  147. package/dist/core/tools/bash.js.map +1 -0
  148. package/dist/core/tools/edit-diff.d.ts +33 -0
  149. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  150. package/dist/core/tools/edit-diff.js +171 -0
  151. package/dist/core/tools/edit-diff.js.map +1 -0
  152. package/dist/core/tools/edit.d.ts +37 -0
  153. package/dist/core/tools/edit.d.ts.map +1 -0
  154. package/dist/core/tools/edit.js +143 -0
  155. package/dist/core/tools/edit.js.map +1 -0
  156. package/dist/core/tools/find.d.ts +37 -0
  157. package/dist/core/tools/find.d.ts.map +1 -0
  158. package/dist/core/tools/find.js +206 -0
  159. package/dist/core/tools/find.js.map +1 -0
  160. package/dist/core/tools/grep.d.ts +43 -0
  161. package/dist/core/tools/grep.d.ts.map +1 -0
  162. package/dist/core/tools/grep.js +239 -0
  163. package/dist/core/tools/grep.js.map +1 -0
  164. package/dist/core/tools/index.d.ts +70 -0
  165. package/dist/core/tools/index.d.ts.map +1 -0
  166. package/dist/core/tools/index.js +56 -0
  167. package/dist/core/tools/index.js.map +1 -0
  168. package/dist/core/tools/ls.d.ts +38 -0
  169. package/dist/core/tools/ls.d.ts.map +1 -0
  170. package/dist/core/tools/ls.js +118 -0
  171. package/dist/core/tools/ls.js.map +1 -0
  172. package/dist/core/tools/path-utils.d.ts +8 -0
  173. package/dist/core/tools/path-utils.d.ts.map +1 -0
  174. package/dist/core/tools/path-utils.js +53 -0
  175. package/dist/core/tools/path-utils.js.map +1 -0
  176. package/dist/core/tools/read.d.ts +37 -0
  177. package/dist/core/tools/read.d.ts.map +1 -0
  178. package/dist/core/tools/read.js +165 -0
  179. package/dist/core/tools/read.js.map +1 -0
  180. package/dist/core/tools/truncate.d.ts +70 -0
  181. package/dist/core/tools/truncate.d.ts.map +1 -0
  182. package/dist/core/tools/truncate.js +205 -0
  183. package/dist/core/tools/truncate.js.map +1 -0
  184. package/dist/core/tools/write.d.ts +27 -0
  185. package/dist/core/tools/write.d.ts.map +1 -0
  186. package/dist/core/tools/write.js +78 -0
  187. package/dist/core/tools/write.js.map +1 -0
  188. package/dist/index.d.ts +19 -0
  189. package/dist/index.d.ts.map +1 -0
  190. package/dist/index.js +35 -0
  191. package/dist/index.js.map +1 -0
  192. package/dist/main.d.ts +8 -0
  193. package/dist/main.d.ts.map +1 -0
  194. package/dist/main.js +354 -0
  195. package/dist/main.js.map +1 -0
  196. package/dist/migrations.d.ts +33 -0
  197. package/dist/migrations.d.ts.map +1 -0
  198. package/dist/migrations.js +261 -0
  199. package/dist/migrations.js.map +1 -0
  200. package/dist/modes/index.d.ts +9 -0
  201. package/dist/modes/index.d.ts.map +1 -0
  202. package/dist/modes/index.js +8 -0
  203. package/dist/modes/index.js.map +1 -0
  204. package/dist/modes/interactive/components/armin.d.ts +34 -0
  205. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  206. package/dist/modes/interactive/components/armin.js +333 -0
  207. package/dist/modes/interactive/components/armin.js.map +1 -0
  208. package/dist/modes/interactive/components/assistant-message.d.ts +15 -0
  209. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  210. package/dist/modes/interactive/components/assistant-message.js +89 -0
  211. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  212. package/dist/modes/interactive/components/bash-execution.d.ts +35 -0
  213. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  214. package/dist/modes/interactive/components/bash-execution.js +161 -0
  215. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  216. package/dist/modes/interactive/components/bordered-loader.d.ts +12 -0
  217. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  218. package/dist/modes/interactive/components/bordered-loader.js +30 -0
  219. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  220. package/dist/modes/interactive/components/branch-summary-message.d.ts +15 -0
  221. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  222. package/dist/modes/interactive/components/branch-summary-message.js +39 -0
  223. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  224. package/dist/modes/interactive/components/compaction-summary-message.d.ts +15 -0
  225. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  226. package/dist/modes/interactive/components/compaction-summary-message.js +40 -0
  227. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  228. package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
  229. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  230. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  231. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  232. package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
  233. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  234. package/dist/modes/interactive/components/custom-editor.js +69 -0
  235. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  236. package/dist/modes/interactive/components/custom-message.d.ts +19 -0
  237. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  238. package/dist/modes/interactive/components/custom-message.js +84 -0
  239. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  240. package/dist/modes/interactive/components/diff.d.ts +12 -0
  241. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  242. package/dist/modes/interactive/components/diff.js +133 -0
  243. package/dist/modes/interactive/components/diff.js.map +1 -0
  244. package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
  245. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  246. package/dist/modes/interactive/components/dynamic-border.js +21 -0
  247. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  248. package/dist/modes/interactive/components/extension-editor.d.ts +15 -0
  249. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  250. package/dist/modes/interactive/components/extension-editor.js +96 -0
  251. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  252. package/dist/modes/interactive/components/extension-input.d.ts +20 -0
  253. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  254. package/dist/modes/interactive/components/extension-input.js +51 -0
  255. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  256. package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
  257. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  258. package/dist/modes/interactive/components/extension-selector.js +73 -0
  259. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  260. package/dist/modes/interactive/components/footer.d.ts +26 -0
  261. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  262. package/dist/modes/interactive/components/footer.js +207 -0
  263. package/dist/modes/interactive/components/footer.js.map +1 -0
  264. package/dist/modes/interactive/components/index.d.ts +29 -0
  265. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  266. package/dist/modes/interactive/components/index.js +30 -0
  267. package/dist/modes/interactive/components/index.js.map +1 -0
  268. package/dist/modes/interactive/components/login-dialog.d.ts +39 -0
  269. package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
  270. package/dist/modes/interactive/components/login-dialog.js +135 -0
  271. package/dist/modes/interactive/components/login-dialog.js.map +1 -0
  272. package/dist/modes/interactive/components/model-selector.d.ts +35 -0
  273. package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
  274. package/dist/modes/interactive/components/model-selector.js +211 -0
  275. package/dist/modes/interactive/components/model-selector.js.map +1 -0
  276. package/dist/modes/interactive/components/oauth-selector.d.ts +19 -0
  277. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
  278. package/dist/modes/interactive/components/oauth-selector.js +98 -0
  279. package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
  280. package/dist/modes/interactive/components/scoped-models-selector.d.ts +46 -0
  281. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
  282. package/dist/modes/interactive/components/scoped-models-selector.js +258 -0
  283. package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
  284. package/dist/modes/interactive/components/session-selector.d.ts +44 -0
  285. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  286. package/dist/modes/interactive/components/session-selector.js +311 -0
  287. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  288. package/dist/modes/interactive/components/settings-selector.d.ts +43 -0
  289. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  290. package/dist/modes/interactive/components/settings-selector.js +219 -0
  291. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  292. package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
  293. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  294. package/dist/modes/interactive/components/show-images-selector.js +35 -0
  295. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  296. package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
  297. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  298. package/dist/modes/interactive/components/theme-selector.js +46 -0
  299. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  300. package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
  301. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  302. package/dist/modes/interactive/components/thinking-selector.js +47 -0
  303. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  304. package/dist/modes/interactive/components/tool-execution.d.ts +70 -0
  305. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  306. package/dist/modes/interactive/components/tool-execution.js +606 -0
  307. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  308. package/dist/modes/interactive/components/tree-selector.d.ts +52 -0
  309. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  310. package/dist/modes/interactive/components/tree-selector.js +745 -0
  311. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  312. package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
  313. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  314. package/dist/modes/interactive/components/user-message-selector.js +113 -0
  315. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  316. package/dist/modes/interactive/components/user-message.d.ts +8 -0
  317. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  318. package/dist/modes/interactive/components/user-message.js +16 -0
  319. package/dist/modes/interactive/components/user-message.js.map +1 -0
  320. package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
  321. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  322. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  323. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  324. package/dist/modes/interactive/interactive-mode.d.ts +261 -0
  325. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  326. package/dist/modes/interactive/interactive-mode.js +3194 -0
  327. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  328. package/dist/modes/interactive/theme/dark.json +85 -0
  329. package/dist/modes/interactive/theme/light.json +84 -0
  330. package/dist/modes/interactive/theme/theme-schema.json +308 -0
  331. package/dist/modes/interactive/theme/theme.d.ts +71 -0
  332. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  333. package/dist/modes/interactive/theme/theme.js +893 -0
  334. package/dist/modes/interactive/theme/theme.js.map +1 -0
  335. package/dist/modes/print-mode.d.ts +28 -0
  336. package/dist/modes/print-mode.d.ts.map +1 -0
  337. package/dist/modes/print-mode.js +140 -0
  338. package/dist/modes/print-mode.js.map +1 -0
  339. package/dist/modes/rpc/rpc-client.d.ts +209 -0
  340. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  341. package/dist/modes/rpc/rpc-client.js +392 -0
  342. package/dist/modes/rpc/rpc-client.js.map +1 -0
  343. package/dist/modes/rpc/rpc-mode.d.ts +20 -0
  344. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  345. package/dist/modes/rpc/rpc-mode.js +486 -0
  346. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  347. package/dist/modes/rpc/rpc-types.d.ts +372 -0
  348. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  349. package/dist/modes/rpc/rpc-types.js +8 -0
  350. package/dist/modes/rpc/rpc-types.js.map +1 -0
  351. package/dist/utils/changelog.d.ts +21 -0
  352. package/dist/utils/changelog.d.ts.map +1 -0
  353. package/dist/utils/changelog.js +87 -0
  354. package/dist/utils/changelog.js.map +1 -0
  355. package/dist/utils/clipboard-image.d.ts +11 -0
  356. package/dist/utils/clipboard-image.d.ts.map +1 -0
  357. package/dist/utils/clipboard-image.js +129 -0
  358. package/dist/utils/clipboard-image.js.map +1 -0
  359. package/dist/utils/clipboard.d.ts +2 -0
  360. package/dist/utils/clipboard.d.ts.map +1 -0
  361. package/dist/utils/clipboard.js +73 -0
  362. package/dist/utils/clipboard.js.map +1 -0
  363. package/dist/utils/image-convert.d.ts +9 -0
  364. package/dist/utils/image-convert.d.ts.map +1 -0
  365. package/dist/utils/image-convert.js +31 -0
  366. package/dist/utils/image-convert.js.map +1 -0
  367. package/dist/utils/image-resize.d.ts +36 -0
  368. package/dist/utils/image-resize.d.ts.map +1 -0
  369. package/dist/utils/image-resize.js +188 -0
  370. package/dist/utils/image-resize.js.map +1 -0
  371. package/dist/utils/mime.d.ts +2 -0
  372. package/dist/utils/mime.d.ts.map +1 -0
  373. package/dist/utils/mime.js +26 -0
  374. package/dist/utils/mime.js.map +1 -0
  375. package/dist/utils/shell.d.ts +26 -0
  376. package/dist/utils/shell.d.ts.map +1 -0
  377. package/dist/utils/shell.js +151 -0
  378. package/dist/utils/shell.js.map +1 -0
  379. package/dist/utils/tools-manager.d.ts +3 -0
  380. package/dist/utils/tools-manager.d.ts.map +1 -0
  381. package/dist/utils/tools-manager.js +187 -0
  382. package/dist/utils/tools-manager.js.map +1 -0
  383. package/dist/utils/vips.d.ts +11 -0
  384. package/dist/utils/vips.d.ts.map +1 -0
  385. package/dist/utils/vips.js +35 -0
  386. package/dist/utils/vips.js.map +1 -0
  387. package/docs/compaction.md +388 -0
  388. package/docs/extensions.md +1524 -0
  389. package/docs/rpc.md +1046 -0
  390. package/docs/sdk.md +1024 -0
  391. package/docs/session.md +255 -0
  392. package/docs/skills.md +317 -0
  393. package/docs/theme.md +617 -0
  394. package/docs/tree.md +201 -0
  395. package/docs/tui.md +797 -0
  396. package/examples/README.md +24 -0
  397. package/examples/extensions/README.md +168 -0
  398. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  399. package/examples/extensions/chalk-logger.ts +26 -0
  400. package/examples/extensions/claude-rules.ts +86 -0
  401. package/examples/extensions/confirm-destructive.ts +59 -0
  402. package/examples/extensions/custom-compaction.ts +114 -0
  403. package/examples/extensions/custom-footer.ts +64 -0
  404. package/examples/extensions/custom-header.ts +72 -0
  405. package/examples/extensions/dirty-repo-guard.ts +56 -0
  406. package/examples/extensions/doom-overlay/README.md +46 -0
  407. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  408. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  409. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  410. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  411. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  412. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  413. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  414. package/examples/extensions/doom-overlay/index.ts +74 -0
  415. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  416. package/examples/extensions/file-trigger.ts +41 -0
  417. package/examples/extensions/git-checkpoint.ts +53 -0
  418. package/examples/extensions/handoff.ts +150 -0
  419. package/examples/extensions/hello.ts +25 -0
  420. package/examples/extensions/interactive-shell.ts +196 -0
  421. package/examples/extensions/mac-system-theme.ts +47 -0
  422. package/examples/extensions/modal-editor.ts +85 -0
  423. package/examples/extensions/model-status.ts +31 -0
  424. package/examples/extensions/notify.ts +25 -0
  425. package/examples/extensions/overlay-qa-tests.ts +881 -0
  426. package/examples/extensions/overlay-test.ts +145 -0
  427. package/examples/extensions/permission-gate.ts +34 -0
  428. package/examples/extensions/pirate.ts +47 -0
  429. package/examples/extensions/plan-mode/README.md +65 -0
  430. package/examples/extensions/plan-mode/index.ts +340 -0
  431. package/examples/extensions/plan-mode/utils.ts +168 -0
  432. package/examples/extensions/preset.ts +398 -0
  433. package/examples/extensions/protected-paths.ts +30 -0
  434. package/examples/extensions/qna.ts +119 -0
  435. package/examples/extensions/question.ts +277 -0
  436. package/examples/extensions/questionnaire.ts +427 -0
  437. package/examples/extensions/rainbow-editor.ts +95 -0
  438. package/examples/extensions/sandbox/index.ts +318 -0
  439. package/examples/extensions/sandbox/package-lock.json +92 -0
  440. package/examples/extensions/sandbox/package.json +19 -0
  441. package/examples/extensions/send-user-message.ts +97 -0
  442. package/examples/extensions/shutdown-command.ts +63 -0
  443. package/examples/extensions/snake.ts +343 -0
  444. package/examples/extensions/ssh.ts +220 -0
  445. package/examples/extensions/status-line.ts +40 -0
  446. package/examples/extensions/subagent/README.md +172 -0
  447. package/examples/extensions/subagent/agents/planner.md +37 -0
  448. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  449. package/examples/extensions/subagent/agents/scout.md +50 -0
  450. package/examples/extensions/subagent/agents/worker.md +24 -0
  451. package/examples/extensions/subagent/agents.ts +156 -0
  452. package/examples/extensions/subagent/index.ts +963 -0
  453. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  454. package/examples/extensions/subagent/prompts/implement.md +10 -0
  455. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  456. package/examples/extensions/summarize.ts +195 -0
  457. package/examples/extensions/timed-confirm.ts +70 -0
  458. package/examples/extensions/todo.ts +299 -0
  459. package/examples/extensions/tool-override.ts +143 -0
  460. package/examples/extensions/tools.ts +146 -0
  461. package/examples/extensions/truncated-tool.ts +192 -0
  462. package/examples/extensions/with-deps/index.ts +36 -0
  463. package/examples/extensions/with-deps/package-lock.json +31 -0
  464. package/examples/extensions/with-deps/package.json +22 -0
  465. package/examples/sdk/01-minimal.ts +22 -0
  466. package/examples/sdk/02-custom-model.ts +49 -0
  467. package/examples/sdk/03-custom-prompt.ts +44 -0
  468. package/examples/sdk/04-skills.ts +47 -0
  469. package/examples/sdk/05-tools.ts +56 -0
  470. package/examples/sdk/06-extensions.ts +79 -0
  471. package/examples/sdk/07-context-files.ts +36 -0
  472. package/examples/sdk/08-prompt-templates.ts +42 -0
  473. package/examples/sdk/09-api-keys-and-oauth.ts +55 -0
  474. package/examples/sdk/10-settings.ts +38 -0
  475. package/examples/sdk/11-sessions.ts +48 -0
  476. package/examples/sdk/12-full-control.ts +72 -0
  477. package/examples/sdk/README.md +150 -0
  478. package/package.json +88 -0
@@ -0,0 +1,1524 @@
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
+ **Key capabilities:**
8
+ - **Custom tools** - Register tools the LLM can call via `pi.registerTool()`
9
+ - **Event interception** - Block or modify tool calls, inject context, customize compaction
10
+ - **User interaction** - Prompt users via `ctx.ui` (select, confirm, input, notify)
11
+ - **Custom UI components** - Full TUI components with keyboard input via `ctx.ui.custom()` for complex interactions
12
+ - **Custom commands** - Register commands like `/mycommand` via `pi.registerCommand()`
13
+ - **Session persistence** - Store state that survives restarts via `pi.appendEntry()`
14
+ - **Custom rendering** - Control how tool calls/results and messages appear in TUI
15
+
16
+ **Example use cases:**
17
+ - Permission gates (confirm before `rm -rf`, `sudo`, etc.)
18
+ - Git checkpointing (stash at each turn, restore on branch)
19
+ - Path protection (block writes to `.env`, `node_modules/`)
20
+ - Custom compaction (summarize conversation your way)
21
+ - Conversation summaries (see `summarize.ts` example)
22
+ - Interactive tools (questions, wizards, custom dialogs)
23
+ - Stateful tools (todo lists, connection pools)
24
+ - External integrations (file watchers, webhooks, CI triggers)
25
+ - Games while you wait (see `snake.ts` example)
26
+
27
+ See [examples/extensions/](../examples/extensions/) for working implementations.
28
+
29
+ ## Table of Contents
30
+
31
+ - [Quick Start](#quick-start)
32
+ - [Extension Locations](#extension-locations)
33
+ - [Available Imports](#available-imports)
34
+ - [Writing an Extension](#writing-an-extension)
35
+ - [Extension Styles](#extension-styles)
36
+ - [Events](#events)
37
+ - [Lifecycle Overview](#lifecycle-overview)
38
+ - [Session Events](#session-events)
39
+ - [Agent Events](#agent-events)
40
+ - [Tool Events](#tool-events)
41
+ - [ExtensionContext](#extensioncontext)
42
+ - [ExtensionCommandContext](#extensioncommandcontext)
43
+ - [ExtensionAPI Methods](#extensionapi-methods)
44
+ - [State Management](#state-management)
45
+ - [Custom Tools](#custom-tools)
46
+ - [Custom UI](#custom-ui)
47
+ - [Error Handling](#error-handling)
48
+ - [Mode Behavior](#mode-behavior)
49
+
50
+ ## Quick Start
51
+
52
+ Create `~/.pi/agent/extensions/my-extension.ts`:
53
+
54
+ ```typescript
55
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
56
+ import { Type } from "@sinclair/typebox";
57
+
58
+ export default function (pi: ExtensionAPI) {
59
+ // React to events
60
+ pi.on("session_start", async (_event, ctx) => {
61
+ ctx.ui.notify("Extension loaded!", "info");
62
+ });
63
+
64
+ pi.on("tool_call", async (event, ctx) => {
65
+ if (event.toolName === "bash" && event.input.command?.includes("rm -rf")) {
66
+ const ok = await ctx.ui.confirm("Dangerous!", "Allow rm -rf?");
67
+ if (!ok) return { block: true, reason: "Blocked by user" };
68
+ }
69
+ });
70
+
71
+ // Register a custom tool
72
+ pi.registerTool({
73
+ name: "greet",
74
+ label: "Greet",
75
+ description: "Greet someone by name",
76
+ parameters: Type.Object({
77
+ name: Type.String({ description: "Name to greet" }),
78
+ }),
79
+ async execute(toolCallId, params, onUpdate, ctx, signal) {
80
+ return {
81
+ content: [{ type: "text", text: `Hello, ${params.name}!` }],
82
+ details: {},
83
+ };
84
+ },
85
+ });
86
+
87
+ // Register a command
88
+ pi.registerCommand("hello", {
89
+ description: "Say hello",
90
+ handler: async (args, ctx) => {
91
+ ctx.ui.notify(`Hello ${args || "world"}!`, "info");
92
+ },
93
+ });
94
+ }
95
+ ```
96
+
97
+ Test with `--extension` (or `-e`) flag:
98
+
99
+ ```bash
100
+ pi -e ./my-extension.ts
101
+ ```
102
+
103
+ ## Extension Locations
104
+
105
+ Extensions are auto-discovered from:
106
+
107
+ | Location | Scope |
108
+ |----------|-------|
109
+ | `~/.pi/agent/extensions/*.ts` | Global (all projects) |
110
+ | `~/.pi/agent/extensions/*/index.ts` | Global (subdirectory) |
111
+ | `.pi/extensions/*.ts` | Project-local |
112
+ | `.pi/extensions/*/index.ts` | Project-local (subdirectory) |
113
+
114
+ Additional paths via `settings.json`:
115
+
116
+ ```json
117
+ {
118
+ "extensions": ["/path/to/extension.ts", "/path/to/extension/dir"]
119
+ }
120
+ ```
121
+
122
+ **Discovery rules:**
123
+
124
+ 1. **Direct files:** `extensions/*.ts` or `*.js` → loaded directly
125
+ 2. **Subdirectory with index:** `extensions/myext/index.ts` → loaded as single extension
126
+ 3. **Subdirectory with package.json:** `extensions/myext/package.json` with `"pi"` field → loads declared paths
127
+
128
+ ```
129
+ ~/.pi/agent/extensions/
130
+ ├── simple.ts # Direct file (auto-discovered)
131
+ ├── my-tool/
132
+ │ └── index.ts # Subdirectory with index (auto-discovered)
133
+ └── my-extension-pack/
134
+ ├── package.json # Declares multiple extensions
135
+ ├── node_modules/ # Dependencies installed here
136
+ └── src/
137
+ ├── safety-gates.ts # First extension
138
+ └── custom-tools.ts # Second extension
139
+ ```
140
+
141
+ ```json
142
+ // my-extension-pack/package.json
143
+ {
144
+ "name": "my-extension-pack",
145
+ "dependencies": {
146
+ "zod": "^3.0.0"
147
+ },
148
+ "pi": {
149
+ "extensions": ["./src/safety-gates.ts", "./src/custom-tools.ts"]
150
+ }
151
+ }
152
+ ```
153
+
154
+ The `package.json` approach enables:
155
+ - Multiple extensions from one package
156
+ - Third-party npm dependencies (resolved via jiti)
157
+ - Nested source structure (no depth limit within the package)
158
+ - Deployment to and installation from npm
159
+
160
+ ## Available Imports
161
+
162
+ | Package | Purpose |
163
+ |---------|---------|
164
+ | `@mariozechner/pi-coding-agent` | Extension types (`ExtensionAPI`, `ExtensionContext`, events) |
165
+ | `@sinclair/typebox` | Schema definitions for tool parameters |
166
+ | `@mariozechner/pi-ai` | AI utilities (`StringEnum` for Google-compatible enums) |
167
+ | `@mariozechner/pi-tui` | TUI components for custom rendering |
168
+
169
+ 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.
170
+
171
+ Node.js built-ins (`node:fs`, `node:path`, etc.) are also available.
172
+
173
+ ## Writing an Extension
174
+
175
+ An extension exports a default function that receives `ExtensionAPI`:
176
+
177
+ ```typescript
178
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
179
+
180
+ export default function (pi: ExtensionAPI) {
181
+ // Subscribe to events
182
+ pi.on("event_name", async (event, ctx) => {
183
+ // ctx.ui for user interaction
184
+ const ok = await ctx.ui.confirm("Title", "Are you sure?");
185
+ ctx.ui.notify("Done!", "success");
186
+ ctx.ui.setStatus("my-ext", "Processing..."); // Footer status
187
+ ctx.ui.setWidget("my-ext", ["Line 1", "Line 2"]); // Widget above editor
188
+ });
189
+
190
+ // Register tools, commands, shortcuts, flags
191
+ pi.registerTool({ ... });
192
+ pi.registerCommand("name", { ... });
193
+ pi.registerShortcut("ctrl+x", { ... });
194
+ pi.registerFlag("my-flag", { ... });
195
+ }
196
+ ```
197
+
198
+ Extensions are loaded via [jiti](https://github.com/unjs/jiti), so TypeScript works without compilation.
199
+
200
+ ### Extension Styles
201
+
202
+ **Single file** - simplest, for small extensions:
203
+
204
+ ```
205
+ ~/.pi/agent/extensions/
206
+ └── my-extension.ts
207
+ ```
208
+
209
+ **Directory with index.ts** - for multi-file extensions:
210
+
211
+ ```
212
+ ~/.pi/agent/extensions/
213
+ └── my-extension/
214
+ ├── index.ts # Entry point (exports default function)
215
+ ├── tools.ts # Helper module
216
+ └── utils.ts # Helper module
217
+ ```
218
+
219
+ **Package with dependencies** - for extensions that need npm packages:
220
+
221
+ ```
222
+ ~/.pi/agent/extensions/
223
+ └── my-extension/
224
+ ├── package.json # Declares dependencies and entry points
225
+ ├── package-lock.json
226
+ ├── node_modules/ # After npm install
227
+ └── src/
228
+ └── index.ts
229
+ ```
230
+
231
+ ```json
232
+ // package.json
233
+ {
234
+ "name": "my-extension",
235
+ "dependencies": {
236
+ "zod": "^3.0.0",
237
+ "chalk": "^5.0.0"
238
+ },
239
+ "pi": {
240
+ "extensions": ["./src/index.ts"]
241
+ }
242
+ }
243
+ ```
244
+
245
+ Run `npm install` in the extension directory, then imports from `node_modules/` work automatically.
246
+
247
+ ## Events
248
+
249
+ ### Lifecycle Overview
250
+
251
+ ```
252
+ pi starts
253
+
254
+ └─► session_start
255
+
256
+
257
+ user sends prompt ─────────────────────────────────────────┐
258
+ │ │
259
+ ├─► before_agent_start (can inject message, modify system prompt)
260
+ ├─► agent_start │
261
+ │ │
262
+ │ ┌─── turn (repeats while LLM calls tools) ───┐ │
263
+ │ │ │ │
264
+ │ ├─► turn_start │ │
265
+ │ ├─► context (can modify messages) │ │
266
+ │ │ │ │
267
+ │ │ LLM responds, may call tools: │ │
268
+ │ │ ├─► tool_call (can block) │ │
269
+ │ │ │ tool executes │ │
270
+ │ │ └─► tool_result (can modify) │ │
271
+ │ │ │ │
272
+ │ └─► turn_end │ │
273
+ │ │
274
+ └─► agent_end │
275
+
276
+ user sends another prompt ◄────────────────────────────────┘
277
+
278
+ /new (new session) or /resume (switch session)
279
+ ├─► session_before_switch (can cancel)
280
+ └─► session_switch
281
+
282
+ /fork
283
+ ├─► session_before_fork (can cancel)
284
+ └─► session_fork
285
+
286
+ /compact or auto-compaction
287
+ ├─► session_before_compact (can cancel or customize)
288
+ └─► session_compact
289
+
290
+ /tree navigation
291
+ ├─► session_before_tree (can cancel or customize)
292
+ └─► session_tree
293
+
294
+ /model or Ctrl+P (model selection/cycling)
295
+ └─► model_select
296
+
297
+ exit (Ctrl+C, Ctrl+D)
298
+ └─► session_shutdown
299
+ ```
300
+
301
+ ### Session Events
302
+
303
+ #### session_start
304
+
305
+ Fired on initial session load.
306
+
307
+ ```typescript
308
+ pi.on("session_start", async (_event, ctx) => {
309
+ ctx.ui.notify(`Session: ${ctx.sessionManager.getSessionFile() ?? "ephemeral"}`, "info");
310
+ });
311
+ ```
312
+
313
+ **Examples:** [claude-rules.ts](../examples/extensions/claude-rules.ts), [custom-header.ts](../examples/extensions/custom-header.ts), [file-trigger.ts](../examples/extensions/file-trigger.ts), [status-line.ts](../examples/extensions/status-line.ts), [todo.ts](../examples/extensions/todo.ts), [tools.ts](../examples/extensions/tools.ts)
314
+
315
+ #### session_before_switch / session_switch
316
+
317
+ Fired when starting a new session (`/new`) or switching sessions (`/resume`).
318
+
319
+ ```typescript
320
+ pi.on("session_before_switch", async (event, ctx) => {
321
+ // event.reason - "new" or "resume"
322
+ // event.targetSessionFile - session we're switching to (only for "resume")
323
+
324
+ if (event.reason === "new") {
325
+ const ok = await ctx.ui.confirm("Clear?", "Delete all messages?");
326
+ if (!ok) return { cancel: true };
327
+ }
328
+ });
329
+
330
+ pi.on("session_switch", async (event, ctx) => {
331
+ // event.reason - "new" or "resume"
332
+ // event.previousSessionFile - session we came from
333
+ });
334
+ ```
335
+
336
+ **Examples:** [confirm-destructive.ts](../examples/extensions/confirm-destructive.ts), [dirty-repo-guard.ts](../examples/extensions/dirty-repo-guard.ts), [status-line.ts](../examples/extensions/status-line.ts), [todo.ts](../examples/extensions/todo.ts)
337
+
338
+ #### session_before_fork / session_fork
339
+
340
+ Fired when forking via `/fork`.
341
+
342
+ ```typescript
343
+ pi.on("session_before_fork", async (event, ctx) => {
344
+ // event.entryId - ID of the entry being forked from
345
+ return { cancel: true }; // Cancel fork
346
+ // OR
347
+ return { skipConversationRestore: true }; // Fork but don't rewind messages
348
+ });
349
+
350
+ pi.on("session_fork", async (event, ctx) => {
351
+ // event.previousSessionFile - previous session file
352
+ });
353
+ ```
354
+
355
+ **Examples:** [confirm-destructive.ts](../examples/extensions/confirm-destructive.ts), [dirty-repo-guard.ts](../examples/extensions/dirty-repo-guard.ts), [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts), [todo.ts](../examples/extensions/todo.ts), [tools.ts](../examples/extensions/tools.ts)
356
+
357
+ #### session_before_compact / session_compact
358
+
359
+ Fired on compaction. See [compaction.md](compaction.md) for details.
360
+
361
+ ```typescript
362
+ pi.on("session_before_compact", async (event, ctx) => {
363
+ const { preparation, branchEntries, customInstructions, signal } = event;
364
+
365
+ // Cancel:
366
+ return { cancel: true };
367
+
368
+ // Custom summary:
369
+ return {
370
+ compaction: {
371
+ summary: "...",
372
+ firstKeptEntryId: preparation.firstKeptEntryId,
373
+ tokensBefore: preparation.tokensBefore,
374
+ }
375
+ };
376
+ });
377
+
378
+ pi.on("session_compact", async (event, ctx) => {
379
+ // event.compactionEntry - the saved compaction
380
+ // event.fromExtension - whether extension provided it
381
+ });
382
+ ```
383
+
384
+ **Examples:** [custom-compaction.ts](../examples/extensions/custom-compaction.ts)
385
+
386
+ #### session_before_tree / session_tree
387
+
388
+ Fired on `/tree` navigation.
389
+
390
+ ```typescript
391
+ pi.on("session_before_tree", async (event, ctx) => {
392
+ const { preparation, signal } = event;
393
+ return { cancel: true };
394
+ // OR provide custom summary:
395
+ return { summary: { summary: "...", details: {} } };
396
+ });
397
+
398
+ pi.on("session_tree", async (event, ctx) => {
399
+ // event.newLeafId, oldLeafId, summaryEntry, fromExtension
400
+ });
401
+ ```
402
+
403
+ **Examples:** [todo.ts](../examples/extensions/todo.ts), [tools.ts](../examples/extensions/tools.ts)
404
+
405
+ #### session_shutdown
406
+
407
+ Fired on exit (Ctrl+C, Ctrl+D, SIGTERM).
408
+
409
+ ```typescript
410
+ pi.on("session_shutdown", async (_event, ctx) => {
411
+ // Cleanup, save state, etc.
412
+ });
413
+ ```
414
+
415
+ **Examples:** [auto-commit-on-exit.ts](../examples/extensions/auto-commit-on-exit.ts)
416
+
417
+ ### Agent Events
418
+
419
+ #### before_agent_start
420
+
421
+ Fired after user submits prompt, before agent loop. Can inject a message and/or modify the system prompt.
422
+
423
+ ```typescript
424
+ pi.on("before_agent_start", async (event, ctx) => {
425
+ // event.prompt - user's prompt text
426
+ // event.images - attached images (if any)
427
+ // event.systemPrompt - current system prompt
428
+
429
+ return {
430
+ // Inject a persistent message (stored in session, sent to LLM)
431
+ message: {
432
+ customType: "my-extension",
433
+ content: "Additional context for the LLM",
434
+ display: true,
435
+ },
436
+ // Replace the system prompt for this turn (chained across extensions)
437
+ systemPrompt: event.systemPrompt + "\n\nExtra instructions for this turn...",
438
+ };
439
+ });
440
+ ```
441
+
442
+ **Examples:** [claude-rules.ts](../examples/extensions/claude-rules.ts), [pirate.ts](../examples/extensions/pirate.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [ssh.ts](../examples/extensions/ssh.ts)
443
+
444
+ #### agent_start / agent_end
445
+
446
+ Fired once per user prompt.
447
+
448
+ ```typescript
449
+ pi.on("agent_start", async (_event, ctx) => {});
450
+
451
+ pi.on("agent_end", async (event, ctx) => {
452
+ // event.messages - messages from this prompt
453
+ });
454
+ ```
455
+
456
+ **Examples:** [chalk-logger.ts](../examples/extensions/chalk-logger.ts), [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts)
457
+
458
+ #### turn_start / turn_end
459
+
460
+ Fired for each turn (one LLM response + tool calls).
461
+
462
+ ```typescript
463
+ pi.on("turn_start", async (event, ctx) => {
464
+ // event.turnIndex, event.timestamp
465
+ });
466
+
467
+ pi.on("turn_end", async (event, ctx) => {
468
+ // event.turnIndex, event.message, event.toolResults
469
+ });
470
+ ```
471
+
472
+ **Examples:** [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [status-line.ts](../examples/extensions/status-line.ts)
473
+
474
+ #### context
475
+
476
+ Fired before each LLM call. Modify messages non-destructively.
477
+
478
+ ```typescript
479
+ pi.on("context", async (event, ctx) => {
480
+ // event.messages - deep copy, safe to modify
481
+ const filtered = event.messages.filter(m => !shouldPrune(m));
482
+ return { messages: filtered };
483
+ });
484
+ ```
485
+
486
+ **Examples:** [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts)
487
+
488
+ ### Model Events
489
+
490
+ #### model_select
491
+
492
+ Fired when the model changes via `/model` command, model cycling (`Ctrl+P`), or session restore.
493
+
494
+ ```typescript
495
+ pi.on("model_select", async (event, ctx) => {
496
+ // event.model - newly selected model
497
+ // event.previousModel - previous model (undefined if first selection)
498
+ // event.source - "set" | "cycle" | "restore"
499
+
500
+ const prev = event.previousModel
501
+ ? `${event.previousModel.provider}/${event.previousModel.id}`
502
+ : "none";
503
+ const next = `${event.model.provider}/${event.model.id}`;
504
+
505
+ ctx.ui.notify(`Model changed (${event.source}): ${prev} -> ${next}`, "info");
506
+ });
507
+ ```
508
+
509
+ Use this to update UI elements (status bars, footers) or perform model-specific initialization when the active model changes.
510
+
511
+ **Examples:** [model-status.ts](../examples/extensions/model-status.ts)
512
+
513
+ ### Tool Events
514
+
515
+ #### tool_call
516
+
517
+ Fired before tool executes. **Can block.**
518
+
519
+ ```typescript
520
+ pi.on("tool_call", async (event, ctx) => {
521
+ // event.toolName - "bash", "read", "write", "edit", etc.
522
+ // event.toolCallId
523
+ // event.input - tool parameters
524
+
525
+ if (shouldBlock(event)) {
526
+ return { block: true, reason: "Not allowed" };
527
+ }
528
+ });
529
+ ```
530
+
531
+ **Examples:** [chalk-logger.ts](../examples/extensions/chalk-logger.ts), [permission-gate.ts](../examples/extensions/permission-gate.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [protected-paths.ts](../examples/extensions/protected-paths.ts)
532
+
533
+ #### tool_result
534
+
535
+ Fired after tool executes. **Can modify result.**
536
+
537
+ ```typescript
538
+ import { isBashToolResult } from "@mariozechner/pi-coding-agent";
539
+
540
+ pi.on("tool_result", async (event, ctx) => {
541
+ // event.toolName, event.toolCallId, event.input
542
+ // event.content, event.details, event.isError
543
+
544
+ if (isBashToolResult(event)) {
545
+ // event.details is typed as BashToolDetails
546
+ }
547
+
548
+ // Modify result:
549
+ return { content: [...], details: {...}, isError: false };
550
+ });
551
+ ```
552
+
553
+ **Examples:** [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts)
554
+
555
+ ### User Bash Events
556
+
557
+ #### user_bash
558
+
559
+ Fired when user executes `!` or `!!` commands. **Can intercept.**
560
+
561
+ ```typescript
562
+ pi.on("user_bash", (event, ctx) => {
563
+ // event.command - the bash command
564
+ // event.excludeFromContext - true if !! prefix
565
+ // event.cwd - working directory
566
+
567
+ // Option 1: Provide custom operations (e.g., SSH)
568
+ return { operations: remoteBashOps };
569
+
570
+ // Option 2: Full replacement - return result directly
571
+ return { result: { output: "...", exitCode: 0, cancelled: false, truncated: false } };
572
+ });
573
+ ```
574
+
575
+ **Examples:** [ssh.ts](../examples/extensions/ssh.ts), [interactive-shell.ts](../examples/extensions/interactive-shell.ts)
576
+
577
+ ## ExtensionContext
578
+
579
+ Every handler receives `ctx: ExtensionContext`:
580
+
581
+ ### ctx.ui
582
+
583
+ UI methods for user interaction. See [Custom UI](#custom-ui) for full details.
584
+
585
+ ### ctx.hasUI
586
+
587
+ `false` in print mode (`-p`), JSON mode, and RPC mode. Always check before using `ctx.ui`.
588
+
589
+ ### ctx.cwd
590
+
591
+ Current working directory.
592
+
593
+ ### ctx.sessionManager
594
+
595
+ Read-only access to session state:
596
+
597
+ ```typescript
598
+ ctx.sessionManager.getEntries() // All entries
599
+ ctx.sessionManager.getBranch() // Current branch
600
+ ctx.sessionManager.getLeafId() // Current leaf entry ID
601
+ ```
602
+
603
+ ### ctx.modelRegistry / ctx.model
604
+
605
+ Access to models and API keys.
606
+
607
+ ### ctx.isIdle() / ctx.abort() / ctx.hasPendingMessages()
608
+
609
+ Control flow helpers.
610
+
611
+ ### ctx.shutdown()
612
+
613
+ Request a graceful shutdown of pi.
614
+
615
+ - **Interactive mode:** Deferred until the agent becomes idle (after processing all queued steering and follow-up messages).
616
+ - **RPC mode:** Deferred until the next idle state (after completing the current command response, when waiting for the next command).
617
+ - **Print mode:** No-op. The process exits automatically when all prompts are processed.
618
+
619
+ Emits `session_shutdown` event to all extensions before exiting. Available in all contexts (event handlers, tools, commands, shortcuts).
620
+
621
+ ```typescript
622
+ pi.on("tool_call", (event, ctx) => {
623
+ if (isFatal(event.input)) {
624
+ ctx.shutdown();
625
+ }
626
+ });
627
+ ```
628
+
629
+ ## ExtensionCommandContext
630
+
631
+ 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.
632
+
633
+ ### ctx.waitForIdle()
634
+
635
+ Wait for the agent to finish streaming:
636
+
637
+ ```typescript
638
+ pi.registerCommand("my-cmd", {
639
+ handler: async (args, ctx) => {
640
+ await ctx.waitForIdle();
641
+ // Agent is now idle, safe to modify session
642
+ },
643
+ });
644
+ ```
645
+
646
+ ### ctx.newSession(options?)
647
+
648
+ Create a new session:
649
+
650
+ ```typescript
651
+ const result = await ctx.newSession({
652
+ parentSession: ctx.sessionManager.getSessionFile(),
653
+ setup: async (sm) => {
654
+ sm.appendMessage({
655
+ role: "user",
656
+ content: [{ type: "text", text: "Context from previous session..." }],
657
+ timestamp: Date.now(),
658
+ });
659
+ },
660
+ });
661
+
662
+ if (result.cancelled) {
663
+ // An extension cancelled the new session
664
+ }
665
+ ```
666
+
667
+ ### ctx.branch(entryId)
668
+
669
+ Branch from a specific entry:
670
+
671
+ ```typescript
672
+ const result = await ctx.branch("entry-id-123");
673
+ if (!result.cancelled) {
674
+ // Now in the branched session
675
+ }
676
+ ```
677
+
678
+ ### ctx.navigateTree(targetId, options?)
679
+
680
+ Navigate to a different point in the session tree:
681
+
682
+ ```typescript
683
+ const result = await ctx.navigateTree("entry-id-456", {
684
+ summarize: true,
685
+ });
686
+ ```
687
+
688
+ ## ExtensionAPI Methods
689
+
690
+ ### pi.on(event, handler)
691
+
692
+ Subscribe to events. See [Events](#events) for event types and return values.
693
+
694
+ ### pi.registerTool(definition)
695
+
696
+ Register a custom tool callable by the LLM. See [Custom Tools](#custom-tools) for full details.
697
+
698
+ ```typescript
699
+ import { Type } from "@sinclair/typebox";
700
+ import { StringEnum } from "@mariozechner/pi-ai";
701
+
702
+ pi.registerTool({
703
+ name: "my_tool",
704
+ label: "My Tool",
705
+ description: "What this tool does",
706
+ parameters: Type.Object({
707
+ action: StringEnum(["list", "add"] as const),
708
+ text: Type.Optional(Type.String()),
709
+ }),
710
+
711
+ async execute(toolCallId, params, onUpdate, ctx, signal) {
712
+ // Stream progress
713
+ onUpdate?.({ content: [{ type: "text", text: "Working..." }] });
714
+
715
+ return {
716
+ content: [{ type: "text", text: "Done" }],
717
+ details: { result: "..." },
718
+ };
719
+ },
720
+
721
+ // Optional: Custom rendering
722
+ renderCall(args, theme) { ... },
723
+ renderResult(result, options, theme) { ... },
724
+ });
725
+ ```
726
+
727
+ **Examples:** [hello.ts](../examples/extensions/hello.ts), [question.ts](../examples/extensions/question.ts), [questionnaire.ts](../examples/extensions/questionnaire.ts), [todo.ts](../examples/extensions/todo.ts), [truncated-tool.ts](../examples/extensions/truncated-tool.ts)
728
+
729
+ ### pi.sendMessage(message, options?)
730
+
731
+ Inject a custom message into the session.
732
+
733
+ ```typescript
734
+ pi.sendMessage({
735
+ customType: "my-extension",
736
+ content: "Message text",
737
+ display: true,
738
+ details: { ... },
739
+ }, {
740
+ triggerTurn: true,
741
+ deliverAs: "steer",
742
+ });
743
+ ```
744
+
745
+ **Options:**
746
+ - `deliverAs` - Delivery mode:
747
+ - `"steer"` (default) - Interrupts streaming. Delivered after current tool finishes, remaining tools skipped.
748
+ - `"followUp"` - Waits for agent to finish. Delivered only when agent has no more tool calls.
749
+ - `"nextTurn"` - Queued for next user prompt. Does not interrupt or trigger anything.
750
+ - `triggerTurn: true` - If agent is idle, trigger an LLM response immediately. Only applies to `"steer"` and `"followUp"` modes (ignored for `"nextTurn"`).
751
+
752
+ **Examples:** [file-trigger.ts](../examples/extensions/file-trigger.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts)
753
+
754
+ ### pi.sendUserMessage(content, options?)
755
+
756
+ 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.
757
+
758
+ ```typescript
759
+ // Simple text message
760
+ pi.sendUserMessage("What is 2+2?");
761
+
762
+ // With content array (text + images)
763
+ pi.sendUserMessage([
764
+ { type: "text", text: "Describe this image:" },
765
+ { type: "image", source: { type: "base64", mediaType: "image/png", data: "..." } },
766
+ ]);
767
+
768
+ // During streaming - must specify delivery mode
769
+ pi.sendUserMessage("Focus on error handling", { deliverAs: "steer" });
770
+ pi.sendUserMessage("And then summarize", { deliverAs: "followUp" });
771
+ ```
772
+
773
+ **Options:**
774
+ - `deliverAs` - Required when agent is streaming:
775
+ - `"steer"` - Interrupts after current tool, remaining tools skipped
776
+ - `"followUp"` - Waits for agent to finish all tools
777
+
778
+ When not streaming, the message is sent immediately and triggers a new turn. When streaming without `deliverAs`, throws an error.
779
+
780
+ See [send-user-message.ts](../examples/extensions/send-user-message.ts) for a complete example.
781
+
782
+ ### pi.appendEntry(customType, data?)
783
+
784
+ Persist extension state (does NOT participate in LLM context).
785
+
786
+ ```typescript
787
+ pi.appendEntry("my-state", { count: 42 });
788
+
789
+ // Restore on reload
790
+ pi.on("session_start", async (_event, ctx) => {
791
+ for (const entry of ctx.sessionManager.getEntries()) {
792
+ if (entry.type === "custom" && entry.customType === "my-state") {
793
+ // Reconstruct from entry.data
794
+ }
795
+ }
796
+ });
797
+ ```
798
+
799
+ **Examples:** [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [snake.ts](../examples/extensions/snake.ts), [tools.ts](../examples/extensions/tools.ts)
800
+
801
+ ### pi.setSessionName(name)
802
+
803
+ Set the session display name (shown in session selector instead of first message).
804
+
805
+ ```typescript
806
+ pi.setSessionName("Refactor auth module");
807
+ ```
808
+
809
+ ### pi.getSessionName()
810
+
811
+ Get the current session name, if set.
812
+
813
+ ```typescript
814
+ const name = pi.getSessionName();
815
+ if (name) {
816
+ console.log(`Session: ${name}`);
817
+ }
818
+ ```
819
+
820
+ ### pi.registerCommand(name, options)
821
+
822
+ Register a command.
823
+
824
+ ```typescript
825
+ pi.registerCommand("stats", {
826
+ description: "Show session statistics",
827
+ handler: async (args, ctx) => {
828
+ const count = ctx.sessionManager.getEntries().length;
829
+ ctx.ui.notify(`${count} entries`, "info");
830
+ }
831
+ });
832
+ ```
833
+
834
+ **Examples:** [custom-footer.ts](../examples/extensions/custom-footer.ts), [custom-header.ts](../examples/extensions/custom-header.ts), [handoff.ts](../examples/extensions/handoff.ts), [pirate.ts](../examples/extensions/pirate.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [qna.ts](../examples/extensions/qna.ts), [send-user-message.ts](../examples/extensions/send-user-message.ts), [snake.ts](../examples/extensions/snake.ts), [summarize.ts](../examples/extensions/summarize.ts), [todo.ts](../examples/extensions/todo.ts), [tools.ts](../examples/extensions/tools.ts)
835
+
836
+ ### pi.registerMessageRenderer(customType, renderer)
837
+
838
+ Register a custom TUI renderer for messages with your `customType`. See [Custom UI](#custom-ui).
839
+
840
+ ### pi.registerShortcut(shortcut, options)
841
+
842
+ Register a keyboard shortcut.
843
+
844
+ ```typescript
845
+ pi.registerShortcut("ctrl+shift+p", {
846
+ description: "Toggle plan mode",
847
+ handler: async (ctx) => {
848
+ ctx.ui.notify("Toggled!");
849
+ },
850
+ });
851
+ ```
852
+
853
+ **Examples:** [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts)
854
+
855
+ ### pi.registerFlag(name, options)
856
+
857
+ Register a CLI flag.
858
+
859
+ ```typescript
860
+ pi.registerFlag("plan", {
861
+ description: "Start in plan mode",
862
+ type: "boolean",
863
+ default: false,
864
+ });
865
+
866
+ // Check value
867
+ if (pi.getFlag("--plan")) {
868
+ // Plan mode enabled
869
+ }
870
+ ```
871
+
872
+ **Examples:** [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts)
873
+
874
+ ### pi.exec(command, args, options?)
875
+
876
+ Execute a shell command.
877
+
878
+ ```typescript
879
+ const result = await pi.exec("git", ["status"], { signal, timeout: 5000 });
880
+ // result.stdout, result.stderr, result.code, result.killed
881
+ ```
882
+
883
+ **Examples:** [auto-commit-on-exit.ts](../examples/extensions/auto-commit-on-exit.ts), [dirty-repo-guard.ts](../examples/extensions/dirty-repo-guard.ts), [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts)
884
+
885
+ ### pi.getActiveTools() / pi.getAllTools() / pi.setActiveTools(names)
886
+
887
+ Manage active tools.
888
+
889
+ ```typescript
890
+ const active = pi.getActiveTools(); // ["read", "bash", "edit", "write"]
891
+ const all = pi.getAllTools(); // [{ name: "read", description: "Read file contents..." }, ...]
892
+ const names = all.map(t => t.name); // Just names if needed
893
+ pi.setActiveTools(["read", "bash"]); // Switch to read-only
894
+ ```
895
+
896
+ **Examples:** [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [tools.ts](../examples/extensions/tools.ts)
897
+
898
+ ### pi.setModel(model)
899
+
900
+ Set the current model. Returns `false` if no API key is available for the model.
901
+
902
+ ```typescript
903
+ const model = ctx.modelRegistry.find("anthropic", "claude-sonnet-4-5");
904
+ if (model) {
905
+ const success = await pi.setModel(model);
906
+ if (!success) {
907
+ ctx.ui.notify("No API key for this model", "error");
908
+ }
909
+ }
910
+ ```
911
+
912
+ **Examples:** [preset.ts](../examples/extensions/preset.ts)
913
+
914
+ ### pi.getThinkingLevel() / pi.setThinkingLevel(level)
915
+
916
+ Get or set the thinking level. Level is clamped to model capabilities (non-reasoning models always use "off").
917
+
918
+ ```typescript
919
+ const current = pi.getThinkingLevel(); // "off" | "minimal" | "low" | "medium" | "high" | "xhigh"
920
+ pi.setThinkingLevel("high");
921
+ ```
922
+
923
+ **Examples:** [preset.ts](../examples/extensions/preset.ts)
924
+
925
+ ### pi.events
926
+
927
+ Shared event bus for communication between extensions:
928
+
929
+ ```typescript
930
+ pi.events.on("my:event", (data) => { ... });
931
+ pi.events.emit("my:event", { ... });
932
+ ```
933
+
934
+ ## State Management
935
+
936
+ Extensions with state should store it in tool result `details` for proper branching support:
937
+
938
+ ```typescript
939
+ export default function (pi: ExtensionAPI) {
940
+ let items: string[] = [];
941
+
942
+ // Reconstruct state from session
943
+ pi.on("session_start", async (_event, ctx) => {
944
+ items = [];
945
+ for (const entry of ctx.sessionManager.getBranch()) {
946
+ if (entry.type === "message" && entry.message.role === "toolResult") {
947
+ if (entry.message.toolName === "my_tool") {
948
+ items = entry.message.details?.items ?? [];
949
+ }
950
+ }
951
+ }
952
+ });
953
+
954
+ pi.registerTool({
955
+ name: "my_tool",
956
+ // ...
957
+ async execute(toolCallId, params, onUpdate, ctx, signal) {
958
+ items.push("new item");
959
+ return {
960
+ content: [{ type: "text", text: "Added" }],
961
+ details: { items: [...items] }, // Store for reconstruction
962
+ };
963
+ },
964
+ });
965
+ }
966
+ ```
967
+
968
+ ## Custom Tools
969
+
970
+ Register tools the LLM can call via `pi.registerTool()`. Tools appear in the system prompt and can have custom rendering.
971
+
972
+ ### Tool Definition
973
+
974
+ ```typescript
975
+ import { Type } from "@sinclair/typebox";
976
+ import { StringEnum } from "@mariozechner/pi-ai";
977
+ import { Text } from "@mariozechner/pi-tui";
978
+
979
+ pi.registerTool({
980
+ name: "my_tool",
981
+ label: "My Tool",
982
+ description: "What this tool does (shown to LLM)",
983
+ parameters: Type.Object({
984
+ action: StringEnum(["list", "add"] as const), // Use StringEnum for Google compatibility
985
+ text: Type.Optional(Type.String()),
986
+ }),
987
+
988
+ async execute(toolCallId, params, onUpdate, ctx, signal) {
989
+ // Check for cancellation
990
+ if (signal?.aborted) {
991
+ return { content: [{ type: "text", text: "Cancelled" }] };
992
+ }
993
+
994
+ // Stream progress updates
995
+ onUpdate?.({
996
+ content: [{ type: "text", text: "Working..." }],
997
+ details: { progress: 50 },
998
+ });
999
+
1000
+ // Run commands via pi.exec (captured from extension closure)
1001
+ const result = await pi.exec("some-command", [], { signal });
1002
+
1003
+ // Return result
1004
+ return {
1005
+ content: [{ type: "text", text: "Done" }], // Sent to LLM
1006
+ details: { data: result }, // For rendering & state
1007
+ };
1008
+ },
1009
+
1010
+ // Optional: Custom rendering
1011
+ renderCall(args, theme) { ... },
1012
+ renderResult(result, options, theme) { ... },
1013
+ });
1014
+ ```
1015
+
1016
+ **Important:** Use `StringEnum` from `@mariozechner/pi-ai` for string enums. `Type.Union`/`Type.Literal` doesn't work with Google's API.
1017
+
1018
+ ### Overriding Built-in Tools
1019
+
1020
+ 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.
1021
+
1022
+ ```bash
1023
+ # Extension's read tool replaces built-in read
1024
+ pi -e ./tool-override.ts
1025
+ ```
1026
+
1027
+ Alternatively, use `--no-tools` to start without any built-in tools:
1028
+ ```bash
1029
+ # No built-in tools, only extension tools
1030
+ pi --no-tools -e ./my-extension.ts
1031
+ ```
1032
+
1033
+ See [examples/extensions/tool-override.ts](../examples/extensions/tool-override.ts) for a complete example that overrides `read` with logging and access control.
1034
+
1035
+ **Rendering:** If your override doesn't provide custom `renderCall`/`renderResult` functions, 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.
1036
+
1037
+ **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.
1038
+
1039
+ Built-in tool implementations:
1040
+ - [read.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/read.ts) - `ReadToolDetails`
1041
+ - [bash.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/bash.ts) - `BashToolDetails`
1042
+ - [edit.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/edit.ts)
1043
+ - [write.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/write.ts)
1044
+ - [grep.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/grep.ts) - `GrepToolDetails`
1045
+ - [find.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/find.ts) - `FindToolDetails`
1046
+ - [ls.ts](https://github.com/badlogic/pi-mono/blob/main/packages/coding-agent/src/core/tools/ls.ts) - `LsToolDetails`
1047
+
1048
+ ### Remote Execution
1049
+
1050
+ Built-in tools support pluggable operations for delegating to remote systems (SSH, containers, etc.):
1051
+
1052
+ ```typescript
1053
+ import { createReadTool, createBashTool, type ReadOperations } from "@mariozechner/pi-coding-agent";
1054
+
1055
+ // Create tool with custom operations
1056
+ const remoteRead = createReadTool(cwd, {
1057
+ operations: {
1058
+ readFile: (path) => sshExec(remote, `cat ${path}`),
1059
+ access: (path) => sshExec(remote, `test -r ${path}`).then(() => {}),
1060
+ }
1061
+ });
1062
+
1063
+ // Register, checking flag at execution time
1064
+ pi.registerTool({
1065
+ ...remoteRead,
1066
+ async execute(id, params, onUpdate, _ctx, signal) {
1067
+ const ssh = getSshConfig();
1068
+ if (ssh) {
1069
+ const tool = createReadTool(cwd, { operations: createRemoteOps(ssh) });
1070
+ return tool.execute(id, params, signal, onUpdate);
1071
+ }
1072
+ return localRead.execute(id, params, signal, onUpdate);
1073
+ },
1074
+ });
1075
+ ```
1076
+
1077
+ **Operations interfaces:** `ReadOperations`, `WriteOperations`, `EditOperations`, `BashOperations`, `LsOperations`, `GrepOperations`, `FindOperations`
1078
+
1079
+ See [examples/extensions/ssh.ts](../examples/extensions/ssh.ts) for a complete SSH example with `--ssh` flag.
1080
+
1081
+ ### Output Truncation
1082
+
1083
+ **Tools MUST truncate their output** to avoid overwhelming the LLM context. Large outputs can cause:
1084
+ - Context overflow errors (prompt too long)
1085
+ - Compaction failures
1086
+ - Degraded model performance
1087
+
1088
+ The built-in limit is **50KB** (~10k tokens) and **2000 lines**, whichever is hit first. Use the exported truncation utilities:
1089
+
1090
+ ```typescript
1091
+ import {
1092
+ truncateHead, // Keep first N lines/bytes (good for file reads, search results)
1093
+ truncateTail, // Keep last N lines/bytes (good for logs, command output)
1094
+ formatSize, // Human-readable size (e.g., "50KB", "1.5MB")
1095
+ DEFAULT_MAX_BYTES, // 50KB
1096
+ DEFAULT_MAX_LINES, // 2000
1097
+ } from "@mariozechner/pi-coding-agent";
1098
+
1099
+ async execute(toolCallId, params, onUpdate, ctx, signal) {
1100
+ const output = await runCommand();
1101
+
1102
+ // Apply truncation
1103
+ const truncation = truncateHead(output, {
1104
+ maxLines: DEFAULT_MAX_LINES,
1105
+ maxBytes: DEFAULT_MAX_BYTES,
1106
+ });
1107
+
1108
+ let result = truncation.content;
1109
+
1110
+ if (truncation.truncated) {
1111
+ // Write full output to temp file
1112
+ const tempFile = writeTempFile(output);
1113
+
1114
+ // Inform the LLM where to find complete output
1115
+ result += `\n\n[Output truncated: ${truncation.outputLines} of ${truncation.totalLines} lines`;
1116
+ result += ` (${formatSize(truncation.outputBytes)} of ${formatSize(truncation.totalBytes)}).`;
1117
+ result += ` Full output saved to: ${tempFile}]`;
1118
+ }
1119
+
1120
+ return { content: [{ type: "text", text: result }] };
1121
+ }
1122
+ ```
1123
+
1124
+ **Key points:**
1125
+ - Use `truncateHead` for content where the beginning matters (search results, file reads)
1126
+ - Use `truncateTail` for content where the end matters (logs, command output)
1127
+ - Always inform the LLM when output is truncated and where to find the full version
1128
+ - Document the truncation limits in your tool's description
1129
+
1130
+ See [examples/extensions/truncated-tool.ts](../examples/extensions/truncated-tool.ts) for a complete example wrapping `rg` (ripgrep) with proper truncation.
1131
+
1132
+ ### Multiple Tools
1133
+
1134
+ One extension can register multiple tools with shared state:
1135
+
1136
+ ```typescript
1137
+ export default function (pi: ExtensionAPI) {
1138
+ let connection = null;
1139
+
1140
+ pi.registerTool({ name: "db_connect", ... });
1141
+ pi.registerTool({ name: "db_query", ... });
1142
+ pi.registerTool({ name: "db_close", ... });
1143
+
1144
+ pi.on("session_shutdown", async () => {
1145
+ connection?.close();
1146
+ });
1147
+ }
1148
+ ```
1149
+
1150
+ ### Custom Rendering
1151
+
1152
+ Tools can provide `renderCall` and `renderResult` for custom TUI display. See [tui.md](tui.md) for the full component API.
1153
+
1154
+ Tool output is wrapped in a `Box` that handles padding and background. Your render methods return `Component` instances (typically `Text`).
1155
+
1156
+ #### renderCall
1157
+
1158
+ Renders the tool call (before/during execution):
1159
+
1160
+ ```typescript
1161
+ import { Text } from "@mariozechner/pi-tui";
1162
+
1163
+ renderCall(args, theme) {
1164
+ let text = theme.fg("toolTitle", theme.bold("my_tool "));
1165
+ text += theme.fg("muted", args.action);
1166
+ if (args.text) {
1167
+ text += " " + theme.fg("dim", `"${args.text}"`);
1168
+ }
1169
+ return new Text(text, 0, 0); // 0,0 padding - Box handles it
1170
+ }
1171
+ ```
1172
+
1173
+ #### renderResult
1174
+
1175
+ Renders the tool result:
1176
+
1177
+ ```typescript
1178
+ renderResult(result, { expanded, isPartial }, theme) {
1179
+ // Handle streaming
1180
+ if (isPartial) {
1181
+ return new Text(theme.fg("warning", "Processing..."), 0, 0);
1182
+ }
1183
+
1184
+ // Handle errors
1185
+ if (result.details?.error) {
1186
+ return new Text(theme.fg("error", `Error: ${result.details.error}`), 0, 0);
1187
+ }
1188
+
1189
+ // Normal result - support expanded view (Ctrl+O)
1190
+ let text = theme.fg("success", "✓ Done");
1191
+ if (expanded && result.details?.items) {
1192
+ for (const item of result.details.items) {
1193
+ text += "\n " + theme.fg("dim", item);
1194
+ }
1195
+ }
1196
+ return new Text(text, 0, 0);
1197
+ }
1198
+ ```
1199
+
1200
+ #### Best Practices
1201
+
1202
+ - Use `Text` with padding `(0, 0)` - the Box handles padding
1203
+ - Use `\n` for multi-line content
1204
+ - Handle `isPartial` for streaming progress
1205
+ - Support `expanded` for detail on demand
1206
+ - Keep default view compact
1207
+
1208
+ #### Fallback
1209
+
1210
+ If `renderCall`/`renderResult` is not defined or throws:
1211
+ - `renderCall`: Shows tool name
1212
+ - `renderResult`: Shows raw text from `content`
1213
+
1214
+ ## Custom UI
1215
+
1216
+ Extensions can interact with users via `ctx.ui` methods and customize how messages/tools render.
1217
+
1218
+ **For custom components, see [tui.md](tui.md)** which has copy-paste patterns for:
1219
+ - Selection dialogs (SelectList)
1220
+ - Async operations with cancel (BorderedLoader)
1221
+ - Settings toggles (SettingsList)
1222
+ - Status indicators (setStatus)
1223
+ - Working message during streaming (setWorkingMessage)
1224
+ - Widgets above editor (setWidget)
1225
+ - Custom footers (setFooter)
1226
+
1227
+ ### Dialogs
1228
+
1229
+ ```typescript
1230
+ // Select from options
1231
+ const choice = await ctx.ui.select("Pick one:", ["A", "B", "C"]);
1232
+
1233
+ // Confirm dialog
1234
+ const ok = await ctx.ui.confirm("Delete?", "This cannot be undone");
1235
+
1236
+ // Text input
1237
+ const name = await ctx.ui.input("Name:", "placeholder");
1238
+
1239
+ // Multi-line editor
1240
+ const text = await ctx.ui.editor("Edit:", "prefilled text");
1241
+
1242
+ // Notification (non-blocking)
1243
+ ctx.ui.notify("Done!", "info"); // "info" | "warning" | "error"
1244
+ ```
1245
+
1246
+ **Examples:**
1247
+ - `ctx.ui.select()`: [confirm-destructive.ts](../examples/extensions/confirm-destructive.ts), [dirty-repo-guard.ts](../examples/extensions/dirty-repo-guard.ts), [git-checkpoint.ts](../examples/extensions/git-checkpoint.ts), [permission-gate.ts](../examples/extensions/permission-gate.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [question.ts](../examples/extensions/question.ts), [questionnaire.ts](../examples/extensions/questionnaire.ts)
1248
+ - `ctx.ui.confirm()`: [confirm-destructive.ts](../examples/extensions/confirm-destructive.ts)
1249
+ - `ctx.ui.editor()`: [handoff.ts](../examples/extensions/handoff.ts)
1250
+ - `ctx.ui.setEditorText()`: [handoff.ts](../examples/extensions/handoff.ts), [qna.ts](../examples/extensions/qna.ts)
1251
+
1252
+ #### Timed Dialogs with Countdown
1253
+
1254
+ Dialogs support a `timeout` option that auto-dismisses with a live countdown display:
1255
+
1256
+ ```typescript
1257
+ // Dialog shows "Title (5s)" → "Title (4s)" → ... → auto-dismisses at 0
1258
+ const confirmed = await ctx.ui.confirm(
1259
+ "Timed Confirmation",
1260
+ "This dialog will auto-cancel in 5 seconds. Confirm?",
1261
+ { timeout: 5000 }
1262
+ );
1263
+
1264
+ if (confirmed) {
1265
+ // User confirmed
1266
+ } else {
1267
+ // User cancelled or timed out
1268
+ }
1269
+ ```
1270
+
1271
+ **Return values on timeout:**
1272
+ - `select()` returns `undefined`
1273
+ - `confirm()` returns `false`
1274
+ - `input()` returns `undefined`
1275
+
1276
+ #### Manual Dismissal with AbortSignal
1277
+
1278
+ For more control (e.g., to distinguish timeout from user cancel), use `AbortSignal`:
1279
+
1280
+ ```typescript
1281
+ const controller = new AbortController();
1282
+ const timeoutId = setTimeout(() => controller.abort(), 5000);
1283
+
1284
+ const confirmed = await ctx.ui.confirm(
1285
+ "Timed Confirmation",
1286
+ "This dialog will auto-cancel in 5 seconds. Confirm?",
1287
+ { signal: controller.signal }
1288
+ );
1289
+
1290
+ clearTimeout(timeoutId);
1291
+
1292
+ if (confirmed) {
1293
+ // User confirmed
1294
+ } else if (controller.signal.aborted) {
1295
+ // Dialog timed out
1296
+ } else {
1297
+ // User cancelled (pressed Escape or selected "No")
1298
+ }
1299
+ ```
1300
+
1301
+ See [examples/extensions/timed-confirm.ts](../examples/extensions/timed-confirm.ts) for complete examples.
1302
+
1303
+ ### Widgets, Status, and Footer
1304
+
1305
+ ```typescript
1306
+ // Status in footer (persistent until cleared)
1307
+ ctx.ui.setStatus("my-ext", "Processing...");
1308
+ ctx.ui.setStatus("my-ext", undefined); // Clear
1309
+
1310
+ // Working message (shown during streaming)
1311
+ ctx.ui.setWorkingMessage("Thinking deeply...");
1312
+ ctx.ui.setWorkingMessage(); // Restore default
1313
+
1314
+ // Widget above editor (string array or factory function)
1315
+ ctx.ui.setWidget("my-widget", ["Line 1", "Line 2"]);
1316
+ ctx.ui.setWidget("my-widget", (tui, theme) => new Text(theme.fg("accent", "Custom"), 0, 0));
1317
+ ctx.ui.setWidget("my-widget", undefined); // Clear
1318
+
1319
+ // Custom footer (replaces built-in footer entirely)
1320
+ ctx.ui.setFooter((tui, theme) => ({
1321
+ render(width) { return [theme.fg("dim", "Custom footer")]; },
1322
+ invalidate() {},
1323
+ }));
1324
+ ctx.ui.setFooter(undefined); // Restore built-in footer
1325
+
1326
+ // Terminal title
1327
+ ctx.ui.setTitle("pi - my-project");
1328
+
1329
+ // Editor text
1330
+ ctx.ui.setEditorText("Prefill text");
1331
+ const current = ctx.ui.getEditorText();
1332
+
1333
+ // Custom editor (vim mode, emacs mode, etc.)
1334
+ ctx.ui.setEditorComponent((tui, theme, keybindings) => new VimEditor(tui, theme, keybindings));
1335
+ ctx.ui.setEditorComponent(undefined); // Restore default editor
1336
+
1337
+ // Theme management
1338
+ const themes = ctx.ui.getAllThemes(); // [{ name: "dark", path: "/..." | undefined }, ...]
1339
+ const lightTheme = ctx.ui.getTheme("light"); // Load without switching
1340
+ const result = ctx.ui.setTheme("light"); // Switch by name
1341
+ if (!result.success) {
1342
+ ctx.ui.notify(`Failed: ${result.error}`, "error");
1343
+ }
1344
+ ctx.ui.setTheme(lightTheme!); // Or switch by Theme object
1345
+ ctx.ui.theme.fg("accent", "styled text"); // Access current theme
1346
+ ```
1347
+
1348
+ **Examples:**
1349
+ - `ctx.ui.setStatus()`: [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [status-line.ts](../examples/extensions/status-line.ts)
1350
+ - `ctx.ui.setWidget()`: [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts)
1351
+ - `ctx.ui.setFooter()`: [custom-footer.ts](../examples/extensions/custom-footer.ts)
1352
+ - `ctx.ui.setHeader()`: [custom-header.ts](../examples/extensions/custom-header.ts)
1353
+ - `ctx.ui.setEditorComponent()`: [modal-editor.ts](../examples/extensions/modal-editor.ts)
1354
+ - `ctx.ui.setTheme()`: [mac-system-theme.ts](../examples/extensions/mac-system-theme.ts)
1355
+
1356
+ ### Custom Components
1357
+
1358
+ For complex UI, use `ctx.ui.custom()`. This temporarily replaces the editor with your component until `done()` is called:
1359
+
1360
+ ```typescript
1361
+ import { Text, Component } from "@mariozechner/pi-tui";
1362
+
1363
+ const result = await ctx.ui.custom<boolean>((tui, theme, keybindings, done) => {
1364
+ const text = new Text("Press Enter to confirm, Escape to cancel", 1, 1);
1365
+
1366
+ text.onKey = (key) => {
1367
+ if (key === "return") done(true);
1368
+ if (key === "escape") done(false);
1369
+ return true;
1370
+ };
1371
+
1372
+ return text;
1373
+ });
1374
+
1375
+ if (result) {
1376
+ // User pressed Enter
1377
+ }
1378
+ ```
1379
+
1380
+ The callback receives:
1381
+ - `tui` - TUI instance (for screen dimensions, focus management)
1382
+ - `theme` - Current theme for styling
1383
+ - `keybindings` - App keybinding manager (for checking shortcuts)
1384
+ - `done(value)` - Call to close component and return value
1385
+
1386
+ See [tui.md](tui.md) for the full component API.
1387
+
1388
+ #### Overlay Mode (Experimental)
1389
+
1390
+ Pass `{ overlay: true }` to render the component as a floating modal on top of existing content, without clearing the screen:
1391
+
1392
+ ```typescript
1393
+ const result = await ctx.ui.custom<string | null>(
1394
+ (tui, theme, keybindings, done) => new MyOverlayComponent({ onClose: done }),
1395
+ { overlay: true }
1396
+ );
1397
+ ```
1398
+
1399
+ For advanced positioning (anchors, margins, percentages, responsive visibility), pass `overlayOptions`. Use `onHandle` to control visibility programmatically:
1400
+
1401
+ ```typescript
1402
+ const result = await ctx.ui.custom<string | null>(
1403
+ (tui, theme, keybindings, done) => new MyOverlayComponent({ onClose: done }),
1404
+ {
1405
+ overlay: true,
1406
+ overlayOptions: { anchor: "top-right", width: "50%", margin: 2 },
1407
+ onHandle: (handle) => { /* handle.setHidden(true/false) */ }
1408
+ }
1409
+ );
1410
+ ```
1411
+
1412
+ See [tui.md](tui.md) for the full `OverlayOptions` API and [overlay-qa-tests.ts](../examples/extensions/overlay-qa-tests.ts) for examples.
1413
+
1414
+ **Examples:** [handoff.ts](../examples/extensions/handoff.ts), [plan-mode/index.ts](../examples/extensions/plan-mode/index.ts), [preset.ts](../examples/extensions/preset.ts), [qna.ts](../examples/extensions/qna.ts), [snake.ts](../examples/extensions/snake.ts), [summarize.ts](../examples/extensions/summarize.ts), [todo.ts](../examples/extensions/todo.ts), [tools.ts](../examples/extensions/tools.ts), [overlay-test.ts](../examples/extensions/overlay-test.ts)
1415
+
1416
+ ### Custom Editor
1417
+
1418
+ Replace the main input editor with a custom implementation (vim mode, emacs mode, etc.):
1419
+
1420
+ ```typescript
1421
+ import { CustomEditor, type ExtensionAPI } from "@mariozechner/pi-coding-agent";
1422
+ import { matchesKey } from "@mariozechner/pi-tui";
1423
+
1424
+ class VimEditor extends CustomEditor {
1425
+ private mode: "normal" | "insert" = "insert";
1426
+
1427
+ handleInput(data: string): void {
1428
+ if (matchesKey(data, "escape") && this.mode === "insert") {
1429
+ this.mode = "normal";
1430
+ return;
1431
+ }
1432
+ if (this.mode === "normal" && data === "i") {
1433
+ this.mode = "insert";
1434
+ return;
1435
+ }
1436
+ super.handleInput(data); // App keybindings + text editing
1437
+ }
1438
+ }
1439
+
1440
+ export default function (pi: ExtensionAPI) {
1441
+ pi.on("session_start", (_event, ctx) => {
1442
+ ctx.ui.setEditorComponent((_tui, theme, keybindings) =>
1443
+ new VimEditor(theme, keybindings)
1444
+ );
1445
+ });
1446
+ }
1447
+ ```
1448
+
1449
+ **Key points:**
1450
+ - Extend `CustomEditor` (not base `Editor`) to get app keybindings (escape to abort, ctrl+d, model switching)
1451
+ - Call `super.handleInput(data)` for keys you don't handle
1452
+ - Factory receives `theme` and `keybindings` from the app
1453
+ - Pass `undefined` to restore default: `ctx.ui.setEditorComponent(undefined)`
1454
+
1455
+ See [tui.md](tui.md) Pattern 7 for a complete example with mode indicator.
1456
+
1457
+ **Examples:** [modal-editor.ts](../examples/extensions/modal-editor.ts)
1458
+
1459
+ ### Message Rendering
1460
+
1461
+ Register a custom renderer for messages with your `customType`:
1462
+
1463
+ ```typescript
1464
+ import { Text } from "@mariozechner/pi-tui";
1465
+
1466
+ pi.registerMessageRenderer("my-extension", (message, options, theme) => {
1467
+ const { expanded } = options;
1468
+ let text = theme.fg("accent", `[${message.customType}] `);
1469
+ text += message.content;
1470
+
1471
+ if (expanded && message.details) {
1472
+ text += "\n" + theme.fg("dim", JSON.stringify(message.details, null, 2));
1473
+ }
1474
+
1475
+ return new Text(text, 0, 0);
1476
+ });
1477
+ ```
1478
+
1479
+ Messages are sent via `pi.sendMessage()`:
1480
+
1481
+ ```typescript
1482
+ pi.sendMessage({
1483
+ customType: "my-extension", // Matches registerMessageRenderer
1484
+ content: "Status update",
1485
+ display: true, // Show in TUI
1486
+ details: { ... }, // Available in renderer
1487
+ });
1488
+ ```
1489
+
1490
+ ### Theme Colors
1491
+
1492
+ All render functions receive a `theme` object:
1493
+
1494
+ ```typescript
1495
+ // Foreground colors
1496
+ theme.fg("toolTitle", text) // Tool names
1497
+ theme.fg("accent", text) // Highlights
1498
+ theme.fg("success", text) // Success (green)
1499
+ theme.fg("error", text) // Errors (red)
1500
+ theme.fg("warning", text) // Warnings (yellow)
1501
+ theme.fg("muted", text) // Secondary text
1502
+ theme.fg("dim", text) // Tertiary text
1503
+
1504
+ // Text styles
1505
+ theme.bold(text)
1506
+ theme.italic(text)
1507
+ theme.strikethrough(text)
1508
+ ```
1509
+
1510
+ ## Error Handling
1511
+
1512
+ - Extension errors are logged, agent continues
1513
+ - `tool_call` errors block the tool (fail-safe)
1514
+ - Tool `execute` errors are reported to the LLM with `isError: true`
1515
+
1516
+ ## Mode Behavior
1517
+
1518
+ | Mode | UI Methods | Notes |
1519
+ |------|-----------|-------|
1520
+ | Interactive | Full TUI | Normal operation |
1521
+ | RPC | JSON protocol | Host handles UI |
1522
+ | Print (`-p`) | No-op | Extensions run but can't prompt |
1523
+
1524
+ In print mode, check `ctx.hasUI` before using UI methods.