@skaft/hamr 0.4.1

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