@nghyane/arcane 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 (738) hide show
  1. package/CHANGELOG.md +3 -0
  2. package/README.md +12 -0
  3. package/examples/README.md +21 -0
  4. package/examples/custom-tools/README.md +109 -0
  5. package/examples/custom-tools/hello/index.ts +20 -0
  6. package/examples/custom-tools/todo/index.ts +206 -0
  7. package/examples/extensions/README.md +143 -0
  8. package/examples/extensions/api-demo.ts +89 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +32 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +550 -0
  13. package/examples/extensions/reload-runtime.ts +37 -0
  14. package/examples/extensions/todo.ts +296 -0
  15. package/examples/extensions/tools.ts +144 -0
  16. package/examples/extensions/with-deps/index.ts +35 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +16 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +116 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +150 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +119 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +43 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +80 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +36 -0
  39. package/examples/sdk/08-slash-commands.ts +41 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/README.md +150 -0
  43. package/package.json +464 -0
  44. package/scripts/format-prompts.ts +184 -0
  45. package/scripts/generate-docs-index.ts +40 -0
  46. package/scripts/generate-template.ts +32 -0
  47. package/src/bun-imports.d.ts +22 -0
  48. package/src/capability/context-file.ts +39 -0
  49. package/src/capability/extension-module.ts +33 -0
  50. package/src/capability/extension.ts +47 -0
  51. package/src/capability/fs.ts +89 -0
  52. package/src/capability/hook.ts +39 -0
  53. package/src/capability/index.ts +432 -0
  54. package/src/capability/instruction.ts +36 -0
  55. package/src/capability/mcp.ts +60 -0
  56. package/src/capability/prompt.ts +34 -0
  57. package/src/capability/rule.ts +223 -0
  58. package/src/capability/settings.ts +34 -0
  59. package/src/capability/skill.ts +48 -0
  60. package/src/capability/slash-command.ts +39 -0
  61. package/src/capability/ssh.ts +41 -0
  62. package/src/capability/system-prompt.ts +34 -0
  63. package/src/capability/tool.ts +37 -0
  64. package/src/capability/types.ts +156 -0
  65. package/src/cli/args.ts +259 -0
  66. package/src/cli/config-cli.ts +357 -0
  67. package/src/cli/file-processor.ts +124 -0
  68. package/src/cli/grep-cli.ts +152 -0
  69. package/src/cli/jupyter-cli.ts +106 -0
  70. package/src/cli/list-models.ts +103 -0
  71. package/src/cli/plugin-cli.ts +661 -0
  72. package/src/cli/session-picker.ts +42 -0
  73. package/src/cli/setup-cli.ts +376 -0
  74. package/src/cli/shell-cli.ts +174 -0
  75. package/src/cli/ssh-cli.ts +179 -0
  76. package/src/cli/stats-cli.ts +197 -0
  77. package/src/cli/update-cli.ts +286 -0
  78. package/src/cli/web-search-cli.ts +143 -0
  79. package/src/cli.ts +65 -0
  80. package/src/commands/commit.ts +36 -0
  81. package/src/commands/config.ts +51 -0
  82. package/src/commands/grep.ts +41 -0
  83. package/src/commands/jupyter.ts +32 -0
  84. package/src/commands/launch.ts +139 -0
  85. package/src/commands/plugin.ts +70 -0
  86. package/src/commands/setup.ts +42 -0
  87. package/src/commands/shell.ts +29 -0
  88. package/src/commands/ssh.ts +60 -0
  89. package/src/commands/stats.ts +29 -0
  90. package/src/commands/update.ts +21 -0
  91. package/src/commands/web-search.ts +42 -0
  92. package/src/commit/agentic/agent.ts +311 -0
  93. package/src/commit/agentic/fallback.ts +96 -0
  94. package/src/commit/agentic/index.ts +359 -0
  95. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  96. package/src/commit/agentic/prompts/session-user.md +25 -0
  97. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  98. package/src/commit/agentic/prompts/system.md +38 -0
  99. package/src/commit/agentic/state.ts +69 -0
  100. package/src/commit/agentic/tools/analyze-file.ts +118 -0
  101. package/src/commit/agentic/tools/git-file-diff.ts +194 -0
  102. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  103. package/src/commit/agentic/tools/git-overview.ts +84 -0
  104. package/src/commit/agentic/tools/index.ts +56 -0
  105. package/src/commit/agentic/tools/propose-changelog.ts +128 -0
  106. package/src/commit/agentic/tools/propose-commit.ts +154 -0
  107. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  108. package/src/commit/agentic/tools/split-commit.ts +280 -0
  109. package/src/commit/agentic/topo-sort.ts +44 -0
  110. package/src/commit/agentic/trivial.ts +51 -0
  111. package/src/commit/agentic/validation.ts +200 -0
  112. package/src/commit/analysis/conventional.ts +165 -0
  113. package/src/commit/analysis/index.ts +4 -0
  114. package/src/commit/analysis/scope.ts +242 -0
  115. package/src/commit/analysis/summary.ts +112 -0
  116. package/src/commit/analysis/validation.ts +66 -0
  117. package/src/commit/changelog/detect.ts +37 -0
  118. package/src/commit/changelog/generate.ts +110 -0
  119. package/src/commit/changelog/index.ts +234 -0
  120. package/src/commit/changelog/parse.ts +44 -0
  121. package/src/commit/cli.ts +93 -0
  122. package/src/commit/git/diff.ts +148 -0
  123. package/src/commit/git/errors.ts +9 -0
  124. package/src/commit/git/index.ts +211 -0
  125. package/src/commit/git/operations.ts +54 -0
  126. package/src/commit/index.ts +5 -0
  127. package/src/commit/map-reduce/index.ts +64 -0
  128. package/src/commit/map-reduce/map-phase.ts +178 -0
  129. package/src/commit/map-reduce/reduce-phase.ts +145 -0
  130. package/src/commit/map-reduce/utils.ts +9 -0
  131. package/src/commit/message.ts +11 -0
  132. package/src/commit/model-selection.ts +69 -0
  133. package/src/commit/pipeline.ts +243 -0
  134. package/src/commit/prompts/analysis-system.md +148 -0
  135. package/src/commit/prompts/analysis-user.md +38 -0
  136. package/src/commit/prompts/changelog-system.md +50 -0
  137. package/src/commit/prompts/changelog-user.md +18 -0
  138. package/src/commit/prompts/file-observer-system.md +24 -0
  139. package/src/commit/prompts/file-observer-user.md +8 -0
  140. package/src/commit/prompts/reduce-system.md +50 -0
  141. package/src/commit/prompts/reduce-user.md +17 -0
  142. package/src/commit/prompts/summary-retry.md +3 -0
  143. package/src/commit/prompts/summary-system.md +38 -0
  144. package/src/commit/prompts/summary-user.md +13 -0
  145. package/src/commit/prompts/types-description.md +2 -0
  146. package/src/commit/types.ts +109 -0
  147. package/src/commit/utils/exclusions.ts +42 -0
  148. package/src/config/file-lock.ts +121 -0
  149. package/src/config/keybindings.ts +280 -0
  150. package/src/config/model-registry.ts +1140 -0
  151. package/src/config/model-resolver.ts +812 -0
  152. package/src/config/prompt-templates.ts +526 -0
  153. package/src/config/resolve-config-value.ts +92 -0
  154. package/src/config/settings-schema.ts +1236 -0
  155. package/src/config/settings.ts +706 -0
  156. package/src/config.ts +414 -0
  157. package/src/cursor.ts +239 -0
  158. package/src/debug/index.ts +431 -0
  159. package/src/debug/log-formatting.ts +60 -0
  160. package/src/debug/log-viewer.ts +903 -0
  161. package/src/debug/profiler.ts +158 -0
  162. package/src/debug/report-bundle.ts +366 -0
  163. package/src/debug/system-info.ts +112 -0
  164. package/src/discovery/agents-md.ts +68 -0
  165. package/src/discovery/agents.ts +199 -0
  166. package/src/discovery/builtin.ts +815 -0
  167. package/src/discovery/claude-plugins.ts +205 -0
  168. package/src/discovery/claude.ts +506 -0
  169. package/src/discovery/cline.ts +83 -0
  170. package/src/discovery/codex.ts +532 -0
  171. package/src/discovery/cursor.ts +218 -0
  172. package/src/discovery/gemini.ts +395 -0
  173. package/src/discovery/github.ts +117 -0
  174. package/src/discovery/helpers.ts +698 -0
  175. package/src/discovery/index.ts +89 -0
  176. package/src/discovery/mcp-json.ts +156 -0
  177. package/src/discovery/opencode.ts +394 -0
  178. package/src/discovery/ssh.ts +160 -0
  179. package/src/discovery/vscode.ts +103 -0
  180. package/src/discovery/windsurf.ts +145 -0
  181. package/src/exa/company.ts +57 -0
  182. package/src/exa/index.ts +62 -0
  183. package/src/exa/linkedin.ts +57 -0
  184. package/src/exa/mcp-client.ts +289 -0
  185. package/src/exa/render.ts +244 -0
  186. package/src/exa/researcher.ts +89 -0
  187. package/src/exa/search.ts +330 -0
  188. package/src/exa/types.ts +166 -0
  189. package/src/exa/websets.ts +247 -0
  190. package/src/exec/bash-executor.ts +184 -0
  191. package/src/exec/exec.ts +53 -0
  192. package/src/export/custom-share.ts +65 -0
  193. package/src/export/html/index.ts +162 -0
  194. package/src/export/html/template.css +889 -0
  195. package/src/export/html/template.generated.ts +2 -0
  196. package/src/export/html/template.html +45 -0
  197. package/src/export/html/template.js +1329 -0
  198. package/src/export/html/template.macro.ts +24 -0
  199. package/src/export/html/vendor/highlight.min.js +1213 -0
  200. package/src/export/html/vendor/marked.min.js +6 -0
  201. package/src/export/ttsr.ts +434 -0
  202. package/src/extensibility/custom-commands/bundled/review/index.ts +433 -0
  203. package/src/extensibility/custom-commands/index.ts +15 -0
  204. package/src/extensibility/custom-commands/loader.ts +231 -0
  205. package/src/extensibility/custom-commands/types.ts +111 -0
  206. package/src/extensibility/custom-tools/index.ts +22 -0
  207. package/src/extensibility/custom-tools/loader.ts +235 -0
  208. package/src/extensibility/custom-tools/types.ts +226 -0
  209. package/src/extensibility/custom-tools/wrapper.ts +45 -0
  210. package/src/extensibility/extensions/index.ts +136 -0
  211. package/src/extensibility/extensions/loader.ts +520 -0
  212. package/src/extensibility/extensions/runner.ts +774 -0
  213. package/src/extensibility/extensions/types.ts +1293 -0
  214. package/src/extensibility/extensions/wrapper.ts +188 -0
  215. package/src/extensibility/hooks/index.ts +16 -0
  216. package/src/extensibility/hooks/loader.ts +273 -0
  217. package/src/extensibility/hooks/runner.ts +441 -0
  218. package/src/extensibility/hooks/tool-wrapper.ts +106 -0
  219. package/src/extensibility/hooks/types.ts +817 -0
  220. package/src/extensibility/plugins/doctor.ts +65 -0
  221. package/src/extensibility/plugins/git-url.ts +281 -0
  222. package/src/extensibility/plugins/index.ts +33 -0
  223. package/src/extensibility/plugins/installer.ts +192 -0
  224. package/src/extensibility/plugins/loader.ts +338 -0
  225. package/src/extensibility/plugins/manager.ts +716 -0
  226. package/src/extensibility/plugins/parser.ts +105 -0
  227. package/src/extensibility/plugins/types.ts +190 -0
  228. package/src/extensibility/skills.ts +385 -0
  229. package/src/extensibility/slash-commands.ts +287 -0
  230. package/src/extensibility/tool-proxy.ts +25 -0
  231. package/src/index.ts +275 -0
  232. package/src/internal-urls/agent-protocol.ts +136 -0
  233. package/src/internal-urls/artifact-protocol.ts +97 -0
  234. package/src/internal-urls/docs-index.generated.ts +54 -0
  235. package/src/internal-urls/docs-protocol.ts +84 -0
  236. package/src/internal-urls/index.ts +31 -0
  237. package/src/internal-urls/json-query.ts +126 -0
  238. package/src/internal-urls/memory-protocol.ts +133 -0
  239. package/src/internal-urls/router.ts +70 -0
  240. package/src/internal-urls/rule-protocol.ts +55 -0
  241. package/src/internal-urls/skill-protocol.ts +111 -0
  242. package/src/internal-urls/types.ts +52 -0
  243. package/src/ipy/executor.ts +556 -0
  244. package/src/ipy/gateway-coordinator.ts +426 -0
  245. package/src/ipy/kernel.ts +892 -0
  246. package/src/ipy/modules.ts +109 -0
  247. package/src/ipy/prelude.py +831 -0
  248. package/src/ipy/prelude.ts +3 -0
  249. package/src/ipy/runtime.ts +222 -0
  250. package/src/lsp/client.ts +867 -0
  251. package/src/lsp/clients/biome-client.ts +202 -0
  252. package/src/lsp/clients/index.ts +50 -0
  253. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  254. package/src/lsp/clients/swiftlint-client.ts +120 -0
  255. package/src/lsp/config.ts +397 -0
  256. package/src/lsp/defaults.json +464 -0
  257. package/src/lsp/edits.ts +109 -0
  258. package/src/lsp/index.ts +1268 -0
  259. package/src/lsp/lspmux.ts +250 -0
  260. package/src/lsp/render.ts +689 -0
  261. package/src/lsp/types.ts +414 -0
  262. package/src/lsp/utils.ts +549 -0
  263. package/src/main.ts +773 -0
  264. package/src/mcp/client.ts +239 -0
  265. package/src/mcp/config-writer.ts +215 -0
  266. package/src/mcp/config.ts +363 -0
  267. package/src/mcp/index.ts +55 -0
  268. package/src/mcp/json-rpc.ts +84 -0
  269. package/src/mcp/loader.ts +124 -0
  270. package/src/mcp/manager.ts +490 -0
  271. package/src/mcp/oauth-discovery.ts +274 -0
  272. package/src/mcp/oauth-flow.ts +229 -0
  273. package/src/mcp/render.ts +123 -0
  274. package/src/mcp/tool-bridge.ts +372 -0
  275. package/src/mcp/tool-cache.ts +121 -0
  276. package/src/mcp/transports/http.ts +332 -0
  277. package/src/mcp/transports/index.ts +6 -0
  278. package/src/mcp/transports/stdio.ts +281 -0
  279. package/src/mcp/types.ts +248 -0
  280. package/src/memories/index.ts +1099 -0
  281. package/src/memories/storage.ts +563 -0
  282. package/src/modes/components/agent-dashboard.ts +1130 -0
  283. package/src/modes/components/assistant-message.ts +144 -0
  284. package/src/modes/components/bash-execution.ts +218 -0
  285. package/src/modes/components/bordered-loader.ts +41 -0
  286. package/src/modes/components/branch-summary-message.ts +45 -0
  287. package/src/modes/components/codemode-group.ts +369 -0
  288. package/src/modes/components/compaction-summary-message.ts +51 -0
  289. package/src/modes/components/countdown-timer.ts +46 -0
  290. package/src/modes/components/custom-editor.ts +181 -0
  291. package/src/modes/components/custom-message.ts +91 -0
  292. package/src/modes/components/diff.ts +186 -0
  293. package/src/modes/components/dynamic-border.ts +25 -0
  294. package/src/modes/components/extensions/extension-dashboard.ts +325 -0
  295. package/src/modes/components/extensions/extension-list.ts +484 -0
  296. package/src/modes/components/extensions/index.ts +9 -0
  297. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  298. package/src/modes/components/extensions/state-manager.ts +586 -0
  299. package/src/modes/components/extensions/types.ts +191 -0
  300. package/src/modes/components/footer.ts +315 -0
  301. package/src/modes/components/history-search.ts +157 -0
  302. package/src/modes/components/hook-editor.ts +101 -0
  303. package/src/modes/components/hook-input.ts +72 -0
  304. package/src/modes/components/hook-message.ts +100 -0
  305. package/src/modes/components/hook-selector.ts +155 -0
  306. package/src/modes/components/index.ts +41 -0
  307. package/src/modes/components/keybinding-hints.ts +65 -0
  308. package/src/modes/components/login-dialog.ts +164 -0
  309. package/src/modes/components/mcp-add-wizard.ts +1295 -0
  310. package/src/modes/components/model-selector.ts +625 -0
  311. package/src/modes/components/oauth-selector.ts +210 -0
  312. package/src/modes/components/plugin-settings.ts +477 -0
  313. package/src/modes/components/python-execution.ts +196 -0
  314. package/src/modes/components/queue-mode-selector.ts +56 -0
  315. package/src/modes/components/read-tool-group.ts +119 -0
  316. package/src/modes/components/session-selector.ts +242 -0
  317. package/src/modes/components/settings-defs.ts +340 -0
  318. package/src/modes/components/settings-selector.ts +529 -0
  319. package/src/modes/components/show-images-selector.ts +45 -0
  320. package/src/modes/components/skill-message.ts +90 -0
  321. package/src/modes/components/status-line/index.ts +4 -0
  322. package/src/modes/components/status-line/presets.ts +94 -0
  323. package/src/modes/components/status-line/segments.ts +352 -0
  324. package/src/modes/components/status-line/separators.ts +55 -0
  325. package/src/modes/components/status-line/types.ts +75 -0
  326. package/src/modes/components/status-line-segment-editor.ts +354 -0
  327. package/src/modes/components/status-line.ts +421 -0
  328. package/src/modes/components/theme-selector.ts +63 -0
  329. package/src/modes/components/thinking-selector.ts +64 -0
  330. package/src/modes/components/todo-display.ts +115 -0
  331. package/src/modes/components/todo-reminder.ts +40 -0
  332. package/src/modes/components/tool-execution.ts +703 -0
  333. package/src/modes/components/tree-selector.ts +904 -0
  334. package/src/modes/components/ttsr-notification.ts +80 -0
  335. package/src/modes/components/user-message-selector.ts +146 -0
  336. package/src/modes/components/user-message.ts +22 -0
  337. package/src/modes/components/visual-truncate.ts +63 -0
  338. package/src/modes/components/welcome.ts +247 -0
  339. package/src/modes/controllers/command-controller.ts +1120 -0
  340. package/src/modes/controllers/event-controller.ts +479 -0
  341. package/src/modes/controllers/extension-ui-controller.ts +778 -0
  342. package/src/modes/controllers/input-controller.ts +671 -0
  343. package/src/modes/controllers/mcp-command-controller.ts +1315 -0
  344. package/src/modes/controllers/selector-controller.ts +712 -0
  345. package/src/modes/controllers/ssh-command-controller.ts +452 -0
  346. package/src/modes/index.ts +15 -0
  347. package/src/modes/interactive-mode.ts +1027 -0
  348. package/src/modes/print-mode.ts +191 -0
  349. package/src/modes/rpc/rpc-client.ts +583 -0
  350. package/src/modes/rpc/rpc-mode.ts +700 -0
  351. package/src/modes/rpc/rpc-types.ts +236 -0
  352. package/src/modes/theme/dark.json +95 -0
  353. package/src/modes/theme/defaults/alabaster.json +93 -0
  354. package/src/modes/theme/defaults/amethyst.json +96 -0
  355. package/src/modes/theme/defaults/anthracite.json +93 -0
  356. package/src/modes/theme/defaults/basalt.json +91 -0
  357. package/src/modes/theme/defaults/birch.json +95 -0
  358. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  359. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  360. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  361. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  362. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  363. package/src/modes/theme/defaults/dark-copper.json +95 -0
  364. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  365. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  366. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  367. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  368. package/src/modes/theme/defaults/dark-ember.json +95 -0
  369. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  370. package/src/modes/theme/defaults/dark-forest.json +96 -0
  371. package/src/modes/theme/defaults/dark-github.json +105 -0
  372. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  373. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  374. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  375. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  376. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  377. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  378. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  379. package/src/modes/theme/defaults/dark-nord.json +97 -0
  380. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  381. package/src/modes/theme/defaults/dark-one.json +100 -0
  382. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  383. package/src/modes/theme/defaults/dark-reef.json +91 -0
  384. package/src/modes/theme/defaults/dark-retro.json +92 -0
  385. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  386. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  387. package/src/modes/theme/defaults/dark-slate.json +95 -0
  388. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  389. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  390. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  391. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  392. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  393. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  394. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  395. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  396. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  397. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  398. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  399. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  400. package/src/modes/theme/defaults/graphite.json +92 -0
  401. package/src/modes/theme/defaults/index.ts +195 -0
  402. package/src/modes/theme/defaults/light-arctic.json +107 -0
  403. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  404. package/src/modes/theme/defaults/light-canyon.json +91 -0
  405. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  406. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  407. package/src/modes/theme/defaults/light-coral.json +95 -0
  408. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  409. package/src/modes/theme/defaults/light-dawn.json +90 -0
  410. package/src/modes/theme/defaults/light-dunes.json +91 -0
  411. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  412. package/src/modes/theme/defaults/light-forest.json +100 -0
  413. package/src/modes/theme/defaults/light-frost.json +95 -0
  414. package/src/modes/theme/defaults/light-github.json +115 -0
  415. package/src/modes/theme/defaults/light-glacier.json +91 -0
  416. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  417. package/src/modes/theme/defaults/light-haze.json +90 -0
  418. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  419. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  420. package/src/modes/theme/defaults/light-lavender.json +95 -0
  421. package/src/modes/theme/defaults/light-meadow.json +91 -0
  422. package/src/modes/theme/defaults/light-mint.json +95 -0
  423. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  424. package/src/modes/theme/defaults/light-ocean.json +99 -0
  425. package/src/modes/theme/defaults/light-one.json +99 -0
  426. package/src/modes/theme/defaults/light-opal.json +91 -0
  427. package/src/modes/theme/defaults/light-orchard.json +91 -0
  428. package/src/modes/theme/defaults/light-paper.json +95 -0
  429. package/src/modes/theme/defaults/light-prism.json +90 -0
  430. package/src/modes/theme/defaults/light-retro.json +98 -0
  431. package/src/modes/theme/defaults/light-sand.json +95 -0
  432. package/src/modes/theme/defaults/light-savanna.json +91 -0
  433. package/src/modes/theme/defaults/light-solarized.json +102 -0
  434. package/src/modes/theme/defaults/light-soleil.json +90 -0
  435. package/src/modes/theme/defaults/light-sunset.json +99 -0
  436. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  437. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  438. package/src/modes/theme/defaults/light-wetland.json +91 -0
  439. package/src/modes/theme/defaults/light-zenith.json +89 -0
  440. package/src/modes/theme/defaults/limestone.json +94 -0
  441. package/src/modes/theme/defaults/mahogany.json +97 -0
  442. package/src/modes/theme/defaults/marble.json +93 -0
  443. package/src/modes/theme/defaults/obsidian.json +91 -0
  444. package/src/modes/theme/defaults/onyx.json +91 -0
  445. package/src/modes/theme/defaults/pearl.json +93 -0
  446. package/src/modes/theme/defaults/porcelain.json +91 -0
  447. package/src/modes/theme/defaults/quartz.json +96 -0
  448. package/src/modes/theme/defaults/sandstone.json +95 -0
  449. package/src/modes/theme/defaults/titanium.json +90 -0
  450. package/src/modes/theme/light.json +93 -0
  451. package/src/modes/theme/mermaid-cache.ts +111 -0
  452. package/src/modes/theme/theme-schema.json +429 -0
  453. package/src/modes/theme/theme.ts +2333 -0
  454. package/src/modes/types.ts +216 -0
  455. package/src/modes/utils/ui-helpers.ts +529 -0
  456. package/src/patch/applicator.ts +1482 -0
  457. package/src/patch/diff.ts +425 -0
  458. package/src/patch/fuzzy.ts +784 -0
  459. package/src/patch/hashline.ts +972 -0
  460. package/src/patch/index.ts +964 -0
  461. package/src/patch/normalize.ts +397 -0
  462. package/src/patch/normative.ts +72 -0
  463. package/src/patch/parser.ts +532 -0
  464. package/src/patch/shared.ts +400 -0
  465. package/src/patch/types.ts +292 -0
  466. package/src/priority.json +35 -0
  467. package/src/prompts/agents/explore.md +48 -0
  468. package/src/prompts/agents/frontmatter.md +9 -0
  469. package/src/prompts/agents/init.md +36 -0
  470. package/src/prompts/agents/librarian.md +53 -0
  471. package/src/prompts/agents/oracle.md +51 -0
  472. package/src/prompts/agents/reviewer.md +70 -0
  473. package/src/prompts/agents/task.md +14 -0
  474. package/src/prompts/compaction/branch-summary-context.md +5 -0
  475. package/src/prompts/compaction/branch-summary-preamble.md +2 -0
  476. package/src/prompts/compaction/branch-summary.md +30 -0
  477. package/src/prompts/compaction/compaction-short-summary.md +9 -0
  478. package/src/prompts/compaction/compaction-summary-context.md +5 -0
  479. package/src/prompts/compaction/compaction-summary.md +38 -0
  480. package/src/prompts/compaction/compaction-turn-prefix.md +17 -0
  481. package/src/prompts/compaction/compaction-update-summary.md +45 -0
  482. package/src/prompts/memories/consolidation.md +30 -0
  483. package/src/prompts/memories/read_path.md +11 -0
  484. package/src/prompts/memories/stage_one_input.md +6 -0
  485. package/src/prompts/memories/stage_one_system.md +21 -0
  486. package/src/prompts/review-request.md +64 -0
  487. package/src/prompts/system/agent-creation-architect.md +65 -0
  488. package/src/prompts/system/agent-creation-user.md +6 -0
  489. package/src/prompts/system/custom-system-prompt.md +68 -0
  490. package/src/prompts/system/file-operations.md +10 -0
  491. package/src/prompts/system/subagent-submit-reminder.md +11 -0
  492. package/src/prompts/system/subagent-system-prompt.md +31 -0
  493. package/src/prompts/system/subagent-user-prompt.md +8 -0
  494. package/src/prompts/system/summarization-system.md +3 -0
  495. package/src/prompts/system/system-prompt.md +300 -0
  496. package/src/prompts/system/title-system.md +2 -0
  497. package/src/prompts/system/ttsr-interrupt.md +7 -0
  498. package/src/prompts/system/web-search.md +28 -0
  499. package/src/prompts/tools/ask.md +44 -0
  500. package/src/prompts/tools/bash.md +24 -0
  501. package/src/prompts/tools/browser.md +33 -0
  502. package/src/prompts/tools/calculator.md +12 -0
  503. package/src/prompts/tools/explore.md +29 -0
  504. package/src/prompts/tools/fetch.md +16 -0
  505. package/src/prompts/tools/find.md +18 -0
  506. package/src/prompts/tools/gemini-image.md +23 -0
  507. package/src/prompts/tools/grep.md +28 -0
  508. package/src/prompts/tools/hashline.md +232 -0
  509. package/src/prompts/tools/librarian.md +24 -0
  510. package/src/prompts/tools/lsp.md +28 -0
  511. package/src/prompts/tools/oracle.md +26 -0
  512. package/src/prompts/tools/patch.md +74 -0
  513. package/src/prompts/tools/python.md +66 -0
  514. package/src/prompts/tools/read.md +36 -0
  515. package/src/prompts/tools/replace.md +38 -0
  516. package/src/prompts/tools/reviewer.md +41 -0
  517. package/src/prompts/tools/ssh.md +51 -0
  518. package/src/prompts/tools/task-summary.md +28 -0
  519. package/src/prompts/tools/task.md +275 -0
  520. package/src/prompts/tools/todo-write.md +65 -0
  521. package/src/prompts/tools/undo-edit.md +7 -0
  522. package/src/prompts/tools/web-search.md +19 -0
  523. package/src/prompts/tools/write.md +18 -0
  524. package/src/sdk.ts +1287 -0
  525. package/src/secrets/index.ts +116 -0
  526. package/src/secrets/obfuscator.ts +269 -0
  527. package/src/secrets/regex.ts +21 -0
  528. package/src/session/agent-session.ts +4669 -0
  529. package/src/session/agent-storage.ts +621 -0
  530. package/src/session/artifacts.ts +132 -0
  531. package/src/session/auth-storage.ts +1433 -0
  532. package/src/session/blob-store.ts +103 -0
  533. package/src/session/compaction/branch-summarization.ts +315 -0
  534. package/src/session/compaction/compaction.ts +864 -0
  535. package/src/session/compaction/index.ts +7 -0
  536. package/src/session/compaction/pruning.ts +91 -0
  537. package/src/session/compaction/utils.ts +171 -0
  538. package/src/session/history-storage.ts +170 -0
  539. package/src/session/messages.ts +317 -0
  540. package/src/session/session-manager.ts +2276 -0
  541. package/src/session/session-storage.ts +342 -0
  542. package/src/session/streaming-output.ts +565 -0
  543. package/src/slash-commands/builtin-registry.ts +439 -0
  544. package/src/ssh/config-writer.ts +183 -0
  545. package/src/ssh/connection-manager.ts +444 -0
  546. package/src/ssh/ssh-executor.ts +127 -0
  547. package/src/ssh/sshfs-mount.ts +135 -0
  548. package/src/stt/downloader.ts +71 -0
  549. package/src/stt/index.ts +3 -0
  550. package/src/stt/recorder.ts +351 -0
  551. package/src/stt/setup.ts +52 -0
  552. package/src/stt/stt-controller.ts +160 -0
  553. package/src/stt/transcribe.py +70 -0
  554. package/src/stt/transcriber.ts +91 -0
  555. package/src/system-prompt.ts +685 -0
  556. package/src/task/agents.ts +155 -0
  557. package/src/task/batch.ts +102 -0
  558. package/src/task/commands.ts +134 -0
  559. package/src/task/discovery.ts +126 -0
  560. package/src/task/executor.ts +908 -0
  561. package/src/task/index.ts +223 -0
  562. package/src/task/output-manager.ts +107 -0
  563. package/src/task/parallel.ts +84 -0
  564. package/src/task/render.ts +326 -0
  565. package/src/task/subprocess-tool-registry.ts +88 -0
  566. package/src/task/template.ts +32 -0
  567. package/src/task/types.ts +144 -0
  568. package/src/tools/ask.ts +523 -0
  569. package/src/tools/bash-interactive.ts +419 -0
  570. package/src/tools/bash-interceptor.ts +105 -0
  571. package/src/tools/bash-normalize.ts +107 -0
  572. package/src/tools/bash-skill-urls.ts +177 -0
  573. package/src/tools/bash.ts +347 -0
  574. package/src/tools/browser.ts +1374 -0
  575. package/src/tools/calculator.ts +537 -0
  576. package/src/tools/context.ts +39 -0
  577. package/src/tools/explore.ts +23 -0
  578. package/src/tools/fetch.ts +1091 -0
  579. package/src/tools/find.ts +540 -0
  580. package/src/tools/fs-cache-invalidation.ts +28 -0
  581. package/src/tools/gemini-image.ts +907 -0
  582. package/src/tools/grep.ts +489 -0
  583. package/src/tools/index.ts +337 -0
  584. package/src/tools/json-tree.ts +231 -0
  585. package/src/tools/jtd-to-json-schema.ts +247 -0
  586. package/src/tools/jtd-to-typescript.ts +198 -0
  587. package/src/tools/librarian.ts +33 -0
  588. package/src/tools/list-limit.ts +40 -0
  589. package/src/tools/notebook.ts +287 -0
  590. package/src/tools/oracle.ts +40 -0
  591. package/src/tools/output-meta.ts +459 -0
  592. package/src/tools/output-utils.ts +63 -0
  593. package/src/tools/path-utils.ts +116 -0
  594. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  595. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  596. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  597. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  598. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  599. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  600. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  601. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  602. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  603. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  604. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  605. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  606. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  607. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  608. package/src/tools/python.ts +1118 -0
  609. package/src/tools/read.ts +1193 -0
  610. package/src/tools/render-utils.ts +680 -0
  611. package/src/tools/renderers.ts +60 -0
  612. package/src/tools/reviewer-tool.ts +41 -0
  613. package/src/tools/ssh.ts +326 -0
  614. package/src/tools/subagent-tool.ts +169 -0
  615. package/src/tools/submit-result.ts +152 -0
  616. package/src/tools/todo-write.ts +255 -0
  617. package/src/tools/tool-errors.ts +92 -0
  618. package/src/tools/tool-result.ts +86 -0
  619. package/src/tools/undo-edit.ts +145 -0
  620. package/src/tools/undo-history.ts +22 -0
  621. package/src/tools/write.ts +274 -0
  622. package/src/tui/code-cell.ts +108 -0
  623. package/src/tui/file-list.ts +47 -0
  624. package/src/tui/index.ts +11 -0
  625. package/src/tui/output-block.ts +144 -0
  626. package/src/tui/status-line.ts +39 -0
  627. package/src/tui/tree-list.ts +53 -0
  628. package/src/tui/types.ts +16 -0
  629. package/src/tui/utils.ts +116 -0
  630. package/src/utils/changelog.ts +98 -0
  631. package/src/utils/event-bus.ts +33 -0
  632. package/src/utils/external-editor.ts +59 -0
  633. package/src/utils/file-display-mode.ts +36 -0
  634. package/src/utils/file-mentions.ts +384 -0
  635. package/src/utils/frontmatter.ts +101 -0
  636. package/src/utils/fuzzy.ts +108 -0
  637. package/src/utils/ignore-files.ts +119 -0
  638. package/src/utils/image-convert.ts +27 -0
  639. package/src/utils/image-resize.ts +236 -0
  640. package/src/utils/mime.ts +30 -0
  641. package/src/utils/open.ts +20 -0
  642. package/src/utils/shell-snapshot.ts +199 -0
  643. package/src/utils/timings.ts +26 -0
  644. package/src/utils/title-generator.ts +167 -0
  645. package/src/utils/tools-manager.ts +362 -0
  646. package/src/web/scrapers/artifacthub.ts +215 -0
  647. package/src/web/scrapers/arxiv.ts +88 -0
  648. package/src/web/scrapers/aur.ts +175 -0
  649. package/src/web/scrapers/biorxiv.ts +141 -0
  650. package/src/web/scrapers/bluesky.ts +284 -0
  651. package/src/web/scrapers/brew.ts +177 -0
  652. package/src/web/scrapers/cheatsh.ts +78 -0
  653. package/src/web/scrapers/chocolatey.ts +158 -0
  654. package/src/web/scrapers/choosealicense.ts +110 -0
  655. package/src/web/scrapers/cisa-kev.ts +100 -0
  656. package/src/web/scrapers/clojars.ts +180 -0
  657. package/src/web/scrapers/coingecko.ts +184 -0
  658. package/src/web/scrapers/crates-io.ts +128 -0
  659. package/src/web/scrapers/crossref.ts +149 -0
  660. package/src/web/scrapers/devto.ts +177 -0
  661. package/src/web/scrapers/discogs.ts +307 -0
  662. package/src/web/scrapers/discourse.ts +221 -0
  663. package/src/web/scrapers/dockerhub.ts +160 -0
  664. package/src/web/scrapers/fdroid.ts +158 -0
  665. package/src/web/scrapers/firefox-addons.ts +214 -0
  666. package/src/web/scrapers/flathub.ts +239 -0
  667. package/src/web/scrapers/github-gist.ts +68 -0
  668. package/src/web/scrapers/github.ts +490 -0
  669. package/src/web/scrapers/gitlab.ts +456 -0
  670. package/src/web/scrapers/go-pkg.ts +275 -0
  671. package/src/web/scrapers/hackage.ts +94 -0
  672. package/src/web/scrapers/hackernews.ts +208 -0
  673. package/src/web/scrapers/hex.ts +121 -0
  674. package/src/web/scrapers/huggingface.ts +385 -0
  675. package/src/web/scrapers/iacr.ts +86 -0
  676. package/src/web/scrapers/index.ts +249 -0
  677. package/src/web/scrapers/jetbrains-marketplace.ts +169 -0
  678. package/src/web/scrapers/lemmy.ts +220 -0
  679. package/src/web/scrapers/lobsters.ts +186 -0
  680. package/src/web/scrapers/mastodon.ts +310 -0
  681. package/src/web/scrapers/maven.ts +152 -0
  682. package/src/web/scrapers/mdn.ts +172 -0
  683. package/src/web/scrapers/metacpan.ts +253 -0
  684. package/src/web/scrapers/musicbrainz.ts +272 -0
  685. package/src/web/scrapers/npm.ts +114 -0
  686. package/src/web/scrapers/nuget.ts +205 -0
  687. package/src/web/scrapers/nvd.ts +243 -0
  688. package/src/web/scrapers/ollama.ts +265 -0
  689. package/src/web/scrapers/open-vsx.ts +119 -0
  690. package/src/web/scrapers/opencorporates.ts +275 -0
  691. package/src/web/scrapers/openlibrary.ts +319 -0
  692. package/src/web/scrapers/orcid.ts +298 -0
  693. package/src/web/scrapers/osv.ts +192 -0
  694. package/src/web/scrapers/packagist.ts +174 -0
  695. package/src/web/scrapers/pub-dev.ts +185 -0
  696. package/src/web/scrapers/pubmed.ts +177 -0
  697. package/src/web/scrapers/pypi.ts +129 -0
  698. package/src/web/scrapers/rawg.ts +124 -0
  699. package/src/web/scrapers/readthedocs.ts +125 -0
  700. package/src/web/scrapers/reddit.ts +104 -0
  701. package/src/web/scrapers/repology.ts +262 -0
  702. package/src/web/scrapers/rfc.ts +209 -0
  703. package/src/web/scrapers/rubygems.ts +117 -0
  704. package/src/web/scrapers/searchcode.ts +217 -0
  705. package/src/web/scrapers/sec-edgar.ts +274 -0
  706. package/src/web/scrapers/semantic-scholar.ts +190 -0
  707. package/src/web/scrapers/snapcraft.ts +200 -0
  708. package/src/web/scrapers/sourcegraph.ts +373 -0
  709. package/src/web/scrapers/spdx.ts +121 -0
  710. package/src/web/scrapers/spotify.ts +217 -0
  711. package/src/web/scrapers/stackoverflow.ts +124 -0
  712. package/src/web/scrapers/terraform.ts +304 -0
  713. package/src/web/scrapers/tldr.ts +51 -0
  714. package/src/web/scrapers/twitter.ts +97 -0
  715. package/src/web/scrapers/types.ts +200 -0
  716. package/src/web/scrapers/utils.ts +142 -0
  717. package/src/web/scrapers/vimeo.ts +152 -0
  718. package/src/web/scrapers/vscode-marketplace.ts +195 -0
  719. package/src/web/scrapers/w3c.ts +163 -0
  720. package/src/web/scrapers/wikidata.ts +357 -0
  721. package/src/web/scrapers/wikipedia.ts +95 -0
  722. package/src/web/scrapers/youtube.ts +312 -0
  723. package/src/web/search/auth.ts +178 -0
  724. package/src/web/search/index.ts +598 -0
  725. package/src/web/search/provider.ts +77 -0
  726. package/src/web/search/providers/anthropic.ts +284 -0
  727. package/src/web/search/providers/base.ts +22 -0
  728. package/src/web/search/providers/brave.ts +165 -0
  729. package/src/web/search/providers/codex.ts +377 -0
  730. package/src/web/search/providers/exa.ts +158 -0
  731. package/src/web/search/providers/gemini.ts +437 -0
  732. package/src/web/search/providers/jina.ts +99 -0
  733. package/src/web/search/providers/kimi.ts +196 -0
  734. package/src/web/search/providers/perplexity.ts +546 -0
  735. package/src/web/search/providers/synthetic.ts +136 -0
  736. package/src/web/search/providers/zai.ts +352 -0
  737. package/src/web/search/render.ts +299 -0
  738. package/src/web/search/types.ts +437 -0
@@ -0,0 +1,1118 @@
1
+ import type * as fs from "node:fs";
2
+ import * as path from "node:path";
3
+ import type { AgentTool, AgentToolContext, AgentToolResult, AgentToolUpdateCallback } from "@nghyane/arcane-agent";
4
+ import type { ImageContent } from "@nghyane/arcane-ai";
5
+ import type { Component } from "@nghyane/arcane-tui";
6
+ import { Text } from "@nghyane/arcane-tui";
7
+ import { getProjectDir } from "@nghyane/arcane-utils/dirs";
8
+ import { type Static, Type } from "@sinclair/typebox";
9
+ import { renderPromptTemplate } from "../config/prompt-templates";
10
+ import type { RenderResultOptions } from "../extensibility/custom-tools/types";
11
+ import { executePython, getPreludeDocs, type PythonExecutorOptions } from "../ipy/executor";
12
+ import type { PreludeHelper, PythonStatusEvent } from "../ipy/kernel";
13
+ import { truncateToVisualLines } from "../modes/components/visual-truncate";
14
+ import type { Theme } from "../modes/theme/theme";
15
+ import pythonDescription from "../prompts/tools/python.md" with { type: "text" };
16
+ import { DEFAULT_MAX_BYTES, OutputSink, type OutputSummary } from "../session/streaming-output";
17
+ import { getTreeBranch, getTreeContinuePrefix, renderCodeCell } from "../tui";
18
+ import type { ToolSession } from ".";
19
+ import type { OutputMeta } from "./output-meta";
20
+ import { allocateOutputArtifact, createTailBuffer } from "./output-utils";
21
+ import { resolveToCwd } from "./path-utils";
22
+ import { replaceTabs, shortenPath, ToolUIKit, truncateToWidth } from "./render-utils";
23
+ import { ToolAbortError, ToolError } from "./tool-errors";
24
+ import { toolResult } from "./tool-result";
25
+
26
+ export const PYTHON_DEFAULT_PREVIEW_LINES = 10;
27
+
28
+ type PreludeCategory = {
29
+ name: string;
30
+ functions: PreludeHelper[];
31
+ };
32
+
33
+ function groupPreludeHelpers(helpers: PreludeHelper[]): PreludeCategory[] {
34
+ const categories: PreludeCategory[] = [];
35
+ const byName = new Map<string, PreludeHelper[]>();
36
+ for (const helper of helpers) {
37
+ let bucket = byName.get(helper.category);
38
+ if (!bucket) {
39
+ bucket = [];
40
+ byName.set(helper.category, bucket);
41
+ categories.push({ name: helper.category, functions: bucket });
42
+ }
43
+ bucket.push(helper);
44
+ }
45
+ return categories;
46
+ }
47
+
48
+ export const pythonSchema = Type.Object({
49
+ cells: Type.Array(
50
+ Type.Object({
51
+ code: Type.String({ description: "Python code to execute" }),
52
+ title: Type.Optional(Type.String({ description: "Cell label, e.g. 'imports', 'helper'" })),
53
+ }),
54
+ { description: "Cells to execute sequentially in persistent kernel" },
55
+ ),
56
+ timeout: Type.Optional(Type.Number({ description: "Timeout in seconds (default: 30)" })),
57
+ cwd: Type.Optional(Type.String({ description: "Working directory (default: cwd)" })),
58
+ reset: Type.Optional(Type.Boolean({ description: "Restart kernel before execution" })),
59
+ });
60
+ export type PythonToolParams = Static<typeof pythonSchema>;
61
+
62
+ export type PythonToolResult = {
63
+ content: Array<{ type: "text"; text: string }>;
64
+ details: PythonToolDetails | undefined;
65
+ };
66
+
67
+ export type PythonProxyExecutor = (params: PythonToolParams, signal?: AbortSignal) => Promise<PythonToolResult>;
68
+
69
+ export interface PythonCellResult {
70
+ index: number;
71
+ title?: string;
72
+ code: string;
73
+ output: string;
74
+ status: "pending" | "running" | "complete" | "error";
75
+ durationMs?: number;
76
+ exitCode?: number;
77
+ statusEvents?: PythonStatusEvent[];
78
+ }
79
+
80
+ export interface PythonToolDetails {
81
+ cells?: PythonCellResult[];
82
+ jsonOutputs?: unknown[];
83
+ images?: ImageContent[];
84
+ /** Structured status events from prelude helpers */
85
+ statusEvents?: PythonStatusEvent[];
86
+ isError?: boolean;
87
+ /** Structured output metadata for notices */
88
+ meta?: OutputMeta;
89
+ }
90
+
91
+ function formatJsonScalar(value: unknown): string {
92
+ if (value === null) return "null";
93
+ if (value === undefined) return "undefined";
94
+ if (typeof value === "string") return JSON.stringify(value);
95
+ if (typeof value === "number" || typeof value === "boolean" || typeof value === "bigint") return String(value);
96
+ if (typeof value === "function") return "[function]";
97
+ return "[object]";
98
+ }
99
+
100
+ function renderJsonTree(value: unknown, theme: Theme, expanded: boolean, maxDepth = expanded ? 6 : 2): string[] {
101
+ const maxItems = expanded ? 20 : 5;
102
+
103
+ const renderNode = (node: unknown, prefix: string, depth: number, isLast: boolean, label?: string): string[] => {
104
+ const branch = getTreeBranch(isLast, theme);
105
+ const displayLabel = label ? `${label}: ` : "";
106
+
107
+ if (depth >= maxDepth || node === null || typeof node !== "object") {
108
+ return [`${prefix}${branch} ${displayLabel}${formatJsonScalar(node)}`];
109
+ }
110
+
111
+ const isArray = Array.isArray(node);
112
+ const entries = isArray
113
+ ? node.map((val, index) => [String(index), val] as const)
114
+ : Object.entries(node as object);
115
+ const header = `${prefix}${branch} ${displayLabel}${isArray ? `Array(${entries.length})` : `Object(${entries.length})`}`;
116
+ const lines = [header];
117
+
118
+ const childPrefix = prefix + getTreeContinuePrefix(isLast, theme);
119
+ const visible = entries.slice(0, maxItems);
120
+ for (let i = 0; i < visible.length; i++) {
121
+ const [key, val] = visible[i];
122
+ const childLast = i === visible.length - 1 && (expanded || entries.length <= maxItems);
123
+ lines.push(...renderNode(val, childPrefix, depth + 1, childLast, isArray ? `[${key}]` : key));
124
+ }
125
+ if (!expanded && entries.length > maxItems) {
126
+ const moreBranch = theme.tree.last;
127
+ lines.push(`${childPrefix}${moreBranch} ${entries.length - maxItems} more item(s)`);
128
+ }
129
+ return lines;
130
+ };
131
+
132
+ return renderNode(value, "", 0, true);
133
+ }
134
+
135
+ export function getPythonToolDescription(): string {
136
+ const helpers = getPreludeDocs();
137
+ const categories = groupPreludeHelpers(helpers);
138
+ return renderPromptTemplate(pythonDescription, { categories });
139
+ }
140
+
141
+ export interface PythonToolOptions {
142
+ proxyExecutor?: PythonProxyExecutor;
143
+ }
144
+
145
+ export class PythonTool implements AgentTool<typeof pythonSchema> {
146
+ readonly name = "python";
147
+ readonly label = "Python";
148
+ readonly description: string;
149
+ readonly parameters = pythonSchema;
150
+ readonly concurrency = "exclusive";
151
+
152
+ readonly #proxyExecutor?: PythonProxyExecutor;
153
+
154
+ constructor(
155
+ private readonly session: ToolSession | null,
156
+ options?: PythonToolOptions,
157
+ ) {
158
+ this.#proxyExecutor = options?.proxyExecutor;
159
+ this.description = getPythonToolDescription();
160
+ }
161
+
162
+ async execute(
163
+ _toolCallId: string,
164
+ params: Static<typeof pythonSchema>,
165
+ signal?: AbortSignal,
166
+ onUpdate?: AgentToolUpdateCallback,
167
+ _ctx?: AgentToolContext,
168
+ ): Promise<AgentToolResult<PythonToolDetails | undefined>> {
169
+ if (this.#proxyExecutor) {
170
+ return this.#proxyExecutor(params, signal);
171
+ }
172
+
173
+ if (!this.session) {
174
+ throw new ToolError("Python tool requires a session when not using proxy executor");
175
+ }
176
+
177
+ const { cells, timeout: rawTimeout = 30, cwd, reset } = params;
178
+ // Clamp to reasonable range: 1s - 600s (10 min)
179
+ const timeoutSec = Math.max(1, Math.min(600, rawTimeout));
180
+ const timeoutMs = timeoutSec * 1000;
181
+ const timeoutSignal = AbortSignal.timeout(timeoutMs);
182
+ const combinedSignal = signal ? AbortSignal.any([signal, timeoutSignal]) : timeoutSignal;
183
+ let outputSink: OutputSink | undefined;
184
+ let outputSummary: OutputSummary | undefined;
185
+ let outputDumped = false;
186
+ const finalizeOutput = async (): Promise<OutputSummary | undefined> => {
187
+ if (outputDumped || !outputSink) return outputSummary;
188
+ outputSummary = await outputSink.dump();
189
+ outputDumped = true;
190
+ return outputSummary;
191
+ };
192
+
193
+ try {
194
+ if (signal?.aborted) {
195
+ throw new ToolAbortError();
196
+ }
197
+
198
+ const commandCwd = cwd ? resolveToCwd(cwd, this.session.cwd) : this.session.cwd;
199
+ let cwdStat: fs.Stats;
200
+ try {
201
+ cwdStat = await Bun.file(commandCwd).stat();
202
+ } catch {
203
+ throw new ToolError(`Working directory does not exist: ${commandCwd}`);
204
+ }
205
+ if (!cwdStat.isDirectory()) {
206
+ throw new ToolError(`Working directory is not a directory: ${commandCwd}`);
207
+ }
208
+
209
+ const tailBuffer = createTailBuffer(DEFAULT_MAX_BYTES * 2);
210
+ const jsonOutputs: unknown[] = [];
211
+ const images: ImageContent[] = [];
212
+ const statusEvents: PythonStatusEvent[] = [];
213
+
214
+ const cellResults: PythonCellResult[] = cells.map((cell, index) => ({
215
+ index,
216
+ title: cell.title,
217
+ code: cell.code,
218
+ output: "",
219
+ status: "pending",
220
+ }));
221
+ const cellOutputs: string[] = [];
222
+
223
+ const appendTail = (text: string) => {
224
+ tailBuffer.append(text);
225
+ };
226
+
227
+ const buildUpdateDetails = (): PythonToolDetails => {
228
+ const details: PythonToolDetails = {
229
+ cells: cellResults.map(cell => ({
230
+ ...cell,
231
+ statusEvents: cell.statusEvents ? [...cell.statusEvents] : undefined,
232
+ })),
233
+ };
234
+ if (jsonOutputs.length > 0) {
235
+ details.jsonOutputs = jsonOutputs;
236
+ }
237
+ if (images.length > 0) {
238
+ details.images = images;
239
+ }
240
+ if (statusEvents.length > 0) {
241
+ details.statusEvents = statusEvents;
242
+ }
243
+ return details;
244
+ };
245
+
246
+ const pushUpdate = () => {
247
+ if (!onUpdate) return;
248
+ const tailText = tailBuffer.text();
249
+ onUpdate({
250
+ content: [{ type: "text", text: tailText }],
251
+ details: buildUpdateDetails(),
252
+ });
253
+ };
254
+
255
+ const sessionFile = this.session.getSessionFile?.() ?? undefined;
256
+ const artifactsDir = this.session.getArtifactsDir?.() ?? undefined;
257
+ const { artifactPath, artifactId } = await allocateOutputArtifact(this.session, "python");
258
+ outputSink = new OutputSink({
259
+ artifactPath,
260
+ artifactId,
261
+ onChunk: chunk => {
262
+ appendTail(chunk);
263
+ pushUpdate();
264
+ },
265
+ });
266
+ const sessionId = sessionFile ? `session:${sessionFile}:cwd:${commandCwd}` : `cwd:${commandCwd}`;
267
+ const baseExecutorOptions: Omit<PythonExecutorOptions, "reset"> = {
268
+ cwd: commandCwd,
269
+ timeoutMs,
270
+ signal: combinedSignal,
271
+ sessionId,
272
+ kernelMode: this.session.settings.get("python.kernelMode"),
273
+ useSharedGateway: this.session.settings.get("python.sharedGateway"),
274
+ sessionFile: sessionFile ?? undefined,
275
+ artifactsDir: artifactsDir ?? undefined,
276
+ };
277
+
278
+ for (let i = 0; i < cells.length; i++) {
279
+ const cell = cells[i];
280
+ const isFirstCell = i === 0;
281
+ const cellResult = cellResults[i];
282
+ cellResult.status = "running";
283
+ cellResult.output = "";
284
+ cellResult.statusEvents = undefined;
285
+ cellResult.exitCode = undefined;
286
+ cellResult.durationMs = undefined;
287
+ pushUpdate();
288
+
289
+ const executorOptions: PythonExecutorOptions = {
290
+ ...baseExecutorOptions,
291
+ reset: isFirstCell ? reset : false,
292
+ onChunk: async chunk => {
293
+ await outputSink!.push(chunk);
294
+ },
295
+ };
296
+
297
+ const startTime = Date.now();
298
+ const result = await executePython(cell.code, executorOptions);
299
+ const durationMs = Date.now() - startTime;
300
+
301
+ const cellStatusEvents: PythonStatusEvent[] = [];
302
+ for (const output of result.displayOutputs) {
303
+ if (output.type === "json") {
304
+ jsonOutputs.push(output.data);
305
+ }
306
+ if (output.type === "image") {
307
+ images.push({ type: "image", data: output.data, mimeType: output.mimeType });
308
+ }
309
+ if (output.type === "status") {
310
+ statusEvents.push(output.event);
311
+ cellStatusEvents.push(output.event);
312
+ }
313
+ }
314
+
315
+ const cellOutput = result.output.trim();
316
+ cellResult.output = cellOutput;
317
+ cellResult.exitCode = result.exitCode;
318
+ cellResult.durationMs = durationMs;
319
+ cellResult.statusEvents = cellStatusEvents.length > 0 ? cellStatusEvents : undefined;
320
+
321
+ let combinedCellOutput = "";
322
+ if (cells.length > 1) {
323
+ const cellHeader = `[${i + 1}/${cells.length}]`;
324
+ const cellTitle = cell.title ? ` ${cell.title}` : "";
325
+ if (cellOutput) {
326
+ combinedCellOutput = `${cellHeader}${cellTitle}\n${cellOutput}`;
327
+ } else {
328
+ combinedCellOutput = `${cellHeader}${cellTitle} (ok)`;
329
+ }
330
+ cellOutputs.push(combinedCellOutput);
331
+ } else if (cellOutput) {
332
+ combinedCellOutput = cellOutput;
333
+ cellOutputs.push(combinedCellOutput);
334
+ }
335
+
336
+ if (combinedCellOutput) {
337
+ const prefix = cellOutputs.length > 1 ? "\n\n" : "";
338
+ appendTail(`${prefix}${combinedCellOutput}`);
339
+ }
340
+
341
+ if (result.cancelled) {
342
+ cellResult.status = "error";
343
+ pushUpdate();
344
+ const errorMsg = result.output || "Command aborted";
345
+ const combinedOutput = cellOutputs.join("\n\n");
346
+ const outputText =
347
+ cells.length > 1
348
+ ? `${combinedOutput}\n\nCell ${i + 1} aborted: ${errorMsg}`
349
+ : combinedOutput || errorMsg;
350
+
351
+ const rawSummary = (await finalizeOutput()) ?? {
352
+ output: "",
353
+ truncated: false,
354
+ totalLines: 0,
355
+ totalBytes: 0,
356
+ outputLines: 0,
357
+ outputBytes: 0,
358
+ };
359
+ const outputLines = combinedOutput.length > 0 ? combinedOutput.split("\n").length : 0;
360
+ const outputBytes = Buffer.byteLength(combinedOutput, "utf-8");
361
+ const missingLines = Math.max(0, rawSummary.totalLines - rawSummary.outputLines);
362
+ const missingBytes = Math.max(0, rawSummary.totalBytes - rawSummary.outputBytes);
363
+ const summaryForMeta: OutputSummary = {
364
+ output: combinedOutput,
365
+ truncated: rawSummary.truncated,
366
+ totalLines: outputLines + missingLines,
367
+ totalBytes: outputBytes + missingBytes,
368
+ outputLines,
369
+ outputBytes,
370
+ artifactId: rawSummary.artifactId,
371
+ };
372
+
373
+ const details: PythonToolDetails = {
374
+ cells: cellResults,
375
+ jsonOutputs: jsonOutputs.length > 0 ? jsonOutputs : undefined,
376
+ images: images.length > 0 ? images : undefined,
377
+ statusEvents: statusEvents.length > 0 ? statusEvents : undefined,
378
+ isError: true,
379
+ };
380
+
381
+ return toolResult(details)
382
+ .text(outputText)
383
+ .truncationFromSummary(summaryForMeta, { direction: "tail" })
384
+ .done();
385
+ }
386
+
387
+ if (result.exitCode !== 0 && result.exitCode !== undefined) {
388
+ cellResult.status = "error";
389
+ pushUpdate();
390
+ const combinedOutput = cellOutputs.join("\n\n");
391
+ const outputText =
392
+ cells.length > 1
393
+ ? `${combinedOutput}\n\nCell ${i + 1} failed (exit code ${result.exitCode}). Earlier cells succeeded—their state persists. Fix only cell ${i + 1}.`
394
+ : combinedOutput
395
+ ? `${combinedOutput}\n\nCommand exited with code ${result.exitCode}`
396
+ : `Command exited with code ${result.exitCode}`;
397
+
398
+ const rawSummary = (await finalizeOutput()) ?? {
399
+ output: "",
400
+ truncated: false,
401
+ totalLines: 0,
402
+ totalBytes: 0,
403
+ outputLines: 0,
404
+ outputBytes: 0,
405
+ };
406
+ const outputLines = combinedOutput.length > 0 ? combinedOutput.split("\n").length : 0;
407
+ const outputBytes = Buffer.byteLength(combinedOutput, "utf-8");
408
+ const missingLines = Math.max(0, rawSummary.totalLines - rawSummary.outputLines);
409
+ const missingBytes = Math.max(0, rawSummary.totalBytes - rawSummary.outputBytes);
410
+ const summaryForMeta: OutputSummary = {
411
+ output: combinedOutput,
412
+ truncated: rawSummary.truncated,
413
+ totalLines: outputLines + missingLines,
414
+ totalBytes: outputBytes + missingBytes,
415
+ outputLines,
416
+ outputBytes,
417
+ artifactId: rawSummary.artifactId,
418
+ };
419
+
420
+ const details: PythonToolDetails = {
421
+ cells: cellResults,
422
+ jsonOutputs: jsonOutputs.length > 0 ? jsonOutputs : undefined,
423
+ images: images.length > 0 ? images : undefined,
424
+ statusEvents: statusEvents.length > 0 ? statusEvents : undefined,
425
+ isError: true,
426
+ };
427
+
428
+ return toolResult(details)
429
+ .text(outputText)
430
+ .truncationFromSummary(summaryForMeta, { direction: "tail" })
431
+ .done();
432
+ }
433
+
434
+ cellResult.status = "complete";
435
+ pushUpdate();
436
+ }
437
+
438
+ const combinedOutput = cellOutputs.join("\n\n");
439
+ const outputText =
440
+ combinedOutput || (jsonOutputs.length > 0 || images.length > 0 ? "(no text output)" : "(no output)");
441
+ const rawSummary = (await finalizeOutput()) ?? {
442
+ output: "",
443
+ truncated: false,
444
+ totalLines: 0,
445
+ totalBytes: 0,
446
+ outputLines: 0,
447
+ outputBytes: 0,
448
+ };
449
+ const outputLines = combinedOutput.length > 0 ? combinedOutput.split("\n").length : 0;
450
+ const outputBytes = Buffer.byteLength(combinedOutput, "utf-8");
451
+ const missingLines = Math.max(0, rawSummary.totalLines - rawSummary.outputLines);
452
+ const missingBytes = Math.max(0, rawSummary.totalBytes - rawSummary.outputBytes);
453
+ const summaryForMeta: OutputSummary = {
454
+ output: combinedOutput,
455
+ truncated: rawSummary.truncated,
456
+ totalLines: outputLines + missingLines,
457
+ totalBytes: outputBytes + missingBytes,
458
+ outputLines,
459
+ outputBytes,
460
+ artifactId: rawSummary.artifactId,
461
+ };
462
+
463
+ const details: PythonToolDetails = {
464
+ cells: cellResults,
465
+ jsonOutputs: jsonOutputs.length > 0 ? jsonOutputs : undefined,
466
+ images: images.length > 0 ? images : undefined,
467
+ statusEvents: statusEvents.length > 0 ? statusEvents : undefined,
468
+ };
469
+
470
+ const resultBuilder = toolResult(details)
471
+ .text(outputText)
472
+ .truncationFromSummary(summaryForMeta, { direction: "tail" });
473
+
474
+ return resultBuilder.done();
475
+ } finally {
476
+ if (!outputDumped) {
477
+ try {
478
+ await finalizeOutput();
479
+ } catch {}
480
+ }
481
+ }
482
+ }
483
+ }
484
+
485
+ interface PythonRenderArgs {
486
+ cells?: Array<{ code: string; title?: string }>;
487
+ timeout?: number;
488
+ cwd?: string;
489
+ }
490
+
491
+ interface PythonRenderContext {
492
+ output?: string;
493
+ expanded?: boolean;
494
+ previewLines?: number;
495
+ timeout?: number;
496
+ }
497
+
498
+ /** Format a status event as a single line for display. */
499
+ function formatStatusEvent(event: PythonStatusEvent, theme: Theme): string {
500
+ const { op, ...data } = event;
501
+
502
+ // Map operations to available theme icons
503
+ type AvailableIcon = "icon.file" | "icon.folder" | "icon.git" | "icon.package";
504
+ const opIcons: Record<string, AvailableIcon> = {
505
+ // File I/O
506
+ read: "icon.file",
507
+ write: "icon.file",
508
+ append: "icon.file",
509
+ cat: "icon.file",
510
+ touch: "icon.file",
511
+ lines: "icon.file",
512
+ // Navigation/Directory
513
+ ls: "icon.folder",
514
+ cd: "icon.folder",
515
+ pwd: "icon.folder",
516
+ mkdir: "icon.folder",
517
+ tree: "icon.folder",
518
+ stat: "icon.folder",
519
+ // Search (use file icon since no search icon)
520
+ find: "icon.file",
521
+ grep: "icon.file",
522
+ rgrep: "icon.file",
523
+ glob: "icon.file",
524
+ // Edit operations (use file icon)
525
+ replace: "icon.file",
526
+ sed: "icon.file",
527
+ rsed: "icon.file",
528
+ delete_lines: "icon.file",
529
+ delete_matching: "icon.file",
530
+ insert_at: "icon.file",
531
+ // Git
532
+ git_status: "icon.git",
533
+ git_diff: "icon.git",
534
+ git_log: "icon.git",
535
+ git_show: "icon.git",
536
+ git_branch: "icon.git",
537
+ git_file_at: "icon.git",
538
+ git_has_changes: "icon.git",
539
+ // Shell/batch (use package icon)
540
+ run: "icon.package",
541
+ sh: "icon.package",
542
+ env: "icon.package",
543
+ batch: "icon.package",
544
+ };
545
+
546
+ const iconKey = opIcons[op] ?? "icon.file";
547
+ const icon = theme.styledSymbol(iconKey, "muted");
548
+
549
+ // Format the status message based on operation type
550
+ const parts: string[] = [];
551
+
552
+ // Error handling
553
+ if (data.error) {
554
+ return `${icon} ${theme.fg("warning", op)}: ${theme.fg("dim", String(data.error))}`;
555
+ }
556
+
557
+ // Build description based on common fields
558
+ switch (op) {
559
+ case "read":
560
+ parts.push(`${data.chars} chars`);
561
+ if (data.path) parts.push(`from ${shortenPath(String(data.path))}`);
562
+ break;
563
+ case "write":
564
+ case "append":
565
+ parts.push(`${data.chars} chars`);
566
+ if (data.path) parts.push(`to ${shortenPath(String(data.path))}`);
567
+ break;
568
+ case "cat":
569
+ parts.push(`${data.files} file${(data.files as number) !== 1 ? "s" : ""}`);
570
+ parts.push(`${data.chars} chars`);
571
+ break;
572
+ case "find":
573
+ case "glob":
574
+ parts.push(`${data.count} match${(data.count as number) !== 1 ? "es" : ""}`);
575
+ if (data.pattern) parts.push(`for "${truncateToWidth(String(data.pattern), 20)}"`);
576
+ break;
577
+ case "grep":
578
+ parts.push(`${data.count} match${(data.count as number) !== 1 ? "es" : ""}`);
579
+ if (data.path) parts.push(`in ${shortenPath(String(data.path))}`);
580
+ break;
581
+ case "rgrep":
582
+ parts.push(`${data.count} match${(data.count as number) !== 1 ? "es" : ""}`);
583
+ if (data.pattern) parts.push(`for "${truncateToWidth(String(data.pattern), 20)}"`);
584
+ break;
585
+ case "ls":
586
+ parts.push(`${data.count} entr${(data.count as number) !== 1 ? "ies" : "y"}`);
587
+ break;
588
+ case "env":
589
+ if (data.action === "set") {
590
+ parts.push(`set ${data.key}=${truncateToWidth(String(data.value ?? ""), 30)}`);
591
+ } else if (data.action === "get") {
592
+ parts.push(`${data.key}=${truncateToWidth(String(data.value ?? ""), 30)}`);
593
+ } else {
594
+ parts.push(`${data.count} variable${(data.count as number) !== 1 ? "s" : ""}`);
595
+ }
596
+ break;
597
+ case "stat":
598
+ if (data.is_dir) {
599
+ parts.push("directory");
600
+ } else {
601
+ parts.push(`${data.size} bytes`);
602
+ }
603
+ if (data.path) parts.push(shortenPath(String(data.path)));
604
+ break;
605
+ case "replace":
606
+ case "sed":
607
+ parts.push(`${data.count} replacement${(data.count as number) !== 1 ? "s" : ""}`);
608
+ if (data.path) parts.push(`in ${shortenPath(String(data.path))}`);
609
+ break;
610
+ case "rsed":
611
+ parts.push(`${data.count} replacement${(data.count as number) !== 1 ? "s" : ""}`);
612
+ if (data.files) parts.push(`in ${data.files} file${(data.files as number) !== 1 ? "s" : ""}`);
613
+ break;
614
+ case "git_status":
615
+ if (data.clean) {
616
+ parts.push("clean");
617
+ } else {
618
+ const statusParts: string[] = [];
619
+ if (data.staged) statusParts.push(`${data.staged} staged`);
620
+ if (data.modified) statusParts.push(`${data.modified} modified`);
621
+ if (data.untracked) statusParts.push(`${data.untracked} untracked`);
622
+ parts.push(statusParts.join(", ") || "unknown");
623
+ }
624
+ if (data.branch) parts.push(`on ${data.branch}`);
625
+ break;
626
+ case "git_log":
627
+ parts.push(`${data.commits} commit${(data.commits as number) !== 1 ? "s" : ""}`);
628
+ break;
629
+ case "git_diff":
630
+ parts.push(`${data.lines} line${(data.lines as number) !== 1 ? "s" : ""}`);
631
+ if (data.staged) parts.push("(staged)");
632
+ break;
633
+ case "diff":
634
+ if (data.identical) {
635
+ parts.push("files identical");
636
+ } else {
637
+ parts.push("files differ");
638
+ }
639
+ break;
640
+ case "batch":
641
+ parts.push(`${data.files} file${(data.files as number) !== 1 ? "s" : ""} processed`);
642
+ break;
643
+ case "wc":
644
+ parts.push(`${data.lines}L ${data.words}W ${data.chars}C`);
645
+ break;
646
+ case "lines":
647
+ parts.push(`${data.count} line${(data.count as number) !== 1 ? "s" : ""}`);
648
+ if (data.start && data.end) parts.push(`(${data.start}-${data.end})`);
649
+ break;
650
+ case "delete_lines":
651
+ case "delete_matching":
652
+ parts.push(`${data.count} line${(data.count as number) !== 1 ? "s" : ""} deleted`);
653
+ break;
654
+ case "insert_at":
655
+ parts.push(`${data.lines_inserted} line${(data.lines_inserted as number) !== 1 ? "s" : ""} inserted`);
656
+ break;
657
+ case "cd":
658
+ case "pwd":
659
+ case "mkdir":
660
+ case "touch":
661
+ if (data.path) parts.push(shortenPath(String(data.path)));
662
+ break;
663
+ case "rm":
664
+ case "mv":
665
+ case "cp":
666
+ if (data.src) parts.push(`${shortenPath(String(data.src))} → ${shortenPath(String(data.dst))}`);
667
+ else if (data.path) parts.push(shortenPath(String(data.path)));
668
+ break;
669
+ default:
670
+ // Generic formatting for other operations
671
+ if (data.count !== undefined) {
672
+ parts.push(String(data.count));
673
+ }
674
+ if (data.path) {
675
+ parts.push(shortenPath(String(data.path)));
676
+ }
677
+ }
678
+
679
+ const desc = parts.length > 0 ? parts.join(" · ") : "";
680
+ return `${icon} ${theme.fg("muted", op)}${desc ? ` ${theme.fg("dim", desc)}` : ""}`;
681
+ }
682
+
683
+ /** Format status event with expanded detail lines. */
684
+ function formatStatusEventExpanded(event: PythonStatusEvent, theme: Theme): string[] {
685
+ const lines: string[] = [];
686
+ const { op, ...data } = event;
687
+
688
+ // Main status line
689
+ lines.push(formatStatusEvent(event, theme));
690
+
691
+ // Add detail lines for operations with list data
692
+ const addItems = (items: unknown[], formatter: (item: unknown) => string, max = 5) => {
693
+ const arr = Array.isArray(items) ? items : [];
694
+ for (let i = 0; i < Math.min(arr.length, max); i++) {
695
+ lines.push(` ${theme.fg("dim", formatter(arr[i]))}`);
696
+ }
697
+ if (arr.length > max) {
698
+ lines.push(` ${theme.fg("dim", `… ${arr.length - max} more`)}`);
699
+ }
700
+ };
701
+
702
+ // Add preview lines (truncated content)
703
+ const addPreview = (preview: string, maxLines = 3) => {
704
+ const previewLines = String(preview).split("\n").slice(0, maxLines);
705
+ for (const line of previewLines) {
706
+ lines.push(` ${theme.fg("toolOutput", truncateToWidth(replaceTabs(line), 80))}`);
707
+ }
708
+ const totalLines = String(preview).split("\n").length;
709
+ if (totalLines > maxLines) {
710
+ lines.push(` ${theme.fg("dim", `… ${totalLines - maxLines} more lines`)}`);
711
+ }
712
+ };
713
+
714
+ switch (op) {
715
+ case "find":
716
+ case "glob":
717
+ if (data.matches) addItems(data.matches as unknown[], m => String(m));
718
+ break;
719
+ case "ls":
720
+ if (data.items) addItems(data.items as unknown[], m => String(m));
721
+ break;
722
+ case "grep":
723
+ if (data.hits) {
724
+ addItems(data.hits as unknown[], h => {
725
+ const hit = h as { line: number; text: string };
726
+ return `${hit.line}: ${truncateToWidth(hit.text, 60)}`;
727
+ });
728
+ }
729
+ break;
730
+ case "rgrep":
731
+ if (data.hits) {
732
+ addItems(data.hits as unknown[], h => {
733
+ const hit = h as { file: string; line: number; text: string };
734
+ return `${shortenPath(hit.file)}:${hit.line}: ${truncateToWidth(hit.text, 50)}`;
735
+ });
736
+ }
737
+ break;
738
+ case "rsed":
739
+ if (data.changed) {
740
+ addItems(data.changed as unknown[], c => {
741
+ const change = c as { file: string; count: number };
742
+ return `${shortenPath(change.file)}: ${change.count} replacement${change.count !== 1 ? "s" : ""}`;
743
+ });
744
+ }
745
+ break;
746
+ case "env":
747
+ if (data.keys) addItems(data.keys as unknown[], k => String(k), 10);
748
+ break;
749
+ case "git_log":
750
+ if (data.entries) {
751
+ addItems(data.entries as unknown[], e => {
752
+ const entry = e as { sha: string; subject: string };
753
+ return `${entry.sha} ${truncateToWidth(entry.subject, 50)}`;
754
+ });
755
+ }
756
+ break;
757
+ case "git_status":
758
+ if (data.files) addItems(data.files as unknown[], f => String(f));
759
+ break;
760
+ case "git_branch":
761
+ if (data.branches) addItems(data.branches as unknown[], b => String(b));
762
+ break;
763
+ case "read":
764
+ case "cat":
765
+ case "head":
766
+ case "tail":
767
+ case "tree":
768
+ case "diff":
769
+ case "lines":
770
+ case "git_diff":
771
+ case "sh":
772
+ if (data.preview) addPreview(String(data.preview));
773
+ break;
774
+ }
775
+
776
+ return lines;
777
+ }
778
+
779
+ /** Render status events as tree lines. */
780
+ function renderStatusEvents(events: PythonStatusEvent[], theme: Theme, expanded: boolean): string[] {
781
+ if (events.length === 0) return [];
782
+
783
+ const maxCollapsed = 3;
784
+ const maxExpanded = 10;
785
+ const displayCount = expanded ? Math.min(events.length, maxExpanded) : Math.min(events.length, maxCollapsed);
786
+
787
+ const lines: string[] = [];
788
+ for (let i = 0; i < displayCount; i++) {
789
+ const isLast = i === displayCount - 1 && (expanded || events.length <= maxCollapsed);
790
+ const branch = isLast ? theme.tree.last : theme.tree.branch;
791
+
792
+ if (expanded) {
793
+ // Show expanded details for each event
794
+ const eventLines = formatStatusEventExpanded(events[i], theme);
795
+ lines.push(`${theme.fg("dim", branch)} ${eventLines[0]}`);
796
+ const continueBranch = isLast ? " " : `${theme.tree.vertical} `;
797
+ for (let j = 1; j < eventLines.length; j++) {
798
+ lines.push(`${theme.fg("dim", continueBranch)}${eventLines[j]}`);
799
+ }
800
+ } else {
801
+ lines.push(`${theme.fg("dim", branch)} ${formatStatusEvent(events[i], theme)}`);
802
+ }
803
+ }
804
+
805
+ if (!expanded && events.length > maxCollapsed) {
806
+ lines.push(`${theme.fg("dim", theme.tree.last)} ${theme.fg("dim", `… ${events.length - maxCollapsed} more`)}`);
807
+ } else if (expanded && events.length > maxExpanded) {
808
+ lines.push(`${theme.fg("dim", theme.tree.last)} ${theme.fg("dim", `… ${events.length - maxExpanded} more`)}`);
809
+ }
810
+
811
+ return lines;
812
+ }
813
+
814
+ function formatCellOutputLines(
815
+ cell: PythonCellResult,
816
+ expanded: boolean,
817
+ previewLines: number,
818
+ theme: Theme,
819
+ ): { lines: string[]; hiddenCount: number } {
820
+ const rawLines = cell.output ? cell.output.split("\n") : [];
821
+ const displayLines = expanded ? rawLines : rawLines.slice(-previewLines);
822
+ const hiddenCount = rawLines.length - displayLines.length;
823
+ const outputLines = displayLines.map(line => {
824
+ const cleaned = replaceTabs(line);
825
+ return cell.status === "error" ? theme.fg("error", cleaned) : theme.fg("toolOutput", cleaned);
826
+ });
827
+
828
+ if (outputLines.length === 0) {
829
+ return { lines: [], hiddenCount: 0 };
830
+ }
831
+
832
+ return { lines: outputLines, hiddenCount };
833
+ }
834
+
835
+ export const pythonToolRenderer = {
836
+ renderCall(args: PythonRenderArgs, _options: RenderResultOptions, uiTheme: Theme): Component {
837
+ const ui = new ToolUIKit(uiTheme);
838
+ const cells = args.cells ?? [];
839
+ const cwd = getProjectDir();
840
+ let displayWorkdir = args.cwd;
841
+
842
+ if (displayWorkdir) {
843
+ const resolvedCwd = path.resolve(cwd);
844
+ const resolvedWorkdir = path.resolve(displayWorkdir);
845
+ if (resolvedWorkdir === resolvedCwd) {
846
+ displayWorkdir = undefined;
847
+ } else {
848
+ const relativePath = path.relative(resolvedCwd, resolvedWorkdir);
849
+ const isWithinCwd =
850
+ relativePath && !relativePath.startsWith("..") && !relativePath.startsWith(`..${path.sep}`);
851
+ if (isWithinCwd) {
852
+ displayWorkdir = relativePath;
853
+ }
854
+ }
855
+ }
856
+
857
+ const workdirLabel = displayWorkdir ? `cd ${displayWorkdir}` : undefined;
858
+ if (cells.length === 0) {
859
+ const prompt = uiTheme.fg("accent", ">>>");
860
+ const prefix = workdirLabel ? `${uiTheme.fg("dim", `${workdirLabel} && `)}` : "";
861
+ const text = ui.title(`${prompt} ${prefix}…`);
862
+ return new Text(text, 0, 0);
863
+ }
864
+
865
+ // Cache state - cells don't change, only width varies
866
+ let cached: { width: number; result: string[] } | undefined;
867
+
868
+ return {
869
+ render: (width: number): string[] => {
870
+ if (cached && cached.width === width) {
871
+ return cached.result;
872
+ }
873
+
874
+ const lines: string[] = [];
875
+ for (let i = 0; i < cells.length; i++) {
876
+ const cell = cells[i];
877
+ const cellTitle = cell.title;
878
+ const combinedTitle =
879
+ cellTitle && workdirLabel ? `${workdirLabel} · ${cellTitle}` : (cellTitle ?? workdirLabel);
880
+ const cellLines = renderCodeCell(
881
+ {
882
+ code: cell.code,
883
+ language: "python",
884
+ index: i,
885
+ total: cells.length,
886
+ title: combinedTitle,
887
+ status: "pending",
888
+ width,
889
+ codeMaxLines: PYTHON_DEFAULT_PREVIEW_LINES,
890
+ expanded: true,
891
+ },
892
+ uiTheme,
893
+ );
894
+ lines.push(...cellLines);
895
+ if (i < cells.length - 1) {
896
+ lines.push("");
897
+ }
898
+ }
899
+ cached = { width, result: lines };
900
+ return lines;
901
+ },
902
+ invalidate: () => {
903
+ cached = undefined;
904
+ },
905
+ };
906
+ },
907
+
908
+ renderResult(
909
+ result: { content: Array<{ type: string; text?: string }>; details?: PythonToolDetails },
910
+ options: RenderResultOptions & { renderContext?: PythonRenderContext },
911
+ uiTheme: Theme,
912
+ ): Component {
913
+ const ui = new ToolUIKit(uiTheme);
914
+ const details = result.details;
915
+
916
+ const output =
917
+ options.renderContext?.output ?? (result.content?.find(c => c.type === "text")?.text ?? "").trimEnd();
918
+
919
+ const jsonOutputs = details?.jsonOutputs ?? [];
920
+ const jsonLines = jsonOutputs.flatMap((value, index) => {
921
+ const header = `JSON output ${index + 1}`;
922
+ const treeLines = renderJsonTree(value, uiTheme, options.renderContext?.expanded ?? options.expanded);
923
+ return [header, ...treeLines];
924
+ });
925
+
926
+ const truncation = details?.meta?.truncation;
927
+ const timeoutSeconds = options.renderContext?.timeout;
928
+ const timeoutLine =
929
+ typeof timeoutSeconds === "number"
930
+ ? uiTheme.fg("dim", ui.wrapBrackets(`Timeout: ${timeoutSeconds}s`))
931
+ : undefined;
932
+ let warningLine: string | undefined;
933
+ if (truncation) {
934
+ const warnings: string[] = [];
935
+ if (truncation.artifactId) {
936
+ warnings.push(`Full output: artifact://${truncation.artifactId}`);
937
+ }
938
+ if (truncation.truncatedBy === "lines") {
939
+ warnings.push(`Truncated: showing ${truncation.outputLines} of ${truncation.totalLines} lines`);
940
+ } else {
941
+ warnings.push(
942
+ `Truncated: ${truncation.outputLines} lines shown (${ui.formatBytes(truncation.outputBytes)} limit)`,
943
+ );
944
+ }
945
+ if (warnings.length > 0) {
946
+ warningLine = uiTheme.fg("warning", ui.wrapBrackets(warnings.join(". ")));
947
+ }
948
+ }
949
+
950
+ const cellResults = details?.cells;
951
+ if (cellResults && cellResults.length > 0) {
952
+ // Cache state following Box pattern
953
+ let cached: { key: string; width: number; result: string[] } | undefined;
954
+
955
+ return {
956
+ render: (width: number): string[] => {
957
+ // Read mutable state at render time
958
+ const expanded = options.renderContext?.expanded ?? options.expanded;
959
+ const previewLines = options.renderContext?.previewLines ?? PYTHON_DEFAULT_PREVIEW_LINES;
960
+ const key = `${expanded}|${previewLines}|${options.spinnerFrame}`;
961
+ if (cached && cached.key === key && cached.width === width) {
962
+ return cached.result;
963
+ }
964
+
965
+ const lines: string[] = [];
966
+ for (let i = 0; i < cellResults.length; i++) {
967
+ const cell = cellResults[i];
968
+ const statusLines = renderStatusEvents(cell.statusEvents ?? [], uiTheme, expanded);
969
+ const outputContent = formatCellOutputLines(cell, expanded, previewLines, uiTheme);
970
+ const outputLines = [...outputContent.lines];
971
+ if (!expanded && outputContent.hiddenCount > 0) {
972
+ outputLines.push(
973
+ uiTheme.fg("dim", `… ${outputContent.hiddenCount} more lines (ctrl+o to expand)`),
974
+ );
975
+ }
976
+ if (statusLines.length > 0) {
977
+ if (outputLines.length > 0) {
978
+ outputLines.push(uiTheme.fg("dim", "Status"));
979
+ }
980
+ outputLines.push(...statusLines);
981
+ }
982
+ const cellLines = renderCodeCell(
983
+ {
984
+ code: cell.code,
985
+ language: "python",
986
+ index: i,
987
+ total: cellResults.length,
988
+ title: cell.title,
989
+ status: cell.status,
990
+ spinnerFrame: options.spinnerFrame,
991
+ duration: cell.durationMs,
992
+ output: outputLines.length > 0 ? outputLines.join("\n") : undefined,
993
+ outputMaxLines: outputLines.length,
994
+ codeMaxLines: expanded ? Number.POSITIVE_INFINITY : PYTHON_DEFAULT_PREVIEW_LINES,
995
+ expanded,
996
+ width,
997
+ },
998
+ uiTheme,
999
+ );
1000
+ lines.push(...cellLines);
1001
+ if (i < cellResults.length - 1) {
1002
+ lines.push("");
1003
+ }
1004
+ }
1005
+ if (jsonLines.length > 0) {
1006
+ if (lines.length > 0) {
1007
+ lines.push("");
1008
+ }
1009
+ lines.push(...jsonLines);
1010
+ }
1011
+ if (timeoutLine) {
1012
+ lines.push(timeoutLine);
1013
+ }
1014
+ if (warningLine) {
1015
+ lines.push(warningLine);
1016
+ }
1017
+ cached = { key, width, result: lines };
1018
+ return lines;
1019
+ },
1020
+ invalidate: () => {
1021
+ cached = undefined;
1022
+ },
1023
+ };
1024
+ }
1025
+
1026
+ const displayOutput = output;
1027
+ const combinedOutput = [displayOutput, ...jsonLines].filter(Boolean).join("\n");
1028
+
1029
+ const statusEvents = details?.statusEvents ?? [];
1030
+ const statusLines = renderStatusEvents(
1031
+ statusEvents,
1032
+ uiTheme,
1033
+ options.renderContext?.expanded ?? options.expanded,
1034
+ );
1035
+
1036
+ if (!combinedOutput && statusLines.length === 0) {
1037
+ const lines = [timeoutLine, warningLine].filter(Boolean) as string[];
1038
+ return new Text(lines.join("\n"), 0, 0);
1039
+ }
1040
+
1041
+ if (!combinedOutput && statusLines.length > 0) {
1042
+ const lines = [uiTheme.fg("dim", "Status"), ...statusLines, timeoutLine, warningLine].filter(
1043
+ Boolean,
1044
+ ) as string[];
1045
+ return new Text(lines.join("\n"), 0, 0);
1046
+ }
1047
+
1048
+ if (options.renderContext?.expanded ?? options.expanded) {
1049
+ const styledOutput = combinedOutput
1050
+ .split("\n")
1051
+ .map(line => uiTheme.fg("toolOutput", line))
1052
+ .join("\n");
1053
+ const lines = [
1054
+ styledOutput,
1055
+ ...(statusLines.length > 0 ? [uiTheme.fg("dim", "Status"), ...statusLines] : []),
1056
+ timeoutLine,
1057
+ warningLine,
1058
+ ].filter(Boolean) as string[];
1059
+ return new Text(lines.join("\n"), 0, 0);
1060
+ }
1061
+
1062
+ const styledOutput = combinedOutput
1063
+ .split("\n")
1064
+ .map(line => uiTheme.fg("toolOutput", line))
1065
+ .join("\n");
1066
+ const textContent = `\n${styledOutput}`;
1067
+
1068
+ let cachedWidth: number | undefined;
1069
+ let cachedLines: string[] | undefined;
1070
+ let cachedSkipped: number | undefined;
1071
+ let cachedPreviewLines: number | undefined;
1072
+
1073
+ return {
1074
+ render: (width: number): string[] => {
1075
+ // Read mutable state at render time
1076
+ const previewLines = options.renderContext?.previewLines ?? PYTHON_DEFAULT_PREVIEW_LINES;
1077
+ if (cachedLines === undefined || cachedWidth !== width || cachedPreviewLines !== previewLines) {
1078
+ const result = truncateToVisualLines(textContent, previewLines, width);
1079
+ cachedLines = result.visualLines;
1080
+ cachedSkipped = result.skippedCount;
1081
+ cachedWidth = width;
1082
+ cachedPreviewLines = previewLines;
1083
+ }
1084
+ const outputLines: string[] = [];
1085
+ if (cachedSkipped && cachedSkipped > 0) {
1086
+ outputLines.push("");
1087
+ const skippedLine = uiTheme.fg(
1088
+ "dim",
1089
+ `… (${cachedSkipped} earlier lines, showing ${cachedLines.length} of ${cachedSkipped + cachedLines.length}) (ctrl+o to expand)`,
1090
+ );
1091
+ outputLines.push(truncateToWidth(skippedLine, width));
1092
+ }
1093
+ outputLines.push(...cachedLines);
1094
+ if (statusLines.length > 0) {
1095
+ outputLines.push(truncateToWidth(uiTheme.fg("dim", "Status"), width));
1096
+ for (const statusLine of statusLines) {
1097
+ outputLines.push(truncateToWidth(statusLine, width));
1098
+ }
1099
+ }
1100
+ if (timeoutLine) {
1101
+ outputLines.push(truncateToWidth(timeoutLine, width));
1102
+ }
1103
+ if (warningLine) {
1104
+ outputLines.push(truncateToWidth(warningLine, width));
1105
+ }
1106
+ return outputLines;
1107
+ },
1108
+ invalidate: () => {
1109
+ cachedWidth = undefined;
1110
+ cachedLines = undefined;
1111
+ cachedSkipped = undefined;
1112
+ cachedPreviewLines = undefined;
1113
+ },
1114
+ };
1115
+ },
1116
+ mergeCallAndResult: true,
1117
+ inline: true,
1118
+ };