@oh-my-pi/pi-coding-agent 7.0.0 → 8.0.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 (501) hide show
  1. package/CHANGELOG.md +87 -0
  2. package/README.md +1 -1
  3. package/docs/hooks.md +2 -2
  4. package/docs/sdk.md +1 -1
  5. package/package.json +10 -10
  6. package/scripts/format-prompts.ts +143 -0
  7. package/scripts/generate-template.ts +1 -1
  8. package/src/cli/args.ts +3 -3
  9. package/src/cli/config-cli.ts +4 -4
  10. package/src/cli/file-processor.ts +3 -3
  11. package/src/cli/list-models.ts +2 -2
  12. package/src/cli/plugin-cli.ts +3 -3
  13. package/src/cli/session-picker.ts +2 -2
  14. package/src/cli/setup-cli.ts +2 -2
  15. package/src/cli/stats-cli.ts +1 -1
  16. package/src/cli/update-cli.ts +2 -2
  17. package/src/{core → config}/keybindings.ts +1 -1
  18. package/src/{core → config}/model-registry.ts +1 -1
  19. package/src/{core → config}/model-resolver.ts +3 -3
  20. package/src/{core → config}/prompt-templates.ts +2 -2
  21. package/src/{core → config}/settings-manager.ts +6 -6
  22. package/src/{core/cursor/exec-bridge.ts → cursor.ts} +4 -4
  23. package/src/discovery/agents-md.ts +4 -4
  24. package/src/discovery/builtin.ts +17 -17
  25. package/src/discovery/claude.ts +12 -12
  26. package/src/discovery/cline.ts +6 -6
  27. package/src/discovery/codex.ts +21 -21
  28. package/src/discovery/cursor.ts +9 -9
  29. package/src/discovery/gemini.ts +9 -9
  30. package/src/discovery/github.ts +6 -6
  31. package/src/discovery/helpers.ts +4 -4
  32. package/src/discovery/index.ts +16 -16
  33. package/src/discovery/mcp-json.ts +4 -4
  34. package/src/discovery/ssh.ts +4 -4
  35. package/src/discovery/vscode.ts +4 -4
  36. package/src/discovery/windsurf.ts +6 -6
  37. package/src/{core/tools/exa → exa}/company.ts +2 -3
  38. package/src/{core/tools/exa → exa}/index.ts +2 -2
  39. package/src/{core/tools/exa → exa}/linkedin.ts +2 -3
  40. package/src/{core/tools/exa → exa}/mcp-client.ts +2 -2
  41. package/src/{core/tools/exa → exa}/render.ts +3 -3
  42. package/src/{core/tools/exa → exa}/researcher.ts +1 -1
  43. package/src/{core/tools/exa → exa}/search.ts +2 -10
  44. package/src/{core/tools/exa → exa}/websets.ts +1 -1
  45. package/src/{core → exec}/bash-executor.ts +22 -6
  46. package/src/{core → export}/custom-share.ts +1 -1
  47. package/src/{core/export-html → export/html}/index.ts +3 -3
  48. package/src/{core → export}/ttsr.ts +2 -2
  49. package/src/{core → extensibility}/custom-commands/bundled/review/index.ts +4 -4
  50. package/src/{core → extensibility}/custom-commands/loader.ts +3 -3
  51. package/src/{core → extensibility}/custom-commands/types.ts +1 -1
  52. package/src/{core → extensibility}/custom-tools/loader.ts +9 -9
  53. package/src/{core → extensibility}/custom-tools/types.ts +6 -6
  54. package/src/{core → extensibility}/custom-tools/wrapper.ts +1 -1
  55. package/src/{core → extensibility}/extensions/loader.ts +8 -8
  56. package/src/{core → extensibility}/extensions/runner.ts +3 -3
  57. package/src/{core → extensibility}/extensions/types.ts +15 -15
  58. package/src/{core → extensibility}/extensions/wrapper.ts +1 -1
  59. package/src/{core → extensibility}/hooks/index.ts +1 -1
  60. package/src/{core → extensibility}/hooks/loader.ts +7 -7
  61. package/src/{core → extensibility}/hooks/runner.ts +4 -4
  62. package/src/{core → extensibility}/hooks/types.ts +9 -9
  63. package/src/{core → extensibility}/plugins/doctor.ts +1 -1
  64. package/src/{core → extensibility}/plugins/installer.ts +2 -3
  65. package/src/{core → extensibility}/plugins/paths.ts +1 -1
  66. package/src/{core → extensibility}/skills.ts +8 -48
  67. package/src/{core → extensibility}/slash-commands.ts +6 -6
  68. package/src/index.ts +127 -128
  69. package/src/internal-urls/agent-protocol.ts +126 -0
  70. package/src/internal-urls/artifact-protocol.ts +93 -0
  71. package/src/internal-urls/index.ts +28 -0
  72. package/src/internal-urls/json-query.ts +126 -0
  73. package/src/internal-urls/router.ts +69 -0
  74. package/src/internal-urls/rule-protocol.ts +56 -0
  75. package/src/internal-urls/skill-protocol.ts +112 -0
  76. package/src/internal-urls/types.ts +48 -0
  77. package/src/{core/python-executor.ts → ipy/executor.ts} +51 -11
  78. package/src/{core/python-gateway-coordinator.ts → ipy/gateway-coordinator.ts} +41 -325
  79. package/src/{core/python-kernel.ts → ipy/kernel.ts} +38 -10
  80. package/src/ipy/prelude.ts +3 -0
  81. package/src/{core/tools/lsp → lsp}/client.ts +7 -6
  82. package/src/{core/tools/lsp → lsp}/clients/biome-client.ts +1 -1
  83. package/src/{core/tools/lsp → lsp}/clients/index.ts +1 -1
  84. package/src/{core/tools/lsp → lsp}/clients/lsp-linter-client.ts +4 -4
  85. package/src/{core/tools/lsp → lsp}/config.ts +1 -1
  86. package/src/{core/tools/lsp → lsp}/index.ts +16 -15
  87. package/src/{core/tools/lsp → lsp}/render.ts +2 -2
  88. package/src/{core/tools/lsp → lsp}/types.ts +14 -16
  89. package/src/{core/tools/lsp → lsp}/utils.ts +1 -1
  90. package/src/main.ts +12 -12
  91. package/src/{core/mcp → mcp}/config.ts +8 -8
  92. package/src/{core/mcp → mcp}/loader.ts +5 -6
  93. package/src/{core/mcp → mcp}/manager.ts +2 -2
  94. package/src/{core/mcp → mcp}/tool-bridge.ts +35 -6
  95. package/src/{core/mcp → mcp}/tool-cache.ts +1 -1
  96. package/src/{core/mcp → mcp}/transports/http.ts +7 -1
  97. package/src/{core/mcp → mcp}/transports/stdio.ts +1 -1
  98. package/src/{core/mcp → mcp}/types.ts +1 -1
  99. package/src/migrations.ts +2 -2
  100. package/src/modes/{interactive/components → components}/armin.ts +1 -1
  101. package/src/modes/{interactive/components → components}/assistant-message.ts +1 -1
  102. package/src/modes/{interactive/components → components}/bash-execution.ts +37 -29
  103. package/src/modes/{interactive/components → components}/bordered-loader.ts +1 -1
  104. package/src/modes/{interactive/components → components}/branch-summary-message.ts +2 -2
  105. package/src/modes/{interactive/components → components}/compaction-summary-message.ts +2 -2
  106. package/src/modes/{interactive/components → components}/custom-message.ts +3 -3
  107. package/src/modes/{interactive/components → components}/diff.ts +1 -1
  108. package/src/modes/{interactive/components → components}/dynamic-border.ts +1 -1
  109. package/src/modes/{interactive/components → components}/extensions/extension-dashboard.ts +3 -3
  110. package/src/modes/{interactive/components → components}/extensions/extension-list.ts +2 -2
  111. package/src/modes/{interactive/components → components}/extensions/inspector-panel.ts +1 -1
  112. package/src/modes/{interactive/components → components}/extensions/state-manager.ts +11 -17
  113. package/src/modes/{interactive/components → components}/extensions/types.ts +1 -1
  114. package/src/modes/{interactive/components → components}/footer.ts +3 -3
  115. package/src/modes/{interactive/components → components}/history-search.ts +2 -2
  116. package/src/modes/{interactive/components → components}/hook-editor.ts +1 -1
  117. package/src/modes/{interactive/components → components}/hook-input.ts +1 -1
  118. package/src/modes/{interactive/components → components}/hook-message.ts +3 -3
  119. package/src/modes/{interactive/components → components}/hook-selector.ts +1 -1
  120. package/src/modes/{interactive/components → components}/keybinding-hints.ts +2 -2
  121. package/src/modes/{interactive/components → components}/login-dialog.ts +1 -1
  122. package/src/modes/{interactive/components → components}/model-selector.ts +5 -5
  123. package/src/modes/{interactive/components → components}/oauth-selector.ts +2 -2
  124. package/src/modes/{interactive/components → components}/plugin-settings.ts +3 -3
  125. package/src/modes/{interactive/components → components}/python-execution.ts +35 -24
  126. package/src/modes/{interactive/components → components}/queue-mode-selector.ts +1 -1
  127. package/src/modes/{interactive/components → components}/read-tool-group.ts +2 -2
  128. package/src/modes/{interactive/components → components}/session-selector.ts +3 -3
  129. package/src/modes/{interactive/components → components}/settings-defs.ts +2 -2
  130. package/src/modes/{interactive/components → components}/settings-selector.ts +2 -2
  131. package/src/modes/{interactive/components → components}/show-images-selector.ts +1 -1
  132. package/src/modes/{interactive/components → components}/status-line/segments.ts +2 -2
  133. package/src/modes/{interactive/components → components}/status-line/separators.ts +1 -1
  134. package/src/modes/{interactive/components → components}/status-line/types.ts +2 -2
  135. package/src/modes/{interactive/components → components}/status-line-segment-editor.ts +2 -2
  136. package/src/modes/{interactive/components → components}/status-line.ts +3 -3
  137. package/src/modes/{interactive/components → components}/theme-selector.ts +1 -1
  138. package/src/modes/{interactive/components → components}/thinking-selector.ts +1 -1
  139. package/src/modes/{interactive/components → components}/todo-display.ts +3 -4
  140. package/src/modes/{interactive/components → components}/todo-reminder.ts +2 -2
  141. package/src/modes/{interactive/components → components}/tool-execution.ts +8 -8
  142. package/src/modes/{interactive/components → components}/tree-selector.ts +3 -3
  143. package/src/modes/{interactive/components → components}/ttsr-notification.ts +2 -2
  144. package/src/modes/{interactive/components → components}/user-message-selector.ts +1 -1
  145. package/src/modes/{interactive/components → components}/user-message.ts +1 -1
  146. package/src/modes/{interactive/components → components}/welcome.ts +2 -2
  147. package/src/modes/{interactive/controllers → controllers}/command-controller.ts +32 -30
  148. package/src/modes/{interactive/controllers → controllers}/event-controller.ts +9 -9
  149. package/src/modes/{interactive/controllers → controllers}/extension-ui-controller.ts +8 -8
  150. package/src/modes/{interactive/controllers → controllers}/input-controller.ts +6 -6
  151. package/src/modes/{interactive/controllers → controllers}/selector-controller.ts +16 -16
  152. package/src/modes/index.ts +1 -1
  153. package/src/modes/{interactive/interactive-mode.ts → interactive-mode.ts} +14 -14
  154. package/src/modes/print-mode.ts +1 -1
  155. package/src/modes/rpc/rpc-client.ts +3 -3
  156. package/src/modes/rpc/rpc-mode.ts +3 -3
  157. package/src/modes/rpc/rpc-types.ts +3 -3
  158. package/src/modes/{interactive/theme → theme}/theme.ts +1 -1
  159. package/src/modes/{interactive/types.ts → types.ts} +8 -9
  160. package/src/modes/{interactive/utils → utils}/ui-helpers.ts +20 -27
  161. package/src/{core/tools/patch → patch}/applicator.ts +1 -1
  162. package/src/{core/tools/patch → patch}/diff.ts +1 -1
  163. package/src/{core/tools/patch → patch}/index.ts +31 -36
  164. package/src/{core/tools/patch → patch}/shared.ts +9 -6
  165. package/src/prompts/agents/explore.md +83 -46
  166. package/src/prompts/agents/init.md +9 -4
  167. package/src/prompts/agents/plan.md +8 -7
  168. package/src/prompts/agents/reviewer.md +36 -18
  169. package/src/prompts/agents/task.md +4 -4
  170. package/src/prompts/compaction/branch-summary-preamble.md +0 -1
  171. package/src/prompts/review-request.md +0 -1
  172. package/src/prompts/system/custom-system-prompt.md +2 -14
  173. package/src/prompts/system/file-operations.md +0 -2
  174. package/src/prompts/system/system-prompt.md +147 -138
  175. package/src/prompts/system/web-search.md +26 -0
  176. package/src/prompts/tools/ask.md +31 -24
  177. package/src/prompts/tools/bash.md +20 -17
  178. package/src/prompts/tools/calculator.md +9 -5
  179. package/src/prompts/tools/fetch.md +16 -0
  180. package/src/prompts/tools/find.md +15 -5
  181. package/src/prompts/tools/gemini-image.md +21 -6
  182. package/src/prompts/tools/grep.md +28 -12
  183. package/src/prompts/tools/lsp.md +35 -14
  184. package/src/prompts/tools/patch.md +39 -41
  185. package/src/prompts/tools/python.md +59 -76
  186. package/src/prompts/tools/read.md +23 -22
  187. package/src/prompts/tools/replace.md +19 -12
  188. package/src/prompts/tools/ssh.md +21 -28
  189. package/src/prompts/tools/task.md +54 -44
  190. package/src/prompts/tools/todo-write.md +52 -163
  191. package/src/prompts/tools/web-search.md +16 -9
  192. package/src/prompts/tools/write.md +13 -2
  193. package/src/{core/sdk.ts → sdk.ts} +65 -34
  194. package/src/{core → session}/agent-session.ts +45 -37
  195. package/src/{core → session}/agent-storage.ts +2 -2
  196. package/src/session/artifacts.ts +110 -0
  197. package/src/{core → session}/auth-storage.ts +1 -1
  198. package/src/{core → session}/compaction/branch-summarization.ts +5 -5
  199. package/src/{core → session}/compaction/compaction.ts +6 -6
  200. package/src/{core → session}/compaction/utils.ts +3 -3
  201. package/src/{core → session}/history-storage.ts +1 -1
  202. package/src/{core → session}/messages.ts +6 -8
  203. package/src/{core → session}/session-manager.ts +2 -2
  204. package/src/{core → session}/storage-migration.ts +2 -2
  205. package/src/session/streaming-output.ts +177 -0
  206. package/src/{core/ssh → ssh}/connection-manager.ts +1 -1
  207. package/src/{core/ssh → ssh}/ssh-executor.ts +19 -4
  208. package/src/{core/ssh → ssh}/sshfs-mount.ts +1 -1
  209. package/src/{core/system-prompt.ts → system-prompt.ts} +8 -37
  210. package/src/{core/tools/task → task}/agents.ts +8 -8
  211. package/src/{core/tools/task → task}/commands.ts +5 -6
  212. package/src/{core/tools/task → task}/discovery.ts +3 -3
  213. package/src/{core/tools/task → task}/executor.ts +34 -44
  214. package/src/{core/tools/task → task}/index.ts +206 -50
  215. package/src/{core/tools/task → task}/render.ts +80 -23
  216. package/src/{core/tools/task → task}/subprocess-tool-registry.ts +1 -1
  217. package/src/task/template.ts +47 -0
  218. package/src/{core/tools/task → task}/types.ts +19 -27
  219. package/src/{core/tools/task → task}/worker-protocol.ts +8 -4
  220. package/src/{core/tools/task → task}/worker.ts +34 -29
  221. package/src/task/worktree.ts +166 -0
  222. package/src/{core/tools → tools}/ask.ts +13 -21
  223. package/src/{core/tools → tools}/bash-interceptor.ts +1 -1
  224. package/src/{core/tools → tools}/bash.ts +61 -63
  225. package/src/{core/tools → tools}/calculator.ts +4 -4
  226. package/src/{core/tools → tools}/complete.ts +1 -1
  227. package/src/{core/tools → tools}/context.ts +2 -2
  228. package/src/{core/tools/web-fetch.ts → tools/fetch.ts} +97 -76
  229. package/src/{core/tools → tools}/find.ts +80 -104
  230. package/src/{core/tools → tools}/gemini-image.ts +420 -29
  231. package/src/{core/tools → tools}/grep.ts +155 -164
  232. package/src/{core/tools → tools}/index.ts +63 -56
  233. package/src/tools/list-limit.ts +40 -0
  234. package/src/{core/tools → tools}/ls.ts +44 -35
  235. package/src/{core/tools → tools}/notebook.ts +3 -3
  236. package/src/tools/output-meta.ts +443 -0
  237. package/src/tools/output-utils.ts +63 -0
  238. package/src/{core/tools → tools}/python.ts +105 -89
  239. package/src/tools/read.ts +882 -0
  240. package/src/{core/tools → tools}/render-utils.ts +1 -1
  241. package/src/{core/tools → tools}/renderers.ts +8 -10
  242. package/src/{core/tools → tools}/review.ts +2 -2
  243. package/src/{core/tools → tools}/ssh.ts +56 -59
  244. package/src/{core/tools → tools}/todo-write.ts +12 -23
  245. package/src/tools/tool-errors.ts +95 -0
  246. package/src/tools/tool-result.ts +92 -0
  247. package/src/{core/tools → tools}/truncate.ts +2 -2
  248. package/src/{core/tools → tools}/write.ts +15 -13
  249. package/src/utils/changelog.ts +1 -1
  250. package/src/{core → utils}/file-mentions.ts +4 -4
  251. package/src/utils/image-convert.ts +1 -1
  252. package/src/utils/image-resize.ts +1 -1
  253. package/src/utils/shell.ts +1 -1
  254. package/src/{core → utils}/title-generator.ts +4 -4
  255. package/src/utils/tools-manager.ts +1 -1
  256. package/src/{core/tools/web-scrapers → web/scrapers}/choosealicense.ts +1 -1
  257. package/src/{core/tools/web-scrapers → web/scrapers}/twitter.ts +3 -2
  258. package/src/{core/tools/web-scrapers → web/scrapers}/types.ts +4 -2
  259. package/src/{core/tools/web-scrapers → web/scrapers}/utils.ts +1 -1
  260. package/src/{core/tools/web-scrapers → web/scrapers}/youtube.ts +14 -13
  261. package/src/{core/tools/web-search → web/search}/auth.ts +4 -4
  262. package/src/{core/tools/web-search → web/search}/index.ts +22 -71
  263. package/src/{core/tools/web-search → web/search}/providers/anthropic.ts +7 -10
  264. package/src/{core/tools/web-search → web/search}/providers/exa.ts +2 -2
  265. package/src/{core/tools/web-search → web/search}/providers/perplexity.ts +4 -16
  266. package/src/{core/tools/web-search → web/search}/render.ts +3 -3
  267. package/scripts/migrate-sessions.sh +0 -93
  268. package/src/core/index.ts +0 -56
  269. package/src/core/python-prelude.ts +0 -3
  270. package/src/core/ssh-executor.ts +0 -5
  271. package/src/core/streaming-output.ts +0 -115
  272. package/src/core/tools/output.ts +0 -519
  273. package/src/core/tools/read.ts +0 -717
  274. package/src/core/tools/task/template.ts +0 -37
  275. package/src/prompts/tools/output.md +0 -47
  276. package/src/prompts/tools/web-fetch.md +0 -9
  277. /package/src/{core/tools/exa → exa}/types.ts +0 -0
  278. /package/src/{core → exec}/exec.ts +0 -0
  279. /package/src/{core/export-html → export/html}/template.css +0 -0
  280. /package/src/{core/export-html → export/html}/template.generated.ts +0 -0
  281. /package/src/{core/export-html → export/html}/template.html +0 -0
  282. /package/src/{core/export-html → export/html}/template.js +0 -0
  283. /package/src/{core/export-html → export/html}/template.macro.ts +0 -0
  284. /package/src/{core/export-html → export/html}/vendor/highlight.min.js +0 -0
  285. /package/src/{core/export-html → export/html}/vendor/marked.min.js +0 -0
  286. /package/src/{core → extensibility}/custom-commands/index.ts +0 -0
  287. /package/src/{core → extensibility}/custom-tools/index.ts +0 -0
  288. /package/src/{core → extensibility}/extensions/index.ts +0 -0
  289. /package/src/{core → extensibility}/hooks/tool-wrapper.ts +0 -0
  290. /package/src/{core → extensibility}/plugins/index.ts +0 -0
  291. /package/src/{core → extensibility}/plugins/loader.ts +0 -0
  292. /package/src/{core → extensibility}/plugins/manager.ts +0 -0
  293. /package/src/{core → extensibility}/plugins/parser.ts +0 -0
  294. /package/src/{core → extensibility}/plugins/types.ts +0 -0
  295. /package/src/{core/python-modules.ts → ipy/modules.ts} +0 -0
  296. /package/src/{core/python-prelude.py → ipy/prelude.py} +0 -0
  297. /package/src/{core/tools/lsp → lsp}/defaults.json +0 -0
  298. /package/src/{core/tools/lsp → lsp}/edits.ts +0 -0
  299. /package/src/{core/tools/lsp → lsp}/lspmux.ts +0 -0
  300. /package/src/{core/tools/lsp → lsp}/rust-analyzer.ts +0 -0
  301. /package/src/{core/mcp → mcp}/client.ts +0 -0
  302. /package/src/{core/mcp → mcp}/index.ts +0 -0
  303. /package/src/{core/mcp → mcp}/json-rpc.ts +0 -0
  304. /package/src/{core/mcp → mcp}/transports/index.ts +0 -0
  305. /package/src/modes/{interactive/components → components}/countdown-timer.ts +0 -0
  306. /package/src/modes/{interactive/components → components}/custom-editor.ts +0 -0
  307. /package/src/modes/{interactive/components → components}/extensions/index.ts +0 -0
  308. /package/src/modes/{interactive/components → components}/index.ts +0 -0
  309. /package/src/modes/{interactive/components → components}/status-line/index.ts +0 -0
  310. /package/src/modes/{interactive/components → components}/status-line/presets.ts +0 -0
  311. /package/src/modes/{interactive/components → components}/visual-truncate.ts +0 -0
  312. /package/src/modes/{interactive/theme → theme}/dark.json +0 -0
  313. /package/src/modes/{interactive/theme → theme}/defaults/alabaster.json +0 -0
  314. /package/src/modes/{interactive/theme → theme}/defaults/amethyst.json +0 -0
  315. /package/src/modes/{interactive/theme → theme}/defaults/anthracite.json +0 -0
  316. /package/src/modes/{interactive/theme → theme}/defaults/basalt.json +0 -0
  317. /package/src/modes/{interactive/theme → theme}/defaults/birch.json +0 -0
  318. /package/src/modes/{interactive/theme → theme}/defaults/dark-abyss.json +0 -0
  319. /package/src/modes/{interactive/theme → theme}/defaults/dark-arctic.json +0 -0
  320. /package/src/modes/{interactive/theme → theme}/defaults/dark-aurora.json +0 -0
  321. /package/src/modes/{interactive/theme → theme}/defaults/dark-catppuccin.json +0 -0
  322. /package/src/modes/{interactive/theme → theme}/defaults/dark-cavern.json +0 -0
  323. /package/src/modes/{interactive/theme → theme}/defaults/dark-copper.json +0 -0
  324. /package/src/modes/{interactive/theme → theme}/defaults/dark-cosmos.json +0 -0
  325. /package/src/modes/{interactive/theme → theme}/defaults/dark-cyberpunk.json +0 -0
  326. /package/src/modes/{interactive/theme → theme}/defaults/dark-dracula.json +0 -0
  327. /package/src/modes/{interactive/theme → theme}/defaults/dark-eclipse.json +0 -0
  328. /package/src/modes/{interactive/theme → theme}/defaults/dark-ember.json +0 -0
  329. /package/src/modes/{interactive/theme → theme}/defaults/dark-equinox.json +0 -0
  330. /package/src/modes/{interactive/theme → theme}/defaults/dark-forest.json +0 -0
  331. /package/src/modes/{interactive/theme → theme}/defaults/dark-github.json +0 -0
  332. /package/src/modes/{interactive/theme → theme}/defaults/dark-gruvbox.json +0 -0
  333. /package/src/modes/{interactive/theme → theme}/defaults/dark-lavender.json +0 -0
  334. /package/src/modes/{interactive/theme → theme}/defaults/dark-lunar.json +0 -0
  335. /package/src/modes/{interactive/theme → theme}/defaults/dark-midnight.json +0 -0
  336. /package/src/modes/{interactive/theme → theme}/defaults/dark-monochrome.json +0 -0
  337. /package/src/modes/{interactive/theme → theme}/defaults/dark-monokai.json +0 -0
  338. /package/src/modes/{interactive/theme → theme}/defaults/dark-nebula.json +0 -0
  339. /package/src/modes/{interactive/theme → theme}/defaults/dark-nord.json +0 -0
  340. /package/src/modes/{interactive/theme → theme}/defaults/dark-ocean.json +0 -0
  341. /package/src/modes/{interactive/theme → theme}/defaults/dark-one.json +0 -0
  342. /package/src/modes/{interactive/theme → theme}/defaults/dark-rainforest.json +0 -0
  343. /package/src/modes/{interactive/theme → theme}/defaults/dark-reef.json +0 -0
  344. /package/src/modes/{interactive/theme → theme}/defaults/dark-retro.json +0 -0
  345. /package/src/modes/{interactive/theme → theme}/defaults/dark-rose-pine.json +0 -0
  346. /package/src/modes/{interactive/theme → theme}/defaults/dark-sakura.json +0 -0
  347. /package/src/modes/{interactive/theme → theme}/defaults/dark-slate.json +0 -0
  348. /package/src/modes/{interactive/theme → theme}/defaults/dark-solarized.json +0 -0
  349. /package/src/modes/{interactive/theme → theme}/defaults/dark-solstice.json +0 -0
  350. /package/src/modes/{interactive/theme → theme}/defaults/dark-starfall.json +0 -0
  351. /package/src/modes/{interactive/theme → theme}/defaults/dark-sunset.json +0 -0
  352. /package/src/modes/{interactive/theme → theme}/defaults/dark-swamp.json +0 -0
  353. /package/src/modes/{interactive/theme → theme}/defaults/dark-synthwave.json +0 -0
  354. /package/src/modes/{interactive/theme → theme}/defaults/dark-taiga.json +0 -0
  355. /package/src/modes/{interactive/theme → theme}/defaults/dark-terminal.json +0 -0
  356. /package/src/modes/{interactive/theme → theme}/defaults/dark-tokyo-night.json +0 -0
  357. /package/src/modes/{interactive/theme → theme}/defaults/dark-tundra.json +0 -0
  358. /package/src/modes/{interactive/theme → theme}/defaults/dark-twilight.json +0 -0
  359. /package/src/modes/{interactive/theme → theme}/defaults/dark-volcanic.json +0 -0
  360. /package/src/modes/{interactive/theme → theme}/defaults/graphite.json +0 -0
  361. /package/src/modes/{interactive/theme → theme}/defaults/index.ts +0 -0
  362. /package/src/modes/{interactive/theme → theme}/defaults/light-arctic.json +0 -0
  363. /package/src/modes/{interactive/theme → theme}/defaults/light-aurora-day.json +0 -0
  364. /package/src/modes/{interactive/theme → theme}/defaults/light-canyon.json +0 -0
  365. /package/src/modes/{interactive/theme → theme}/defaults/light-catppuccin.json +0 -0
  366. /package/src/modes/{interactive/theme → theme}/defaults/light-cirrus.json +0 -0
  367. /package/src/modes/{interactive/theme → theme}/defaults/light-coral.json +0 -0
  368. /package/src/modes/{interactive/theme → theme}/defaults/light-cyberpunk.json +0 -0
  369. /package/src/modes/{interactive/theme → theme}/defaults/light-dawn.json +0 -0
  370. /package/src/modes/{interactive/theme → theme}/defaults/light-dunes.json +0 -0
  371. /package/src/modes/{interactive/theme → theme}/defaults/light-eucalyptus.json +0 -0
  372. /package/src/modes/{interactive/theme → theme}/defaults/light-forest.json +0 -0
  373. /package/src/modes/{interactive/theme → theme}/defaults/light-frost.json +0 -0
  374. /package/src/modes/{interactive/theme → theme}/defaults/light-github.json +0 -0
  375. /package/src/modes/{interactive/theme → theme}/defaults/light-glacier.json +0 -0
  376. /package/src/modes/{interactive/theme → theme}/defaults/light-gruvbox.json +0 -0
  377. /package/src/modes/{interactive/theme → theme}/defaults/light-haze.json +0 -0
  378. /package/src/modes/{interactive/theme → theme}/defaults/light-honeycomb.json +0 -0
  379. /package/src/modes/{interactive/theme → theme}/defaults/light-lagoon.json +0 -0
  380. /package/src/modes/{interactive/theme → theme}/defaults/light-lavender.json +0 -0
  381. /package/src/modes/{interactive/theme → theme}/defaults/light-meadow.json +0 -0
  382. /package/src/modes/{interactive/theme → theme}/defaults/light-mint.json +0 -0
  383. /package/src/modes/{interactive/theme → theme}/defaults/light-monochrome.json +0 -0
  384. /package/src/modes/{interactive/theme → theme}/defaults/light-ocean.json +0 -0
  385. /package/src/modes/{interactive/theme → theme}/defaults/light-one.json +0 -0
  386. /package/src/modes/{interactive/theme → theme}/defaults/light-opal.json +0 -0
  387. /package/src/modes/{interactive/theme → theme}/defaults/light-orchard.json +0 -0
  388. /package/src/modes/{interactive/theme → theme}/defaults/light-paper.json +0 -0
  389. /package/src/modes/{interactive/theme → theme}/defaults/light-prism.json +0 -0
  390. /package/src/modes/{interactive/theme → theme}/defaults/light-retro.json +0 -0
  391. /package/src/modes/{interactive/theme → theme}/defaults/light-sand.json +0 -0
  392. /package/src/modes/{interactive/theme → theme}/defaults/light-savanna.json +0 -0
  393. /package/src/modes/{interactive/theme → theme}/defaults/light-solarized.json +0 -0
  394. /package/src/modes/{interactive/theme → theme}/defaults/light-soleil.json +0 -0
  395. /package/src/modes/{interactive/theme → theme}/defaults/light-sunset.json +0 -0
  396. /package/src/modes/{interactive/theme → theme}/defaults/light-synthwave.json +0 -0
  397. /package/src/modes/{interactive/theme → theme}/defaults/light-tokyo-night.json +0 -0
  398. /package/src/modes/{interactive/theme → theme}/defaults/light-wetland.json +0 -0
  399. /package/src/modes/{interactive/theme → theme}/defaults/light-zenith.json +0 -0
  400. /package/src/modes/{interactive/theme → theme}/defaults/limestone.json +0 -0
  401. /package/src/modes/{interactive/theme → theme}/defaults/mahogany.json +0 -0
  402. /package/src/modes/{interactive/theme → theme}/defaults/marble.json +0 -0
  403. /package/src/modes/{interactive/theme → theme}/defaults/obsidian.json +0 -0
  404. /package/src/modes/{interactive/theme → theme}/defaults/onyx.json +0 -0
  405. /package/src/modes/{interactive/theme → theme}/defaults/pearl.json +0 -0
  406. /package/src/modes/{interactive/theme → theme}/defaults/porcelain.json +0 -0
  407. /package/src/modes/{interactive/theme → theme}/defaults/quartz.json +0 -0
  408. /package/src/modes/{interactive/theme → theme}/defaults/sandstone.json +0 -0
  409. /package/src/modes/{interactive/theme → theme}/defaults/titanium.json +0 -0
  410. /package/src/modes/{interactive/theme → theme}/light.json +0 -0
  411. /package/src/modes/{interactive/theme → theme}/theme-schema.json +0 -0
  412. /package/src/{core/tools/patch → patch}/fuzzy.ts +0 -0
  413. /package/src/{core/tools/patch → patch}/normalize.ts +0 -0
  414. /package/src/{core/tools/patch → patch}/normative.ts +0 -0
  415. /package/src/{core/tools/patch → patch}/parser.ts +0 -0
  416. /package/src/{core/tools/patch → patch}/types.ts +0 -0
  417. /package/src/{core → session}/compaction/index.ts +0 -0
  418. /package/src/{core → session}/session-storage.ts +0 -0
  419. /package/src/{core/tools/task → task}/name-generator.ts +0 -0
  420. /package/src/{core/tools/task → task}/omp-command.ts +0 -0
  421. /package/src/{core/tools/task → task}/parallel.ts +0 -0
  422. /package/src/{core/tools → tools}/jtd-to-json-schema.ts +0 -0
  423. /package/src/{core/tools → tools}/path-utils.ts +0 -0
  424. /package/src/{core → utils}/event-bus.ts +0 -0
  425. /package/src/{core → utils}/frontmatter.ts +0 -0
  426. /package/src/{core → utils}/terminal-notify.ts +0 -0
  427. /package/src/{core → utils}/timings.ts +0 -0
  428. /package/src/{core → utils}/utils.ts +0 -0
  429. /package/src/{core/tools/web-scrapers → web/scrapers}/artifacthub.ts +0 -0
  430. /package/src/{core/tools/web-scrapers → web/scrapers}/arxiv.ts +0 -0
  431. /package/src/{core/tools/web-scrapers → web/scrapers}/aur.ts +0 -0
  432. /package/src/{core/tools/web-scrapers → web/scrapers}/biorxiv.ts +0 -0
  433. /package/src/{core/tools/web-scrapers → web/scrapers}/bluesky.ts +0 -0
  434. /package/src/{core/tools/web-scrapers → web/scrapers}/brew.ts +0 -0
  435. /package/src/{core/tools/web-scrapers → web/scrapers}/cheatsh.ts +0 -0
  436. /package/src/{core/tools/web-scrapers → web/scrapers}/chocolatey.ts +0 -0
  437. /package/src/{core/tools/web-scrapers → web/scrapers}/cisa-kev.ts +0 -0
  438. /package/src/{core/tools/web-scrapers → web/scrapers}/clojars.ts +0 -0
  439. /package/src/{core/tools/web-scrapers → web/scrapers}/coingecko.ts +0 -0
  440. /package/src/{core/tools/web-scrapers → web/scrapers}/crates-io.ts +0 -0
  441. /package/src/{core/tools/web-scrapers → web/scrapers}/crossref.ts +0 -0
  442. /package/src/{core/tools/web-scrapers → web/scrapers}/devto.ts +0 -0
  443. /package/src/{core/tools/web-scrapers → web/scrapers}/discogs.ts +0 -0
  444. /package/src/{core/tools/web-scrapers → web/scrapers}/discourse.ts +0 -0
  445. /package/src/{core/tools/web-scrapers → web/scrapers}/dockerhub.ts +0 -0
  446. /package/src/{core/tools/web-scrapers → web/scrapers}/fdroid.ts +0 -0
  447. /package/src/{core/tools/web-scrapers → web/scrapers}/firefox-addons.ts +0 -0
  448. /package/src/{core/tools/web-scrapers → web/scrapers}/flathub.ts +0 -0
  449. /package/src/{core/tools/web-scrapers → web/scrapers}/github-gist.ts +0 -0
  450. /package/src/{core/tools/web-scrapers → web/scrapers}/github.ts +0 -0
  451. /package/src/{core/tools/web-scrapers → web/scrapers}/gitlab.ts +0 -0
  452. /package/src/{core/tools/web-scrapers → web/scrapers}/go-pkg.ts +0 -0
  453. /package/src/{core/tools/web-scrapers → web/scrapers}/hackage.ts +0 -0
  454. /package/src/{core/tools/web-scrapers → web/scrapers}/hackernews.ts +0 -0
  455. /package/src/{core/tools/web-scrapers → web/scrapers}/hex.ts +0 -0
  456. /package/src/{core/tools/web-scrapers → web/scrapers}/huggingface.ts +0 -0
  457. /package/src/{core/tools/web-scrapers → web/scrapers}/iacr.ts +0 -0
  458. /package/src/{core/tools/web-scrapers → web/scrapers}/index.ts +0 -0
  459. /package/src/{core/tools/web-scrapers → web/scrapers}/jetbrains-marketplace.ts +0 -0
  460. /package/src/{core/tools/web-scrapers → web/scrapers}/lemmy.ts +0 -0
  461. /package/src/{core/tools/web-scrapers → web/scrapers}/lobsters.ts +0 -0
  462. /package/src/{core/tools/web-scrapers → web/scrapers}/mastodon.ts +0 -0
  463. /package/src/{core/tools/web-scrapers → web/scrapers}/maven.ts +0 -0
  464. /package/src/{core/tools/web-scrapers → web/scrapers}/mdn.ts +0 -0
  465. /package/src/{core/tools/web-scrapers → web/scrapers}/metacpan.ts +0 -0
  466. /package/src/{core/tools/web-scrapers → web/scrapers}/musicbrainz.ts +0 -0
  467. /package/src/{core/tools/web-scrapers → web/scrapers}/npm.ts +0 -0
  468. /package/src/{core/tools/web-scrapers → web/scrapers}/nuget.ts +0 -0
  469. /package/src/{core/tools/web-scrapers → web/scrapers}/nvd.ts +0 -0
  470. /package/src/{core/tools/web-scrapers → web/scrapers}/ollama.ts +0 -0
  471. /package/src/{core/tools/web-scrapers → web/scrapers}/open-vsx.ts +0 -0
  472. /package/src/{core/tools/web-scrapers → web/scrapers}/opencorporates.ts +0 -0
  473. /package/src/{core/tools/web-scrapers → web/scrapers}/openlibrary.ts +0 -0
  474. /package/src/{core/tools/web-scrapers → web/scrapers}/orcid.ts +0 -0
  475. /package/src/{core/tools/web-scrapers → web/scrapers}/osv.ts +0 -0
  476. /package/src/{core/tools/web-scrapers → web/scrapers}/packagist.ts +0 -0
  477. /package/src/{core/tools/web-scrapers → web/scrapers}/pub-dev.ts +0 -0
  478. /package/src/{core/tools/web-scrapers → web/scrapers}/pubmed.ts +0 -0
  479. /package/src/{core/tools/web-scrapers → web/scrapers}/pypi.ts +0 -0
  480. /package/src/{core/tools/web-scrapers → web/scrapers}/rawg.ts +0 -0
  481. /package/src/{core/tools/web-scrapers → web/scrapers}/readthedocs.ts +0 -0
  482. /package/src/{core/tools/web-scrapers → web/scrapers}/reddit.ts +0 -0
  483. /package/src/{core/tools/web-scrapers → web/scrapers}/repology.ts +0 -0
  484. /package/src/{core/tools/web-scrapers → web/scrapers}/rfc.ts +0 -0
  485. /package/src/{core/tools/web-scrapers → web/scrapers}/rubygems.ts +0 -0
  486. /package/src/{core/tools/web-scrapers → web/scrapers}/searchcode.ts +0 -0
  487. /package/src/{core/tools/web-scrapers → web/scrapers}/sec-edgar.ts +0 -0
  488. /package/src/{core/tools/web-scrapers → web/scrapers}/semantic-scholar.ts +0 -0
  489. /package/src/{core/tools/web-scrapers → web/scrapers}/snapcraft.ts +0 -0
  490. /package/src/{core/tools/web-scrapers → web/scrapers}/sourcegraph.ts +0 -0
  491. /package/src/{core/tools/web-scrapers → web/scrapers}/spdx.ts +0 -0
  492. /package/src/{core/tools/web-scrapers → web/scrapers}/spotify.ts +0 -0
  493. /package/src/{core/tools/web-scrapers → web/scrapers}/stackoverflow.ts +0 -0
  494. /package/src/{core/tools/web-scrapers → web/scrapers}/terraform.ts +0 -0
  495. /package/src/{core/tools/web-scrapers → web/scrapers}/tldr.ts +0 -0
  496. /package/src/{core/tools/web-scrapers → web/scrapers}/vimeo.ts +0 -0
  497. /package/src/{core/tools/web-scrapers → web/scrapers}/vscode-marketplace.ts +0 -0
  498. /package/src/{core/tools/web-scrapers → web/scrapers}/w3c.ts +0 -0
  499. /package/src/{core/tools/web-scrapers → web/scrapers}/wikidata.ts +0 -0
  500. /package/src/{core/tools/web-scrapers → web/scrapers}/wikipedia.ts +0 -0
  501. /package/src/{core/tools/web-search → web/search}/types.ts +0 -0
@@ -1,3 +0,0 @@
1
- import pythonPrelude from "./python-prelude.py" with { type: "text" };
2
-
3
- export const PYTHON_PRELUDE = pythonPrelude;
@@ -1,5 +0,0 @@
1
- /**
2
- * Backwards-compatible re-export for SSH execution.
3
- */
4
-
5
- export { executeSSH, type SSHExecutorOptions, type SSHResult } from "./ssh/ssh-executor";
@@ -1,115 +0,0 @@
1
- import { tmpdir } from "node:os";
2
- import { join } from "node:path";
3
- import { sanitizeText } from "@oh-my-pi/pi-utils";
4
- import { nanoid } from "nanoid";
5
- import { DEFAULT_MAX_BYTES } from "./tools/truncate";
6
-
7
- export interface OutputResult {
8
- output: string;
9
- truncated: boolean;
10
- fullOutputPath?: string;
11
- }
12
-
13
- export interface OutputSinkOptions {
14
- allocateFilePath?: () => string;
15
- spillThreshold?: number;
16
- maxColumn?: number;
17
- onChunk?: (chunk: string) => void;
18
- }
19
-
20
- function defaultFilePathAllocator(): string {
21
- return join(tmpdir(), `omp-${nanoid()}.log`);
22
- }
23
-
24
- /**
25
- * Line-buffered output sink with file spill support.
26
- *
27
- * Uses a single string buffer with line position tracking.
28
- * When memory limit exceeded, spills ~half to file in one batch operation.
29
- */
30
- export class OutputSink {
31
- #buffer = "";
32
- #file?: {
33
- path: string;
34
- sink: Bun.FileSink;
35
- };
36
-
37
- readonly #allocateFilePath: () => string;
38
- readonly #spillThreshold: number;
39
- readonly #onChunk?: (chunk: string) => void;
40
-
41
- constructor(options?: OutputSinkOptions) {
42
- const {
43
- allocateFilePath = defaultFilePathAllocator,
44
- spillThreshold = DEFAULT_MAX_BYTES,
45
- onChunk,
46
- } = options ?? {};
47
-
48
- this.#allocateFilePath = allocateFilePath;
49
- this.#spillThreshold = spillThreshold;
50
- this.#onChunk = onChunk;
51
- }
52
-
53
- async #pushSanitized(data: string): Promise<void> {
54
- this.#onChunk?.(data);
55
-
56
- const bufferOverflow = data.length + this.#buffer.length > this.#spillThreshold;
57
- const overflow = this.#file || bufferOverflow;
58
-
59
- const sink = overflow ? await this.#fileSink() : null;
60
-
61
- this.#buffer += data;
62
- await sink?.write(data);
63
-
64
- if (bufferOverflow) {
65
- this.#buffer = this.#buffer.slice(-this.#spillThreshold);
66
- }
67
- }
68
-
69
- async #fileSink(): Promise<Bun.FileSink> {
70
- if (!this.#file) {
71
- const filePath = this.#allocateFilePath();
72
- this.#file = {
73
- path: filePath,
74
- sink: Bun.file(filePath).writer(),
75
- };
76
- await this.#file.sink.write(this.#buffer);
77
- }
78
- return this.#file.sink;
79
- }
80
-
81
- async push(chunk: string): Promise<void> {
82
- chunk = sanitizeText(chunk);
83
- await this.#pushSanitized(chunk);
84
- }
85
-
86
- createInput(): WritableStream<Uint8Array | string> {
87
- const dec = new TextDecoder("utf-8", { ignoreBOM: true });
88
- const finalize = async () => {
89
- await this.push(dec.decode());
90
- };
91
-
92
- return new WritableStream<Uint8Array | string>({
93
- write: async (chunk) => {
94
- if (typeof chunk === "string") {
95
- await this.push(chunk);
96
- } else {
97
- await this.push(dec.decode(chunk, { stream: true }));
98
- }
99
- },
100
- close: finalize,
101
- abort: finalize,
102
- });
103
- }
104
-
105
- async dump(notice?: string): Promise<OutputResult> {
106
- const noticeLine = notice ? `[${notice}]\n` : "";
107
-
108
- if (this.#file) {
109
- await this.#file.sink.end();
110
- return { output: `${noticeLine}...${this.#buffer}`, truncated: true, fullOutputPath: this.#file.path };
111
- } else {
112
- return { output: `${noticeLine}${this.#buffer}`, truncated: false };
113
- }
114
- }
115
- }
@@ -1,519 +0,0 @@
1
- /**
2
- * Output tool for reading agent/task outputs by ID.
3
- *
4
- * Resolves IDs like "reviewer_0" to artifact paths in the current session.
5
- */
6
-
7
- import * as fs from "node:fs";
8
- import * as path from "node:path";
9
- import type { AgentTool, AgentToolContext, AgentToolResult, AgentToolUpdateCallback } from "@oh-my-pi/pi-agent-core";
10
- import { StringEnum } from "@oh-my-pi/pi-ai";
11
- import type { Component } from "@oh-my-pi/pi-tui";
12
- import { Text } from "@oh-my-pi/pi-tui";
13
- import { Type } from "@sinclair/typebox";
14
- import type { Theme } from "../../modes/interactive/theme/theme";
15
- import outputDescription from "../../prompts/tools/output.md" with { type: "text" };
16
- import type { RenderResultOptions } from "../custom-tools/types";
17
- import { renderPromptTemplate } from "../prompt-templates";
18
- import type { ToolSession } from "./index";
19
- import {
20
- formatCount,
21
- formatEmptyMessage,
22
- formatExpandHint,
23
- formatMeta,
24
- formatMoreItems,
25
- TRUNCATE_LENGTHS,
26
- truncate,
27
- } from "./render-utils";
28
-
29
- const outputSchema = Type.Object({
30
- ids: Type.Array(Type.String(), {
31
- description: "Agent output IDs to read (e.g., ['reviewer_0', 'explore_1'])",
32
- }),
33
- format: Type.Optional(
34
- StringEnum(["raw", "json", "stripped"], {
35
- description: "Output format: raw (default), json (structured), stripped (no ANSI)",
36
- }),
37
- ),
38
- query: Type.Optional(
39
- Type.String({
40
- description: "jq-like query for JSON outputs (e.g., .result.items[0].name). Requires JSON output.",
41
- }),
42
- ),
43
- offset: Type.Optional(
44
- Type.Number({
45
- description: "Line number to start reading from (1-indexed)",
46
- minimum: 1,
47
- }),
48
- ),
49
- limit: Type.Optional(
50
- Type.Number({
51
- description: "Maximum number of lines to read",
52
- minimum: 1,
53
- }),
54
- ),
55
- });
56
-
57
- /** Metadata for a single output file */
58
- interface OutputProvenance {
59
- agent: string;
60
- index: number;
61
- }
62
-
63
- interface OutputRange {
64
- startLine: number;
65
- endLine: number;
66
- totalLines: number;
67
- }
68
-
69
- interface OutputEntry {
70
- id: string;
71
- path: string;
72
- lineCount: number;
73
- charCount: number;
74
- provenance?: OutputProvenance;
75
- previewLines?: string[];
76
- range?: OutputRange;
77
- query?: string;
78
- }
79
-
80
- export interface OutputToolDetails {
81
- outputs: OutputEntry[];
82
- notFound?: string[];
83
- availableIds?: string[];
84
- }
85
-
86
- /** Strip ANSI escape codes from text */
87
- function stripAnsi(text: string): string {
88
- return text.replace(/\x1b\[[0-9;]*m/g, "");
89
- }
90
-
91
- function parseQuery(query: string): Array<string | number> {
92
- let input = query.trim();
93
- if (!input) return [];
94
- if (input.startsWith(".")) input = input.slice(1);
95
- if (!input) return [];
96
-
97
- const tokens: Array<string | number> = [];
98
- let i = 0;
99
-
100
- const isIdentChar = (ch: string) => /[A-Za-z0-9_-]/.test(ch);
101
-
102
- while (i < input.length) {
103
- const ch = input[i];
104
- if (ch === ".") {
105
- i++;
106
- continue;
107
- }
108
- if (ch === "[") {
109
- const closeIndex = input.indexOf("]", i + 1);
110
- if (closeIndex === -1) {
111
- throw new Error(`Invalid query: missing ] in ${query}`);
112
- }
113
- const raw = input.slice(i + 1, closeIndex).trim();
114
- if (!raw) {
115
- throw new Error(`Invalid query: empty [] in ${query}`);
116
- }
117
- const quote = raw[0];
118
- if ((quote === '"' || quote === "'") && raw.endsWith(quote)) {
119
- let inner = raw.slice(1, -1);
120
- inner = inner.replace(/\\(["'\\])/g, "$1");
121
- tokens.push(inner);
122
- } else if (/^\d+$/.test(raw)) {
123
- tokens.push(Number(raw));
124
- } else {
125
- tokens.push(raw);
126
- }
127
- i = closeIndex + 1;
128
- continue;
129
- }
130
-
131
- const start = i;
132
- while (i < input.length && isIdentChar(input[i])) {
133
- i++;
134
- }
135
- if (start === i) {
136
- throw new Error(`Invalid query: unexpected token '${input[i]}' in ${query}`);
137
- }
138
- const ident = input.slice(start, i);
139
- tokens.push(ident);
140
- }
141
-
142
- return tokens;
143
- }
144
-
145
- function applyQuery(data: unknown, query: string): unknown {
146
- const tokens = parseQuery(query);
147
- let current: unknown = data;
148
- for (const token of tokens) {
149
- if (current === null || current === undefined) return undefined;
150
- if (typeof token === "number") {
151
- if (!Array.isArray(current)) return undefined;
152
- current = current[token];
153
- continue;
154
- }
155
- if (typeof current !== "object") return undefined;
156
- const record = current as Record<string, unknown>;
157
- current = record[token];
158
- }
159
- return current;
160
- }
161
-
162
- /** List available output IDs in artifacts directory */
163
- function listAvailableOutputs(artifactsDir: string): string[] {
164
- try {
165
- const files = fs.readdirSync(artifactsDir);
166
- return files.filter((f) => f.endsWith(".md")).map((f) => f.replace(".md", ""));
167
- } catch {
168
- return [];
169
- }
170
- }
171
-
172
- /** Format byte count for display */
173
- function formatBytes(bytes: number): string {
174
- if (bytes < 1024) return `${bytes}B`;
175
- if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)}K`;
176
- return `${(bytes / (1024 * 1024)).toFixed(1)}M`;
177
- }
178
-
179
- function parseOutputProvenance(id: string): OutputProvenance | undefined {
180
- const match = id.match(/^(.*)_(\d+)$/);
181
- if (!match) return undefined;
182
- const agent = match[1];
183
- const index = Number(match[2]);
184
- if (!agent || Number.isNaN(index)) return undefined;
185
- return { agent, index };
186
- }
187
-
188
- function extractPreviewLines(content: string, maxLines: number): string[] {
189
- const lines = content.split("\n");
190
- const preview: string[] = [];
191
- const structuralTokens = new Set(["{", "}", "[", "]"]);
192
-
193
- const isStructuralLine = (line: string): boolean => {
194
- const trimmed = line.trim();
195
- if (!trimmed) return true;
196
- const cleaned = trimmed.replace(/,+$/, "");
197
- return structuralTokens.has(cleaned);
198
- };
199
-
200
- const trimmedContent = content.trim();
201
- const firstMeaningful = lines.find((line) => line.trim());
202
- if (
203
- firstMeaningful &&
204
- isStructuralLine(firstMeaningful) &&
205
- (trimmedContent.startsWith("{") || trimmedContent.startsWith("[")) &&
206
- trimmedContent.length <= 200_000
207
- ) {
208
- try {
209
- const parsed = JSON.parse(trimmedContent);
210
- const minified = JSON.stringify(parsed);
211
- if (minified) return [minified];
212
- } catch {
213
- // Fall back to line-based previews.
214
- }
215
- }
216
-
217
- for (const line of lines) {
218
- if (isStructuralLine(line)) continue;
219
- preview.push(line);
220
- if (preview.length >= maxLines) break;
221
- }
222
-
223
- if (preview.length === 0) {
224
- for (const line of lines) {
225
- if (!line.trim()) continue;
226
- preview.push(line);
227
- if (preview.length >= maxLines) break;
228
- }
229
- }
230
-
231
- return preview;
232
- }
233
-
234
- type OutputParams = {
235
- ids: string[];
236
- format?: "raw" | "json" | "stripped";
237
- query?: string;
238
- offset?: number;
239
- limit?: number;
240
- };
241
-
242
- /**
243
- * Output tool for reading agent/task outputs by ID.
244
- *
245
- * Resolves IDs like "reviewer_0" to artifact paths in the current session.
246
- */
247
- export class OutputTool implements AgentTool<typeof outputSchema, OutputToolDetails> {
248
- public readonly name = "output";
249
- public readonly label = "Output";
250
- public readonly description: string;
251
- public readonly parameters = outputSchema;
252
-
253
- private readonly session: ToolSession;
254
-
255
- constructor(session: ToolSession) {
256
- this.session = session;
257
- this.description = renderPromptTemplate(outputDescription);
258
- }
259
-
260
- public async execute(
261
- _toolCallId: string,
262
- params: OutputParams,
263
- _signal?: AbortSignal,
264
- _onUpdate?: AgentToolUpdateCallback<OutputToolDetails>,
265
- _context?: AgentToolContext,
266
- ): Promise<AgentToolResult<OutputToolDetails>> {
267
- const sessionFile = this.session.getSessionFile();
268
-
269
- if (!sessionFile) {
270
- return {
271
- content: [{ type: "text", text: "No session - output artifacts unavailable" }],
272
- details: { outputs: [], notFound: params.ids },
273
- };
274
- }
275
-
276
- const artifactsDir = sessionFile.slice(0, -6); // strip .jsonl extension
277
- if (!fs.existsSync(artifactsDir)) {
278
- return {
279
- content: [{ type: "text", text: "No artifacts directory found" }],
280
- details: { outputs: [], notFound: params.ids },
281
- };
282
- }
283
-
284
- const outputs: OutputEntry[] = [];
285
- const notFound: string[] = [];
286
- const outputContentById = new Map<string, string>();
287
- const query = params.query?.trim();
288
- const wantsQuery = query !== undefined && query.length > 0;
289
- const format = params.format ?? (wantsQuery ? "json" : "raw");
290
-
291
- if (wantsQuery && (params.offset !== undefined || params.limit !== undefined)) {
292
- throw new Error("query cannot be combined with offset/limit");
293
- }
294
-
295
- const queryResults: Array<{ id: string; value: unknown }> = [];
296
-
297
- for (const id of params.ids) {
298
- const outputPath = path.join(artifactsDir, `${id}.md`);
299
- const file = Bun.file(outputPath);
300
- if (!(await file.exists())) {
301
- notFound.push(id);
302
- continue;
303
- }
304
-
305
- const rawContent = await file.text();
306
- const rawLines = rawContent.split("\n");
307
- const totalLines = rawLines.length;
308
- const totalChars = rawContent.length;
309
-
310
- let selectedContent = rawContent;
311
- let range: OutputRange | undefined;
312
-
313
- if (wantsQuery && query) {
314
- let jsonValue: unknown;
315
- try {
316
- jsonValue = JSON.parse(rawContent);
317
- } catch (err) {
318
- const message = err instanceof Error ? err.message : String(err);
319
- throw new Error(`Output ${id} is not valid JSON: ${message}`);
320
- }
321
- const value = applyQuery(jsonValue, query);
322
- queryResults.push({ id, value });
323
- try {
324
- selectedContent = JSON.stringify(value, null, 2) ?? "null";
325
- } catch {
326
- selectedContent = String(value);
327
- }
328
- } else if (params.offset !== undefined || params.limit !== undefined) {
329
- const startLine = Math.max(1, params.offset ?? 1);
330
- if (startLine > totalLines) {
331
- throw new Error(
332
- `Offset ${params.offset ?? startLine} is beyond end of output (${totalLines} lines) for ${id}`,
333
- );
334
- }
335
- const effectiveLimit = params.limit ?? totalLines - startLine + 1;
336
- const endLine = Math.min(totalLines, startLine + effectiveLimit - 1);
337
- const selectedLines = rawLines.slice(startLine - 1, endLine);
338
- selectedContent = selectedLines.join("\n");
339
- range = { startLine, endLine, totalLines };
340
- }
341
-
342
- outputContentById.set(id, selectedContent);
343
- outputs.push({
344
- id,
345
- path: outputPath,
346
- lineCount: wantsQuery ? selectedContent.split("\n").length : totalLines,
347
- charCount: wantsQuery ? selectedContent.length : totalChars,
348
- provenance: parseOutputProvenance(id),
349
- previewLines: extractPreviewLines(selectedContent, 4),
350
- range,
351
- query: query,
352
- });
353
- }
354
-
355
- // Error case: some IDs not found
356
- if (notFound.length > 0) {
357
- const available = listAvailableOutputs(artifactsDir);
358
- const errorMsg =
359
- available.length > 0
360
- ? `Not found: ${notFound.join(", ")}\nAvailable: ${available.join(", ")}`
361
- : `Not found: ${notFound.join(", ")}\nNo outputs available in current session`;
362
-
363
- return {
364
- content: [{ type: "text", text: errorMsg }],
365
- details: { outputs, notFound, availableIds: available },
366
- };
367
- }
368
-
369
- // Success: build response based on format
370
- let contentText: string;
371
-
372
- if (format === "json") {
373
- const jsonData = wantsQuery
374
- ? queryResults
375
- : outputs.map((o) => ({
376
- id: o.id,
377
- lineCount: o.lineCount,
378
- charCount: o.charCount,
379
- provenance: o.provenance,
380
- previewLines: o.previewLines,
381
- range: o.range,
382
- content: outputContentById.get(o.id) ?? "",
383
- }));
384
- contentText = JSON.stringify(jsonData, null, 2);
385
- } else {
386
- // raw or stripped
387
- const parts = outputs.map((o) => {
388
- let content = outputContentById.get(o.id) ?? "";
389
- if (format === "stripped") {
390
- content = stripAnsi(content);
391
- }
392
- if (o.range && o.range.endLine < o.range.totalLines) {
393
- const nextOffset = o.range.endLine + 1;
394
- content += `\n\n[Showing lines ${o.range.startLine}-${o.range.endLine} of ${o.range.totalLines}. Use offset=${nextOffset} to continue]`;
395
- }
396
- // Add header for multiple outputs
397
- if (outputs.length > 1) {
398
- return `=== ${o.id} (${o.lineCount} lines, ${formatBytes(o.charCount)}) ===\n${content}`;
399
- }
400
- return content;
401
- });
402
- contentText = parts.join("\n\n");
403
- }
404
-
405
- return {
406
- content: [{ type: "text", text: contentText }],
407
- details: { outputs },
408
- };
409
- }
410
- }
411
-
412
- // =============================================================================
413
- // TUI Renderer
414
- // =============================================================================
415
-
416
- interface OutputRenderArgs {
417
- ids: string[];
418
- format?: "raw" | "json" | "stripped";
419
- query?: string;
420
- offset?: number;
421
- limit?: number;
422
- }
423
-
424
- type OutputEntryItem = OutputToolDetails["outputs"][number];
425
-
426
- function formatOutputMeta(entry: OutputEntryItem, uiTheme: Theme): string {
427
- const metaParts: string[] = [];
428
- if (entry.range) {
429
- metaParts.push(`lines ${entry.range.startLine}-${entry.range.endLine} of ${entry.range.totalLines}`);
430
- } else {
431
- metaParts.push(formatCount("line", entry.lineCount));
432
- }
433
- metaParts.push(formatBytes(entry.charCount));
434
- if (entry.provenance) {
435
- metaParts.push(`agent ${entry.provenance.agent}(${entry.provenance.index})`);
436
- }
437
- return uiTheme.fg("dim", metaParts.join(uiTheme.sep.dot));
438
- }
439
-
440
- export const outputToolRenderer = {
441
- renderCall(args: OutputRenderArgs, uiTheme: Theme): Component {
442
- const ids = args.ids?.join(", ") ?? "?";
443
- const label = uiTheme.fg("toolTitle", uiTheme.bold("Output"));
444
- let text = `${label} ${uiTheme.fg("accent", ids)}`;
445
-
446
- const meta: string[] = [];
447
- if (args.format && args.format !== "raw") meta.push(`format:${args.format}`);
448
- if (args.query) meta.push(`query:${args.query}`);
449
- if (args.offset !== undefined) meta.push(`offset:${args.offset}`);
450
- if (args.limit !== undefined) meta.push(`limit:${args.limit}`);
451
- text += formatMeta(meta, uiTheme);
452
-
453
- return new Text(text, 0, 0);
454
- },
455
-
456
- renderResult(
457
- result: { content: Array<{ type: string; text?: string }>; details?: OutputToolDetails },
458
- { expanded }: RenderResultOptions,
459
- uiTheme: Theme,
460
- ): Component {
461
- const details = result.details;
462
-
463
- if (details?.notFound?.length) {
464
- const icon = uiTheme.styledSymbol("status.error", "error");
465
- let text = `${icon} ${uiTheme.fg("error", `Error: Not found: ${details.notFound.join(", ")}`)}`;
466
- if (details.availableIds?.length) {
467
- text += `\n ${uiTheme.fg("dim", uiTheme.tree.last)} ${uiTheme.fg("muted", `Available: ${details.availableIds.join(", ")}`)}`;
468
- } else {
469
- text += `\n ${uiTheme.fg("dim", uiTheme.tree.last)} ${uiTheme.fg("muted", "No outputs available in current session")}`;
470
- }
471
- return new Text(text, 0, 0);
472
- }
473
-
474
- const outputs = details?.outputs ?? [];
475
-
476
- if (outputs.length === 0) {
477
- const textContent = result.content?.find((c) => c.type === "text")?.text;
478
- return new Text(formatEmptyMessage(textContent || "No outputs", uiTheme), 0, 0);
479
- }
480
-
481
- const icon = uiTheme.styledSymbol("status.success", "success");
482
- const summary = `read ${formatCount("output", outputs.length)}`;
483
- const previewLimit = expanded ? 3 : 1;
484
- const maxOutputs = expanded ? outputs.length : Math.min(outputs.length, 5);
485
- const hasMoreOutputs = outputs.length > maxOutputs;
486
- const hasMorePreview = outputs.some((o) => (o.previewLines?.length ?? 0) > previewLimit);
487
- const expandHint = formatExpandHint(uiTheme, expanded, hasMoreOutputs || hasMorePreview);
488
- let text = `${icon} ${uiTheme.fg("dim", summary)}${expandHint}`;
489
-
490
- for (let i = 0; i < maxOutputs; i++) {
491
- const o = outputs[i];
492
- const isLast = i === maxOutputs - 1 && !hasMoreOutputs;
493
- const branch = isLast ? uiTheme.tree.last : uiTheme.tree.branch;
494
- text += `\n ${uiTheme.fg("dim", branch)} ${uiTheme.fg("accent", o.id)} ${formatOutputMeta(o, uiTheme)}`;
495
-
496
- const previewLines = o.previewLines ?? [];
497
- const shownPreview = previewLines.slice(0, previewLimit);
498
- if (shownPreview.length > 0) {
499
- const childPrefix = isLast ? " " : ` ${uiTheme.fg("dim", uiTheme.tree.vertical)} `;
500
- for (const line of shownPreview) {
501
- const previewText = truncate(line, TRUNCATE_LENGTHS.CONTENT, uiTheme.format.ellipsis);
502
- text += `\n${childPrefix}${uiTheme.fg("dim", uiTheme.tree.hook)} ${uiTheme.fg(
503
- "muted",
504
- "preview:",
505
- )} ${uiTheme.fg("toolOutput", previewText)}`;
506
- }
507
- }
508
- }
509
-
510
- if (hasMoreOutputs) {
511
- text += `\n ${uiTheme.fg("dim", uiTheme.tree.last)} ${uiTheme.fg(
512
- "muted",
513
- formatMoreItems(outputs.length - maxOutputs, "output", uiTheme),
514
- )}`;
515
- }
516
-
517
- return new Text(text, 0, 0);
518
- },
519
- };