@oh-my-pi/pi-coding-agent 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (337) hide show
  1. package/CHANGELOG.md +1629 -0
  2. package/README.md +1041 -0
  3. package/docs/compaction.md +403 -0
  4. package/docs/config-usage.md +113 -0
  5. package/docs/custom-tools.md +541 -0
  6. package/docs/extension-loading.md +1004 -0
  7. package/docs/hooks.md +867 -0
  8. package/docs/rpc.md +1040 -0
  9. package/docs/sdk.md +994 -0
  10. package/docs/session-tree-plan.md +441 -0
  11. package/docs/session.md +240 -0
  12. package/docs/skills.md +290 -0
  13. package/docs/theme.md +670 -0
  14. package/docs/tree.md +197 -0
  15. package/docs/tui.md +341 -0
  16. package/examples/README.md +21 -0
  17. package/examples/custom-tools/README.md +124 -0
  18. package/examples/custom-tools/hello/index.ts +20 -0
  19. package/examples/custom-tools/question/index.ts +84 -0
  20. package/examples/custom-tools/subagent/README.md +172 -0
  21. package/examples/custom-tools/subagent/agents/planner.md +37 -0
  22. package/examples/custom-tools/subagent/agents/scout.md +50 -0
  23. package/examples/custom-tools/subagent/agents/worker.md +24 -0
  24. package/examples/custom-tools/subagent/agents.ts +156 -0
  25. package/examples/custom-tools/subagent/commands/implement-and-review.md +10 -0
  26. package/examples/custom-tools/subagent/commands/implement.md +10 -0
  27. package/examples/custom-tools/subagent/commands/scout-and-plan.md +9 -0
  28. package/examples/custom-tools/subagent/index.ts +1002 -0
  29. package/examples/custom-tools/todo/index.ts +212 -0
  30. package/examples/hooks/README.md +56 -0
  31. package/examples/hooks/auto-commit-on-exit.ts +49 -0
  32. package/examples/hooks/confirm-destructive.ts +59 -0
  33. package/examples/hooks/custom-compaction.ts +116 -0
  34. package/examples/hooks/dirty-repo-guard.ts +52 -0
  35. package/examples/hooks/file-trigger.ts +41 -0
  36. package/examples/hooks/git-checkpoint.ts +53 -0
  37. package/examples/hooks/handoff.ts +150 -0
  38. package/examples/hooks/permission-gate.ts +34 -0
  39. package/examples/hooks/protected-paths.ts +30 -0
  40. package/examples/hooks/qna.ts +119 -0
  41. package/examples/hooks/snake.ts +343 -0
  42. package/examples/hooks/status-line.ts +40 -0
  43. package/examples/sdk/01-minimal.ts +22 -0
  44. package/examples/sdk/02-custom-model.ts +49 -0
  45. package/examples/sdk/03-custom-prompt.ts +44 -0
  46. package/examples/sdk/04-skills.ts +44 -0
  47. package/examples/sdk/05-tools.ts +90 -0
  48. package/examples/sdk/06-hooks.ts +61 -0
  49. package/examples/sdk/07-context-files.ts +36 -0
  50. package/examples/sdk/08-slash-commands.ts +42 -0
  51. package/examples/sdk/09-api-keys-and-oauth.ts +55 -0
  52. package/examples/sdk/10-settings.ts +38 -0
  53. package/examples/sdk/11-sessions.ts +48 -0
  54. package/examples/sdk/12-full-control.ts +95 -0
  55. package/examples/sdk/README.md +154 -0
  56. package/package.json +89 -0
  57. package/src/bun-imports.d.ts +16 -0
  58. package/src/capability/context-file.ts +40 -0
  59. package/src/capability/extension.ts +48 -0
  60. package/src/capability/hook.ts +40 -0
  61. package/src/capability/index.ts +616 -0
  62. package/src/capability/instruction.ts +37 -0
  63. package/src/capability/mcp.ts +52 -0
  64. package/src/capability/prompt.ts +35 -0
  65. package/src/capability/rule.ts +56 -0
  66. package/src/capability/settings.ts +35 -0
  67. package/src/capability/skill.ts +49 -0
  68. package/src/capability/slash-command.ts +40 -0
  69. package/src/capability/system-prompt.ts +35 -0
  70. package/src/capability/tool.ts +38 -0
  71. package/src/capability/types.ts +166 -0
  72. package/src/cli/args.ts +259 -0
  73. package/src/cli/file-processor.ts +121 -0
  74. package/src/cli/list-models.ts +104 -0
  75. package/src/cli/plugin-cli.ts +661 -0
  76. package/src/cli/session-picker.ts +41 -0
  77. package/src/cli/update-cli.ts +274 -0
  78. package/src/cli.ts +10 -0
  79. package/src/config.ts +391 -0
  80. package/src/core/agent-session.ts +2178 -0
  81. package/src/core/auth-storage.ts +258 -0
  82. package/src/core/bash-executor.ts +197 -0
  83. package/src/core/compaction/branch-summarization.ts +315 -0
  84. package/src/core/compaction/compaction.ts +664 -0
  85. package/src/core/compaction/index.ts +7 -0
  86. package/src/core/compaction/utils.ts +153 -0
  87. package/src/core/custom-commands/bundled/review/index.ts +156 -0
  88. package/src/core/custom-commands/index.ts +15 -0
  89. package/src/core/custom-commands/loader.ts +226 -0
  90. package/src/core/custom-commands/types.ts +112 -0
  91. package/src/core/custom-tools/index.ts +22 -0
  92. package/src/core/custom-tools/loader.ts +248 -0
  93. package/src/core/custom-tools/types.ts +185 -0
  94. package/src/core/custom-tools/wrapper.ts +29 -0
  95. package/src/core/exec.ts +139 -0
  96. package/src/core/export-html/index.ts +159 -0
  97. package/src/core/export-html/template.css +774 -0
  98. package/src/core/export-html/template.generated.ts +2 -0
  99. package/src/core/export-html/template.html +45 -0
  100. package/src/core/export-html/template.js +1185 -0
  101. package/src/core/export-html/template.macro.ts +24 -0
  102. package/src/core/file-mentions.ts +54 -0
  103. package/src/core/hooks/index.ts +16 -0
  104. package/src/core/hooks/loader.ts +288 -0
  105. package/src/core/hooks/runner.ts +434 -0
  106. package/src/core/hooks/tool-wrapper.ts +98 -0
  107. package/src/core/hooks/types.ts +770 -0
  108. package/src/core/index.ts +53 -0
  109. package/src/core/logger.ts +112 -0
  110. package/src/core/mcp/client.ts +185 -0
  111. package/src/core/mcp/config.ts +248 -0
  112. package/src/core/mcp/index.ts +45 -0
  113. package/src/core/mcp/loader.ts +99 -0
  114. package/src/core/mcp/manager.ts +235 -0
  115. package/src/core/mcp/tool-bridge.ts +156 -0
  116. package/src/core/mcp/transports/http.ts +316 -0
  117. package/src/core/mcp/transports/index.ts +6 -0
  118. package/src/core/mcp/transports/stdio.ts +252 -0
  119. package/src/core/mcp/types.ts +228 -0
  120. package/src/core/messages.ts +211 -0
  121. package/src/core/model-registry.ts +334 -0
  122. package/src/core/model-resolver.ts +494 -0
  123. package/src/core/plugins/doctor.ts +67 -0
  124. package/src/core/plugins/index.ts +38 -0
  125. package/src/core/plugins/installer.ts +189 -0
  126. package/src/core/plugins/loader.ts +339 -0
  127. package/src/core/plugins/manager.ts +672 -0
  128. package/src/core/plugins/parser.ts +105 -0
  129. package/src/core/plugins/paths.ts +37 -0
  130. package/src/core/plugins/types.ts +190 -0
  131. package/src/core/sdk.ts +900 -0
  132. package/src/core/session-manager.ts +1837 -0
  133. package/src/core/settings-manager.ts +860 -0
  134. package/src/core/skills.ts +352 -0
  135. package/src/core/slash-commands.ts +132 -0
  136. package/src/core/system-prompt.ts +442 -0
  137. package/src/core/timings.ts +25 -0
  138. package/src/core/title-generator.ts +110 -0
  139. package/src/core/tools/ask.ts +193 -0
  140. package/src/core/tools/bash-interceptor.ts +120 -0
  141. package/src/core/tools/bash.ts +91 -0
  142. package/src/core/tools/context.ts +32 -0
  143. package/src/core/tools/edit-diff.ts +487 -0
  144. package/src/core/tools/edit.ts +140 -0
  145. package/src/core/tools/exa/company.ts +59 -0
  146. package/src/core/tools/exa/index.ts +63 -0
  147. package/src/core/tools/exa/linkedin.ts +59 -0
  148. package/src/core/tools/exa/mcp-client.ts +368 -0
  149. package/src/core/tools/exa/render.ts +200 -0
  150. package/src/core/tools/exa/researcher.ts +90 -0
  151. package/src/core/tools/exa/search.ts +338 -0
  152. package/src/core/tools/exa/types.ts +167 -0
  153. package/src/core/tools/exa/websets.ts +248 -0
  154. package/src/core/tools/find.ts +244 -0
  155. package/src/core/tools/grep.ts +584 -0
  156. package/src/core/tools/index.ts +283 -0
  157. package/src/core/tools/ls.ts +142 -0
  158. package/src/core/tools/lsp/client.ts +767 -0
  159. package/src/core/tools/lsp/clients/biome-client.ts +207 -0
  160. package/src/core/tools/lsp/clients/index.ts +49 -0
  161. package/src/core/tools/lsp/clients/lsp-linter-client.ts +98 -0
  162. package/src/core/tools/lsp/config.ts +845 -0
  163. package/src/core/tools/lsp/edits.ts +110 -0
  164. package/src/core/tools/lsp/index.ts +1364 -0
  165. package/src/core/tools/lsp/render.ts +560 -0
  166. package/src/core/tools/lsp/rust-analyzer.ts +145 -0
  167. package/src/core/tools/lsp/types.ts +495 -0
  168. package/src/core/tools/lsp/utils.ts +526 -0
  169. package/src/core/tools/notebook.ts +182 -0
  170. package/src/core/tools/output.ts +198 -0
  171. package/src/core/tools/path-utils.ts +61 -0
  172. package/src/core/tools/read.ts +507 -0
  173. package/src/core/tools/renderers.ts +820 -0
  174. package/src/core/tools/review.ts +275 -0
  175. package/src/core/tools/rulebook.ts +124 -0
  176. package/src/core/tools/task/agents.ts +158 -0
  177. package/src/core/tools/task/artifacts.ts +114 -0
  178. package/src/core/tools/task/commands.ts +157 -0
  179. package/src/core/tools/task/discovery.ts +217 -0
  180. package/src/core/tools/task/executor.ts +531 -0
  181. package/src/core/tools/task/index.ts +548 -0
  182. package/src/core/tools/task/model-resolver.ts +176 -0
  183. package/src/core/tools/task/parallel.ts +38 -0
  184. package/src/core/tools/task/render.ts +502 -0
  185. package/src/core/tools/task/subprocess-tool-registry.ts +89 -0
  186. package/src/core/tools/task/types.ts +142 -0
  187. package/src/core/tools/truncate.ts +265 -0
  188. package/src/core/tools/web-fetch.ts +2511 -0
  189. package/src/core/tools/web-search/auth.ts +199 -0
  190. package/src/core/tools/web-search/index.ts +583 -0
  191. package/src/core/tools/web-search/providers/anthropic.ts +198 -0
  192. package/src/core/tools/web-search/providers/exa.ts +196 -0
  193. package/src/core/tools/web-search/providers/perplexity.ts +195 -0
  194. package/src/core/tools/web-search/render.ts +372 -0
  195. package/src/core/tools/web-search/types.ts +180 -0
  196. package/src/core/tools/write.ts +63 -0
  197. package/src/core/ttsr.ts +211 -0
  198. package/src/core/utils.ts +187 -0
  199. package/src/discovery/agents-md.ts +75 -0
  200. package/src/discovery/builtin.ts +647 -0
  201. package/src/discovery/claude.ts +623 -0
  202. package/src/discovery/cline.ts +104 -0
  203. package/src/discovery/codex.ts +571 -0
  204. package/src/discovery/cursor.ts +266 -0
  205. package/src/discovery/gemini.ts +368 -0
  206. package/src/discovery/github.ts +120 -0
  207. package/src/discovery/helpers.test.ts +127 -0
  208. package/src/discovery/helpers.ts +249 -0
  209. package/src/discovery/index.ts +84 -0
  210. package/src/discovery/mcp-json.ts +127 -0
  211. package/src/discovery/vscode.ts +99 -0
  212. package/src/discovery/windsurf.ts +219 -0
  213. package/src/index.ts +192 -0
  214. package/src/main.ts +507 -0
  215. package/src/migrations.ts +156 -0
  216. package/src/modes/cleanup.ts +23 -0
  217. package/src/modes/index.ts +48 -0
  218. package/src/modes/interactive/components/armin.ts +382 -0
  219. package/src/modes/interactive/components/assistant-message.ts +86 -0
  220. package/src/modes/interactive/components/bash-execution.ts +199 -0
  221. package/src/modes/interactive/components/bordered-loader.ts +41 -0
  222. package/src/modes/interactive/components/branch-summary-message.ts +42 -0
  223. package/src/modes/interactive/components/compaction-summary-message.ts +45 -0
  224. package/src/modes/interactive/components/custom-editor.ts +122 -0
  225. package/src/modes/interactive/components/diff.ts +147 -0
  226. package/src/modes/interactive/components/dynamic-border.ts +25 -0
  227. package/src/modes/interactive/components/extensions/extension-dashboard.ts +296 -0
  228. package/src/modes/interactive/components/extensions/extension-list.ts +479 -0
  229. package/src/modes/interactive/components/extensions/index.ts +9 -0
  230. package/src/modes/interactive/components/extensions/inspector-panel.ts +313 -0
  231. package/src/modes/interactive/components/extensions/state-manager.ts +558 -0
  232. package/src/modes/interactive/components/extensions/types.ts +191 -0
  233. package/src/modes/interactive/components/hook-editor.ts +117 -0
  234. package/src/modes/interactive/components/hook-input.ts +64 -0
  235. package/src/modes/interactive/components/hook-message.ts +96 -0
  236. package/src/modes/interactive/components/hook-selector.ts +91 -0
  237. package/src/modes/interactive/components/model-selector.ts +560 -0
  238. package/src/modes/interactive/components/oauth-selector.ts +136 -0
  239. package/src/modes/interactive/components/plugin-settings.ts +481 -0
  240. package/src/modes/interactive/components/queue-mode-selector.ts +56 -0
  241. package/src/modes/interactive/components/session-selector.ts +220 -0
  242. package/src/modes/interactive/components/settings-defs.ts +597 -0
  243. package/src/modes/interactive/components/settings-selector.ts +545 -0
  244. package/src/modes/interactive/components/show-images-selector.ts +45 -0
  245. package/src/modes/interactive/components/status-line/index.ts +4 -0
  246. package/src/modes/interactive/components/status-line/presets.ts +94 -0
  247. package/src/modes/interactive/components/status-line/segments.ts +350 -0
  248. package/src/modes/interactive/components/status-line/separators.ts +55 -0
  249. package/src/modes/interactive/components/status-line/types.ts +81 -0
  250. package/src/modes/interactive/components/status-line-segment-editor.ts +357 -0
  251. package/src/modes/interactive/components/status-line.ts +384 -0
  252. package/src/modes/interactive/components/theme-selector.ts +62 -0
  253. package/src/modes/interactive/components/thinking-selector.ts +64 -0
  254. package/src/modes/interactive/components/tool-execution.ts +946 -0
  255. package/src/modes/interactive/components/tree-selector.ts +877 -0
  256. package/src/modes/interactive/components/ttsr-notification.ts +82 -0
  257. package/src/modes/interactive/components/user-message-selector.ts +159 -0
  258. package/src/modes/interactive/components/user-message.ts +18 -0
  259. package/src/modes/interactive/components/visual-truncate.ts +50 -0
  260. package/src/modes/interactive/components/welcome.ts +228 -0
  261. package/src/modes/interactive/interactive-mode.ts +2669 -0
  262. package/src/modes/interactive/theme/dark.json +102 -0
  263. package/src/modes/interactive/theme/defaults/dark-arctic.json +111 -0
  264. package/src/modes/interactive/theme/defaults/dark-catppuccin.json +106 -0
  265. package/src/modes/interactive/theme/defaults/dark-cyberpunk.json +109 -0
  266. package/src/modes/interactive/theme/defaults/dark-dracula.json +105 -0
  267. package/src/modes/interactive/theme/defaults/dark-forest.json +103 -0
  268. package/src/modes/interactive/theme/defaults/dark-github.json +112 -0
  269. package/src/modes/interactive/theme/defaults/dark-gruvbox.json +119 -0
  270. package/src/modes/interactive/theme/defaults/dark-monochrome.json +101 -0
  271. package/src/modes/interactive/theme/defaults/dark-monokai.json +105 -0
  272. package/src/modes/interactive/theme/defaults/dark-nord.json +104 -0
  273. package/src/modes/interactive/theme/defaults/dark-ocean.json +108 -0
  274. package/src/modes/interactive/theme/defaults/dark-one.json +107 -0
  275. package/src/modes/interactive/theme/defaults/dark-retro.json +99 -0
  276. package/src/modes/interactive/theme/defaults/dark-rose-pine.json +95 -0
  277. package/src/modes/interactive/theme/defaults/dark-solarized.json +96 -0
  278. package/src/modes/interactive/theme/defaults/dark-sunset.json +106 -0
  279. package/src/modes/interactive/theme/defaults/dark-synthwave.json +102 -0
  280. package/src/modes/interactive/theme/defaults/dark-tokyo-night.json +108 -0
  281. package/src/modes/interactive/theme/defaults/index.ts +67 -0
  282. package/src/modes/interactive/theme/defaults/light-arctic.json +106 -0
  283. package/src/modes/interactive/theme/defaults/light-catppuccin.json +105 -0
  284. package/src/modes/interactive/theme/defaults/light-cyberpunk.json +103 -0
  285. package/src/modes/interactive/theme/defaults/light-forest.json +107 -0
  286. package/src/modes/interactive/theme/defaults/light-github.json +114 -0
  287. package/src/modes/interactive/theme/defaults/light-gruvbox.json +115 -0
  288. package/src/modes/interactive/theme/defaults/light-monochrome.json +100 -0
  289. package/src/modes/interactive/theme/defaults/light-ocean.json +106 -0
  290. package/src/modes/interactive/theme/defaults/light-one.json +105 -0
  291. package/src/modes/interactive/theme/defaults/light-retro.json +105 -0
  292. package/src/modes/interactive/theme/defaults/light-solarized.json +101 -0
  293. package/src/modes/interactive/theme/defaults/light-sunset.json +106 -0
  294. package/src/modes/interactive/theme/defaults/light-synthwave.json +105 -0
  295. package/src/modes/interactive/theme/defaults/light-tokyo-night.json +118 -0
  296. package/src/modes/interactive/theme/light.json +99 -0
  297. package/src/modes/interactive/theme/theme-schema.json +424 -0
  298. package/src/modes/interactive/theme/theme.ts +2211 -0
  299. package/src/modes/print-mode.ts +163 -0
  300. package/src/modes/rpc/rpc-client.ts +527 -0
  301. package/src/modes/rpc/rpc-mode.ts +494 -0
  302. package/src/modes/rpc/rpc-types.ts +203 -0
  303. package/src/prompts/architect-plan.md +10 -0
  304. package/src/prompts/branch-summary-preamble.md +3 -0
  305. package/src/prompts/branch-summary.md +28 -0
  306. package/src/prompts/browser.md +71 -0
  307. package/src/prompts/compaction-summary.md +34 -0
  308. package/src/prompts/compaction-turn-prefix.md +16 -0
  309. package/src/prompts/compaction-update-summary.md +41 -0
  310. package/src/prompts/explore.md +82 -0
  311. package/src/prompts/implement-with-critic.md +11 -0
  312. package/src/prompts/implement.md +11 -0
  313. package/src/prompts/init.md +30 -0
  314. package/src/prompts/plan.md +54 -0
  315. package/src/prompts/reviewer.md +81 -0
  316. package/src/prompts/summarization-system.md +3 -0
  317. package/src/prompts/system-prompt.md +27 -0
  318. package/src/prompts/task.md +56 -0
  319. package/src/prompts/title-system.md +8 -0
  320. package/src/prompts/tools/ask.md +24 -0
  321. package/src/prompts/tools/bash.md +23 -0
  322. package/src/prompts/tools/edit.md +9 -0
  323. package/src/prompts/tools/find.md +6 -0
  324. package/src/prompts/tools/grep.md +12 -0
  325. package/src/prompts/tools/lsp.md +14 -0
  326. package/src/prompts/tools/output.md +23 -0
  327. package/src/prompts/tools/read.md +25 -0
  328. package/src/prompts/tools/web-fetch.md +8 -0
  329. package/src/prompts/tools/web-search.md +10 -0
  330. package/src/prompts/tools/write.md +10 -0
  331. package/src/utils/changelog.ts +99 -0
  332. package/src/utils/clipboard.ts +265 -0
  333. package/src/utils/fuzzy.ts +108 -0
  334. package/src/utils/mime.ts +30 -0
  335. package/src/utils/shell-snapshot.ts +218 -0
  336. package/src/utils/shell.ts +364 -0
  337. package/src/utils/tools-manager.ts +265 -0
@@ -0,0 +1,248 @@
1
+ /**
2
+ * Custom tool loader - loads TypeScript tool modules using native Bun import.
3
+ *
4
+ * Dependencies (@sinclair/typebox and pi-coding-agent) are injected via the CustomToolAPI
5
+ * to avoid import resolution issues with custom tools loaded from user directories.
6
+ */
7
+
8
+ import * as os from "node:os";
9
+ import * as path from "node:path";
10
+ import * as typebox from "@sinclair/typebox";
11
+ import { toolCapability } from "../../capability/tool";
12
+ import { type CustomTool, loadSync } from "../../discovery";
13
+ import * as piCodingAgent from "../../index";
14
+ import { theme } from "../../modes/interactive/theme/theme";
15
+ import type { ExecOptions } from "../exec";
16
+ import { execCommand } from "../exec";
17
+ import type { HookUIContext } from "../hooks/types";
18
+ import { logger } from "../logger";
19
+ import { getAllPluginToolPaths } from "../plugins/loader";
20
+ import type { CustomToolAPI, CustomToolFactory, CustomToolsLoadResult, LoadedCustomTool } from "./types";
21
+
22
+ const UNICODE_SPACES = /[\u00A0\u2000-\u200A\u202F\u205F\u3000]/g;
23
+
24
+ function normalizeUnicodeSpaces(str: string): string {
25
+ return str.replace(UNICODE_SPACES, " ");
26
+ }
27
+
28
+ function expandPath(p: string): string {
29
+ const normalized = normalizeUnicodeSpaces(p);
30
+ if (normalized.startsWith("~/")) {
31
+ return path.join(os.homedir(), normalized.slice(2));
32
+ }
33
+ if (normalized.startsWith("~")) {
34
+ return path.join(os.homedir(), normalized.slice(1));
35
+ }
36
+ return normalized;
37
+ }
38
+
39
+ /**
40
+ * Resolve tool path.
41
+ * - Absolute paths used as-is
42
+ * - Paths starting with ~ expanded to home directory
43
+ * - Relative paths resolved from cwd
44
+ */
45
+ function resolveToolPath(toolPath: string, cwd: string): string {
46
+ const expanded = expandPath(toolPath);
47
+
48
+ if (path.isAbsolute(expanded)) {
49
+ return expanded;
50
+ }
51
+
52
+ // Relative paths resolved from cwd
53
+ return path.resolve(cwd, expanded);
54
+ }
55
+
56
+ /**
57
+ * Create a no-op UI context for headless modes.
58
+ */
59
+ function createNoOpUIContext(): HookUIContext {
60
+ return {
61
+ select: async () => undefined,
62
+ confirm: async () => false,
63
+ input: async () => undefined,
64
+ notify: () => {},
65
+ setStatus: () => {},
66
+ custom: async () => undefined as never,
67
+ setEditorText: () => {},
68
+ getEditorText: () => "",
69
+ editor: async () => undefined,
70
+ get theme() {
71
+ return theme;
72
+ },
73
+ };
74
+ }
75
+
76
+ /** Error with source metadata */
77
+ interface ToolLoadError {
78
+ path: string;
79
+ error: string;
80
+ source?: { provider: string; providerName: string; level: "user" | "project" };
81
+ }
82
+
83
+ /**
84
+ * Load a single tool module using native Bun import.
85
+ */
86
+ async function loadTool(
87
+ toolPath: string,
88
+ cwd: string,
89
+ sharedApi: CustomToolAPI,
90
+ source?: { provider: string; providerName: string; level: "user" | "project" },
91
+ ): Promise<{ tools: LoadedCustomTool[] | null; error: ToolLoadError | null }> {
92
+ const resolvedPath = resolveToolPath(toolPath, cwd);
93
+
94
+ // Skip declarative tool files (.md, .json) - these are metadata only, not executable modules
95
+ if (resolvedPath.endsWith(".md") || resolvedPath.endsWith(".json")) {
96
+ return {
97
+ tools: null,
98
+ error: {
99
+ path: toolPath,
100
+ error: "Declarative tool files (.md, .json) cannot be loaded as executable modules",
101
+ source,
102
+ },
103
+ };
104
+ }
105
+
106
+ try {
107
+ const module = await import(resolvedPath);
108
+ const factory = (module.default ?? module) as CustomToolFactory;
109
+
110
+ if (typeof factory !== "function") {
111
+ return { tools: null, error: { path: toolPath, error: "Tool must export a default function", source } };
112
+ }
113
+
114
+ const toolResult = await factory(sharedApi);
115
+ const toolsArray = Array.isArray(toolResult) ? toolResult : [toolResult];
116
+
117
+ const loadedTools: LoadedCustomTool[] = toolsArray.map((tool) => ({
118
+ path: toolPath,
119
+ resolvedPath,
120
+ tool,
121
+ source,
122
+ }));
123
+
124
+ return { tools: loadedTools, error: null };
125
+ } catch (err) {
126
+ const message = err instanceof Error ? err.message : String(err);
127
+ return { tools: null, error: { path: toolPath, error: `Failed to load tool: ${message}`, source } };
128
+ }
129
+ }
130
+
131
+ /** Tool path with optional source metadata */
132
+ interface ToolPathWithSource {
133
+ path: string;
134
+ source?: { provider: string; providerName: string; level: "user" | "project" };
135
+ }
136
+
137
+ /**
138
+ * Load all tools from configuration.
139
+ * @param pathsWithSources - Array of tool paths with optional source metadata
140
+ * @param cwd - Current working directory for resolving relative paths
141
+ * @param builtInToolNames - Names of built-in tools to check for conflicts
142
+ */
143
+ export async function loadCustomTools(
144
+ pathsWithSources: ToolPathWithSource[],
145
+ cwd: string,
146
+ builtInToolNames: string[],
147
+ ): Promise<CustomToolsLoadResult> {
148
+ const tools: LoadedCustomTool[] = [];
149
+ const errors: ToolLoadError[] = [];
150
+ const seenNames = new Set<string>(builtInToolNames);
151
+
152
+ // Shared API object - all tools get the same instance
153
+ const sharedApi: CustomToolAPI = {
154
+ cwd,
155
+ exec: (command: string, args: string[], options?: ExecOptions) =>
156
+ execCommand(command, args, options?.cwd ?? cwd, options),
157
+ ui: createNoOpUIContext(),
158
+ hasUI: false,
159
+ logger,
160
+ typebox,
161
+ pi: piCodingAgent,
162
+ };
163
+
164
+ for (const { path: toolPath, source } of pathsWithSources) {
165
+ const { tools: loadedTools, error } = await loadTool(toolPath, cwd, sharedApi, source);
166
+
167
+ if (error) {
168
+ errors.push(error);
169
+ continue;
170
+ }
171
+
172
+ if (loadedTools) {
173
+ for (const loadedTool of loadedTools) {
174
+ // Check for name conflicts
175
+ if (seenNames.has(loadedTool.tool.name)) {
176
+ errors.push({
177
+ path: toolPath,
178
+ error: `Tool name "${loadedTool.tool.name}" conflicts with existing tool`,
179
+ source,
180
+ });
181
+ continue;
182
+ }
183
+
184
+ seenNames.add(loadedTool.tool.name);
185
+ tools.push(loadedTool);
186
+ }
187
+ }
188
+ }
189
+
190
+ return {
191
+ tools,
192
+ errors,
193
+ setUIContext(uiContext, hasUI) {
194
+ sharedApi.ui = uiContext;
195
+ sharedApi.hasUI = hasUI;
196
+ },
197
+ };
198
+ }
199
+
200
+ /**
201
+ * Discover and load tools from standard locations via capability system:
202
+ * 1. User and project tools discovered by capability providers
203
+ * 2. Installed plugins (~/.omp/plugins/node_modules/*)
204
+ * 3. Explicitly configured paths from settings or CLI
205
+ *
206
+ * @param configuredPaths - Explicit paths from settings.json and CLI --tool flags
207
+ * @param cwd - Current working directory
208
+ * @param builtInToolNames - Names of built-in tools to check for conflicts
209
+ */
210
+ export async function discoverAndLoadCustomTools(
211
+ configuredPaths: string[],
212
+ cwd: string,
213
+ builtInToolNames: string[],
214
+ ): Promise<CustomToolsLoadResult> {
215
+ const allPathsWithSources: ToolPathWithSource[] = [];
216
+ const seen = new Set<string>();
217
+
218
+ // Helper to add paths without duplicates
219
+ const addPath = (p: string, source?: { provider: string; providerName: string; level: "user" | "project" }) => {
220
+ const resolved = path.resolve(p);
221
+ if (!seen.has(resolved)) {
222
+ seen.add(resolved);
223
+ allPathsWithSources.push({ path: p, source });
224
+ }
225
+ };
226
+
227
+ // 1. Discover tools via capability system (user + project from all providers)
228
+ const discoveredTools = loadSync<CustomTool>(toolCapability.id, { cwd });
229
+ for (const tool of discoveredTools.items) {
230
+ addPath(tool.path, {
231
+ provider: tool._source.provider,
232
+ providerName: tool._source.providerName,
233
+ level: tool.level,
234
+ });
235
+ }
236
+
237
+ // 2. Plugin tools: ~/.omp/plugins/node_modules/*/
238
+ for (const pluginPath of getAllPluginToolPaths(cwd)) {
239
+ addPath(pluginPath, { provider: "plugin", providerName: "Plugin", level: "user" });
240
+ }
241
+
242
+ // 3. Explicitly configured paths (can override/add)
243
+ for (const configPath of configuredPaths) {
244
+ addPath(resolveToolPath(configPath, cwd), { provider: "config", providerName: "Config", level: "project" });
245
+ }
246
+
247
+ return loadCustomTools(allPathsWithSources, cwd, builtInToolNames);
248
+ }
@@ -0,0 +1,185 @@
1
+ /**
2
+ * Custom tool types.
3
+ *
4
+ * Custom tools are TypeScript modules that define additional tools for the agent.
5
+ * They can provide custom rendering for tool calls and results in the TUI.
6
+ */
7
+
8
+ import type { AgentToolResult, AgentToolUpdateCallback } from "@oh-my-pi/pi-agent-core";
9
+ import type { Model } from "@oh-my-pi/pi-ai";
10
+ import type { Component } from "@oh-my-pi/pi-tui";
11
+ import type { Static, TSchema } from "@sinclair/typebox";
12
+ import type { Theme } from "../../modes/interactive/theme/theme";
13
+ import type { ExecOptions, ExecResult } from "../exec";
14
+ import type { HookUIContext } from "../hooks/types";
15
+ import type { Logger } from "../logger";
16
+ import type { ModelRegistry } from "../model-registry";
17
+ import type { ReadonlySessionManager } from "../session-manager";
18
+
19
+ /** Alias for clarity */
20
+ export type CustomToolUIContext = HookUIContext;
21
+
22
+ /** Re-export for custom tools to use in execute signature */
23
+ export type { AgentToolResult, AgentToolUpdateCallback };
24
+
25
+ // Re-export for backward compatibility
26
+ export type { ExecOptions, ExecResult } from "../exec";
27
+
28
+ /** API passed to custom tool factory (stable across session changes) */
29
+ export interface CustomToolAPI {
30
+ /** Current working directory */
31
+ cwd: string;
32
+ /** Execute a command */
33
+ exec(command: string, args: string[], options?: ExecOptions): Promise<ExecResult>;
34
+ /** UI methods for user interaction (select, confirm, input, notify, custom) */
35
+ ui: CustomToolUIContext;
36
+ /** Whether UI is available (false in print/RPC mode) */
37
+ hasUI: boolean;
38
+ /** File logger for error/warning/debug messages */
39
+ logger: Logger;
40
+ /** Injected @sinclair/typebox module */
41
+ typebox: typeof import("@sinclair/typebox");
42
+ /** Injected pi-coding-agent exports */
43
+ pi: typeof import("../../index.js");
44
+ }
45
+
46
+ /**
47
+ * Context passed to tool execute and onSession callbacks.
48
+ * Provides access to session state and model information.
49
+ */
50
+ export interface CustomToolContext {
51
+ /** Session manager (read-only) */
52
+ sessionManager: ReadonlySessionManager;
53
+ /** Model registry - use for API key resolution and model retrieval */
54
+ modelRegistry: ModelRegistry;
55
+ /** Current model (may be undefined if no model is selected yet) */
56
+ model: Model<any> | undefined;
57
+ /** Whether the agent is idle (not streaming) */
58
+ isIdle(): boolean;
59
+ /** Whether there are queued messages waiting to be processed */
60
+ hasQueuedMessages(): boolean;
61
+ /** Abort the current agent operation (fire-and-forget, does not wait) */
62
+ abort(): void;
63
+ }
64
+
65
+ /** Session event passed to onSession callback */
66
+ export interface CustomToolSessionEvent {
67
+ /** Reason for the session event */
68
+ reason: "start" | "switch" | "branch" | "tree" | "shutdown";
69
+ /** Previous session file path, or undefined for "start" and "shutdown" */
70
+ previousSessionFile: string | undefined;
71
+ }
72
+
73
+ /** Rendering options passed to renderResult */
74
+ export interface RenderResultOptions {
75
+ /** Whether the result view is expanded */
76
+ expanded: boolean;
77
+ /** Whether this is a partial/streaming result */
78
+ isPartial: boolean;
79
+ /** Current spinner frame index for animated elements (0-9, only provided during partial results) */
80
+ spinnerFrame?: number;
81
+ }
82
+
83
+ export type CustomToolResult<TDetails = any> = AgentToolResult<TDetails>;
84
+
85
+ /**
86
+ * Custom tool definition.
87
+ *
88
+ * Custom tools are standalone - they don't extend AgentTool directly.
89
+ * When loaded, they are wrapped in an AgentTool for the agent to use.
90
+ *
91
+ * The execute callback receives a ToolContext with access to session state,
92
+ * model registry, and current model.
93
+ *
94
+ * @example
95
+ * ```typescript
96
+ * const factory: CustomToolFactory = (pi) => ({
97
+ * name: "my_tool",
98
+ * label: "My Tool",
99
+ * description: "Does something useful",
100
+ * parameters: Type.Object({ input: Type.String() }),
101
+ *
102
+ * async execute(toolCallId, params, onUpdate, ctx, signal) {
103
+ * // Access session state via ctx.sessionManager
104
+ * // Access model registry via ctx.modelRegistry
105
+ * // Current model via ctx.model
106
+ * return { content: [{ type: "text", text: "Done" }] };
107
+ * },
108
+ *
109
+ * onSession(event, ctx) {
110
+ * if (event.reason === "shutdown") {
111
+ * // Cleanup
112
+ * }
113
+ * // Reconstruct state from ctx.sessionManager.getEntries()
114
+ * }
115
+ * });
116
+ * ```
117
+ */
118
+ export interface CustomTool<TParams extends TSchema = TSchema, TDetails = any> {
119
+ /** Tool name (used in LLM tool calls) */
120
+ name: string;
121
+ /** Human-readable label for UI */
122
+ label: string;
123
+ /** Description for LLM */
124
+ description: string;
125
+ /** Parameter schema (TypeBox) */
126
+ parameters: TParams;
127
+ /** If true, tool is excluded unless explicitly listed in --tools or agent's tools field */
128
+ hidden?: boolean;
129
+
130
+ /**
131
+ * Execute the tool.
132
+ * @param toolCallId - Unique ID for this tool call
133
+ * @param params - Parsed parameters matching the schema
134
+ * @param onUpdate - Callback for streaming partial results (for UI, not LLM)
135
+ * @param ctx - Context with session manager, model registry, and current model
136
+ * @param signal - Optional abort signal for cancellation
137
+ */
138
+ execute(
139
+ toolCallId: string,
140
+ params: Static<TParams>,
141
+ onUpdate: AgentToolUpdateCallback<TDetails> | undefined,
142
+ ctx: CustomToolContext,
143
+ signal?: AbortSignal,
144
+ ): Promise<AgentToolResult<TDetails>>;
145
+
146
+ /** Called on session lifecycle events - use to reconstruct state or cleanup resources */
147
+ onSession?: (event: CustomToolSessionEvent, ctx: CustomToolContext) => void | Promise<void>;
148
+ /** Custom rendering for tool call display - return a Component */
149
+ renderCall?: (args: Static<TParams>, theme: Theme) => Component;
150
+
151
+ /** Custom rendering for tool result display - return a Component */
152
+ renderResult?: (result: CustomToolResult<TDetails>, options: RenderResultOptions, theme: Theme) => Component;
153
+ }
154
+
155
+ /** Factory function that creates a custom tool or array of tools */
156
+ export type CustomToolFactory = (
157
+ pi: CustomToolAPI,
158
+ ) => CustomTool<any, any> | CustomTool<any, any>[] | Promise<CustomTool<any, any> | CustomTool<any, any>[]>;
159
+
160
+ /** Loaded custom tool with metadata and wrapped AgentTool */
161
+ export interface LoadedCustomTool {
162
+ /** Original path (as specified) */
163
+ path: string;
164
+ /** Resolved absolute path */
165
+ resolvedPath: string;
166
+ /** The original custom tool instance */
167
+ tool: CustomTool;
168
+ /** Source metadata (provider and level) */
169
+ source?: { provider: string; providerName: string; level: "user" | "project" };
170
+ }
171
+
172
+ /** Error with source metadata */
173
+ export interface ToolLoadError {
174
+ path: string;
175
+ error: string;
176
+ source?: { provider: string; providerName: string; level: "user" | "project" };
177
+ }
178
+
179
+ /** Result from loading custom tools */
180
+ export interface CustomToolsLoadResult {
181
+ tools: LoadedCustomTool[];
182
+ errors: ToolLoadError[];
183
+ /** Update the UI context for all loaded tools. Call when mode initializes. */
184
+ setUIContext(uiContext: CustomToolUIContext, hasUI: boolean): void;
185
+ }
@@ -0,0 +1,29 @@
1
+ /**
2
+ * Wraps CustomTool instances into AgentTool for use with the agent.
3
+ */
4
+
5
+ import type { AgentTool } from "@oh-my-pi/pi-agent-core";
6
+ import type { CustomTool, CustomToolContext, LoadedCustomTool } from "./types";
7
+
8
+ /**
9
+ * Wrap a CustomTool into an AgentTool.
10
+ * The wrapper injects the ToolContext into execute calls.
11
+ */
12
+ export function wrapCustomTool(tool: CustomTool, getContext: () => CustomToolContext): AgentTool {
13
+ return {
14
+ name: tool.name,
15
+ label: tool.label,
16
+ description: tool.description,
17
+ parameters: tool.parameters,
18
+ hidden: tool.hidden,
19
+ execute: (toolCallId, params, signal, onUpdate, context) =>
20
+ tool.execute(toolCallId, params, onUpdate, context ?? getContext(), signal),
21
+ };
22
+ }
23
+
24
+ /**
25
+ * Wrap all loaded custom tools into AgentTools.
26
+ */
27
+ export function wrapCustomTools(loadedTools: LoadedCustomTool[], getContext: () => CustomToolContext): AgentTool[] {
28
+ return loadedTools.map((lt) => wrapCustomTool(lt.tool, getContext));
29
+ }
@@ -0,0 +1,139 @@
1
+ /**
2
+ * Shared command execution utilities for hooks and custom tools.
3
+ */
4
+
5
+ import type { Subprocess } from "bun";
6
+ import { logger } from "./logger";
7
+
8
+ /**
9
+ * Options for executing shell commands.
10
+ */
11
+ export interface ExecOptions {
12
+ /** AbortSignal to cancel the command */
13
+ signal?: AbortSignal;
14
+ /** Timeout in milliseconds */
15
+ timeout?: number;
16
+ /** Working directory */
17
+ cwd?: string;
18
+ }
19
+
20
+ /**
21
+ * Result of executing a shell command.
22
+ */
23
+ export interface ExecResult {
24
+ stdout: string;
25
+ stderr: string;
26
+ code: number;
27
+ killed: boolean;
28
+ }
29
+
30
+ /**
31
+ * Execute a shell command and return stdout/stderr/code.
32
+ * Supports timeout and abort signal.
33
+ */
34
+ export async function execCommand(
35
+ command: string,
36
+ args: string[],
37
+ cwd: string,
38
+ options?: ExecOptions,
39
+ ): Promise<ExecResult> {
40
+ return new Promise((resolve) => {
41
+ const proc: Subprocess = Bun.spawn([command, ...args], {
42
+ cwd,
43
+ stdin: "ignore",
44
+ stdout: "pipe",
45
+ stderr: "pipe",
46
+ });
47
+
48
+ let stdout = "";
49
+ let stderr = "";
50
+ let killed = false;
51
+ let timeoutId: Timer | undefined;
52
+
53
+ const killProcess = () => {
54
+ if (!killed) {
55
+ killed = true;
56
+ proc.kill();
57
+ // Force kill after 5 seconds if first kill doesn't work
58
+ setTimeout(() => {
59
+ try {
60
+ proc.kill(9);
61
+ } catch {
62
+ // Ignore if already dead
63
+ }
64
+ }, 5000);
65
+ }
66
+ };
67
+
68
+ // Handle abort signal
69
+ if (options?.signal) {
70
+ if (options.signal.aborted) {
71
+ killProcess();
72
+ } else {
73
+ options.signal.addEventListener("abort", killProcess, { once: true });
74
+ }
75
+ }
76
+
77
+ // Handle timeout
78
+ if (options?.timeout && options.timeout > 0) {
79
+ timeoutId = setTimeout(() => {
80
+ killProcess();
81
+ }, options.timeout);
82
+ }
83
+
84
+ // Read streams asynchronously
85
+ (async () => {
86
+ try {
87
+ const stdoutReader = (proc.stdout as ReadableStream<Uint8Array>).getReader();
88
+ const stderrReader = (proc.stderr as ReadableStream<Uint8Array>).getReader();
89
+
90
+ // Read both streams and wait for process exit
91
+ const [stdoutResult, stderrResult, exitCode] = await Promise.all([
92
+ (async () => {
93
+ const chunks: Uint8Array[] = [];
94
+ try {
95
+ while (true) {
96
+ const { done, value } = await stdoutReader.read();
97
+ if (done) break;
98
+ chunks.push(value);
99
+ }
100
+ } finally {
101
+ stdoutReader.releaseLock();
102
+ }
103
+ return Buffer.concat(chunks).toString();
104
+ })(),
105
+ (async () => {
106
+ const chunks: Uint8Array[] = [];
107
+ try {
108
+ while (true) {
109
+ const { done, value } = await stderrReader.read();
110
+ if (done) break;
111
+ chunks.push(value);
112
+ }
113
+ } finally {
114
+ stderrReader.releaseLock();
115
+ }
116
+ return Buffer.concat(chunks).toString();
117
+ })(),
118
+ proc.exited,
119
+ ]);
120
+
121
+ stdout = stdoutResult;
122
+ stderr = stderrResult;
123
+
124
+ if (timeoutId) clearTimeout(timeoutId);
125
+ if (options?.signal) {
126
+ options.signal.removeEventListener("abort", killProcess);
127
+ }
128
+ resolve({ stdout, stderr, code: exitCode ?? 0, killed });
129
+ } catch (err) {
130
+ logger.debug("Process stream error", { error: String(err) });
131
+ if (timeoutId) clearTimeout(timeoutId);
132
+ if (options?.signal) {
133
+ options.signal.removeEventListener("abort", killProcess);
134
+ }
135
+ resolve({ stdout, stderr, code: 1, killed });
136
+ }
137
+ })();
138
+ });
139
+ }