@prometheus-ai/agent 0.5.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 (1954) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +36 -0
  3. package/dist/types/async/index.d.ts +2 -0
  4. package/dist/types/async/job-manager.d.ts +88 -0
  5. package/dist/types/async/support.d.ts +2 -0
  6. package/dist/types/auto-thinking/classifier.d.ts +35 -0
  7. package/dist/types/autoresearch/dashboard.d.ts +4 -0
  8. package/dist/types/autoresearch/git.d.ts +36 -0
  9. package/dist/types/autoresearch/helpers.d.ts +24 -0
  10. package/dist/types/autoresearch/index.d.ts +2 -0
  11. package/dist/types/autoresearch/state.d.ts +17 -0
  12. package/dist/types/autoresearch/storage.d.ts +142 -0
  13. package/dist/types/autoresearch/tools/init-experiment.d.ts +31 -0
  14. package/dist/types/autoresearch/tools/log-experiment.d.ts +23 -0
  15. package/dist/types/autoresearch/tools/run-experiment.d.ts +8 -0
  16. package/dist/types/autoresearch/tools/update-notes.d.ts +12 -0
  17. package/dist/types/autoresearch/types.d.ts +154 -0
  18. package/dist/types/capability/context-file.d.ts +30 -0
  19. package/dist/types/capability/extension-module.d.ts +15 -0
  20. package/dist/types/capability/extension.d.ts +28 -0
  21. package/dist/types/capability/fs.d.ts +20 -0
  22. package/dist/types/capability/hook.d.ts +19 -0
  23. package/dist/types/capability/index.d.ts +80 -0
  24. package/dist/types/capability/instruction.d.ts +17 -0
  25. package/dist/types/capability/mcp.d.ts +45 -0
  26. package/dist/types/capability/prompt.d.ts +15 -0
  27. package/dist/types/capability/rule-buckets.d.ts +30 -0
  28. package/dist/types/capability/rule.d.ts +68 -0
  29. package/dist/types/capability/settings.d.ts +15 -0
  30. package/dist/types/capability/skill.d.ts +43 -0
  31. package/dist/types/capability/slash-command.d.ts +17 -0
  32. package/dist/types/capability/ssh.d.ts +23 -0
  33. package/dist/types/capability/system-prompt.d.ts +15 -0
  34. package/dist/types/capability/tool.d.ts +19 -0
  35. package/dist/types/capability/types.d.ts +158 -0
  36. package/dist/types/cli/agents-cli.d.ts +12 -0
  37. package/dist/types/cli/args.d.ts +55 -0
  38. package/dist/types/cli/auth-broker-cli.d.ts +25 -0
  39. package/dist/types/cli/auth-gateway-cli.d.ts +26 -0
  40. package/dist/types/cli/classify-install-target.d.ts +12 -0
  41. package/dist/types/cli/claude-trace-cli.d.ts +52 -0
  42. package/dist/types/cli/commands/init-xdg.d.ts +1 -0
  43. package/dist/types/cli/completion-gen.d.ts +80 -0
  44. package/dist/types/cli/config-cli.d.ts +22 -0
  45. package/dist/types/cli/dry-balance-cli.d.ts +104 -0
  46. package/dist/types/cli/extension-flags.d.ts +36 -0
  47. package/dist/types/cli/file-processor.d.ts +11 -0
  48. package/dist/types/cli/gateway-cli.d.ts +95 -0
  49. package/dist/types/cli/grep-cli.d.ts +17 -0
  50. package/dist/types/cli/grievances-cli.d.ts +37 -0
  51. package/dist/types/cli/initial-message.d.ts +17 -0
  52. package/dist/types/cli/list-models.d.ts +30 -0
  53. package/dist/types/cli/plugin-cli.d.ts +32 -0
  54. package/dist/types/cli/read-cli.d.ts +4 -0
  55. package/dist/types/cli/session-picker.d.ts +10 -0
  56. package/dist/types/cli/setup-cli.d.ts +24 -0
  57. package/dist/types/cli/shell-cli.d.ts +8 -0
  58. package/dist/types/cli/ssh-cli.d.ts +21 -0
  59. package/dist/types/cli/stats-cli.d.ts +17 -0
  60. package/dist/types/cli/tiny-models-cli.d.ts +9 -0
  61. package/dist/types/cli/update-cli.d.ts +106 -0
  62. package/dist/types/cli/web-search-cli.d.ts +20 -0
  63. package/dist/types/cli/worktree-cli.d.ts +26 -0
  64. package/dist/types/cli-commands.d.ts +19 -0
  65. package/dist/types/cli.d.ts +3 -0
  66. package/dist/types/commands/acp.d.ts +12 -0
  67. package/dist/types/commands/agents.d.ts +34 -0
  68. package/dist/types/commands/auth-broker.d.ts +54 -0
  69. package/dist/types/commands/auth-gateway.d.ts +35 -0
  70. package/dist/types/commands/commit.d.ts +27 -0
  71. package/dist/types/commands/complete.d.ts +6 -0
  72. package/dist/types/commands/completions.d.ts +13 -0
  73. package/dist/types/commands/config.d.ts +30 -0
  74. package/dist/types/commands/dry-balance.d.ts +31 -0
  75. package/dist/types/commands/gateway.d.ts +69 -0
  76. package/dist/types/commands/grep.d.ts +42 -0
  77. package/dist/types/commands/grievances.d.ts +40 -0
  78. package/dist/types/commands/install.d.ts +51 -0
  79. package/dist/types/commands/launch.d.ts +129 -0
  80. package/dist/types/commands/plugin.d.ts +52 -0
  81. package/dist/types/commands/read.d.ts +15 -0
  82. package/dist/types/commands/setup.d.ts +34 -0
  83. package/dist/types/commands/shell.d.ts +21 -0
  84. package/dist/types/commands/ssh.d.ts +48 -0
  85. package/dist/types/commands/stats.d.ts +25 -0
  86. package/dist/types/commands/tiny-models.d.ts +22 -0
  87. package/dist/types/commands/update.d.ts +20 -0
  88. package/dist/types/commands/web-search.d.ts +33 -0
  89. package/dist/types/commands/worktree.d.ts +34 -0
  90. package/dist/types/commit/agentic/agent.d.ts +31 -0
  91. package/dist/types/commit/agentic/fallback.d.ts +5 -0
  92. package/dist/types/commit/agentic/index.d.ts +2 -0
  93. package/dist/types/commit/agentic/state.d.ts +58 -0
  94. package/dist/types/commit/agentic/tools/analyze-file.d.ts +19 -0
  95. package/dist/types/commit/agentic/tools/git-file-diff.d.ts +10 -0
  96. package/dist/types/commit/agentic/tools/git-hunk.d.ts +9 -0
  97. package/dist/types/commit/agentic/tools/git-overview.d.ts +9 -0
  98. package/dist/types/commit/agentic/tools/index.d.ts +16 -0
  99. package/dist/types/commit/agentic/tools/propose-changelog.d.ts +28 -0
  100. package/dist/types/commit/agentic/tools/propose-commit.d.ts +36 -0
  101. package/dist/types/commit/agentic/tools/recent-commits.d.ts +7 -0
  102. package/dist/types/commit/agentic/tools/schemas.d.ts +27 -0
  103. package/dist/types/commit/agentic/tools/split-commit.d.ts +53 -0
  104. package/dist/types/commit/agentic/topo-sort.d.ts +4 -0
  105. package/dist/types/commit/agentic/trivial.d.ts +7 -0
  106. package/dist/types/commit/agentic/validation.d.ts +20 -0
  107. package/dist/types/commit/analysis/conventional.d.ts +22 -0
  108. package/dist/types/commit/analysis/index.d.ts +4 -0
  109. package/dist/types/commit/analysis/scope.d.ts +6 -0
  110. package/dist/types/commit/analysis/summary.d.ts +19 -0
  111. package/dist/types/commit/analysis/validation.d.ts +8 -0
  112. package/dist/types/commit/changelog/detect.d.ts +2 -0
  113. package/dist/types/commit/changelog/generate.d.ts +30 -0
  114. package/dist/types/commit/changelog/index.d.ts +30 -0
  115. package/dist/types/commit/changelog/parse.d.ts +2 -0
  116. package/dist/types/commit/cli.d.ts +3 -0
  117. package/dist/types/commit/git/diff.d.ts +5 -0
  118. package/dist/types/commit/index.d.ts +4 -0
  119. package/dist/types/commit/map-reduce/index.d.ts +28 -0
  120. package/dist/types/commit/map-reduce/map-phase.d.ts +17 -0
  121. package/dist/types/commit/map-reduce/reduce-phase.d.ts +13 -0
  122. package/dist/types/commit/map-reduce/utils.d.ts +2 -0
  123. package/dist/types/commit/message.d.ts +2 -0
  124. package/dist/types/commit/model-selection.d.ts +15 -0
  125. package/dist/types/commit/pipeline.d.ts +5 -0
  126. package/dist/types/commit/shared-llm.d.ts +54 -0
  127. package/dist/types/commit/types.d.ts +78 -0
  128. package/dist/types/commit/utils/exclusions.d.ts +4 -0
  129. package/dist/types/commit/utils.d.ts +20 -0
  130. package/dist/types/config/append-only-context-mode.d.ts +8 -0
  131. package/dist/types/config/config-file.d.ts +62 -0
  132. package/dist/types/config/file-lock.d.ts +29 -0
  133. package/dist/types/config/keybindings.d.ts +352 -0
  134. package/dist/types/config/model-equivalence.d.ts +24 -0
  135. package/dist/types/config/model-id-affixes.d.ts +10 -0
  136. package/dist/types/config/model-registry.d.ts +400 -0
  137. package/dist/types/config/model-resolver.d.ts +230 -0
  138. package/dist/types/config/models-config-schema.d.ts +523 -0
  139. package/dist/types/config/prompt-templates.d.ts +32 -0
  140. package/dist/types/config/resolve-config-value.d.ts +17 -0
  141. package/dist/types/config/settings-schema.d.ts +4352 -0
  142. package/dist/types/config/settings.d.ts +156 -0
  143. package/dist/types/config.d.ts +92 -0
  144. package/dist/types/cursor.d.ts +22 -0
  145. package/dist/types/dap/client.d.ts +38 -0
  146. package/dist/types/dap/config.d.ts +6 -0
  147. package/dist/types/dap/index.d.ts +4 -0
  148. package/dist/types/dap/session.d.ts +108 -0
  149. package/dist/types/dap/types.d.ts +524 -0
  150. package/dist/types/debug/index.d.ts +15 -0
  151. package/dist/types/debug/log-formatting.d.ts +4 -0
  152. package/dist/types/debug/log-viewer.d.ts +68 -0
  153. package/dist/types/debug/profiler.d.ts +24 -0
  154. package/dist/types/debug/protocol-probe.d.ts +38 -0
  155. package/dist/types/debug/raw-sse-buffer.d.ts +44 -0
  156. package/dist/types/debug/raw-sse.d.ts +18 -0
  157. package/dist/types/debug/report-bundle.d.ts +55 -0
  158. package/dist/types/debug/system-info.d.ts +26 -0
  159. package/dist/types/debug/terminal-info.d.ts +34 -0
  160. package/dist/types/discovery/agents-md.d.ts +1 -0
  161. package/dist/types/discovery/agents.d.ts +12 -0
  162. package/dist/types/discovery/builtin-defaults.d.ts +1 -0
  163. package/dist/types/discovery/builtin-rules/index.d.ts +7 -0
  164. package/dist/types/discovery/builtin.d.ts +1 -0
  165. package/dist/types/discovery/claude-plugins.d.ts +1 -0
  166. package/dist/types/discovery/claude.d.ts +1 -0
  167. package/dist/types/discovery/cline.d.ts +1 -0
  168. package/dist/types/discovery/codex.d.ts +1 -0
  169. package/dist/types/discovery/context-files.d.ts +17 -0
  170. package/dist/types/discovery/cursor.d.ts +16 -0
  171. package/dist/types/discovery/gemini.d.ts +1 -0
  172. package/dist/types/discovery/github.d.ts +1 -0
  173. package/dist/types/discovery/helpers.d.ts +269 -0
  174. package/dist/types/discovery/index.d.ts +53 -0
  175. package/dist/types/discovery/mcp-json.d.ts +1 -0
  176. package/dist/types/discovery/opencode.d.ts +1 -0
  177. package/dist/types/discovery/plugin-dir-roots.d.ts +15 -0
  178. package/dist/types/discovery/prometheus-extension-roots.d.ts +43 -0
  179. package/dist/types/discovery/prometheus-plugins.d.ts +1 -0
  180. package/dist/types/discovery/ssh.d.ts +1 -0
  181. package/dist/types/discovery/substitute-plugin-root.d.ts +1 -0
  182. package/dist/types/discovery/vscode.d.ts +1 -0
  183. package/dist/types/discovery/windsurf.d.ts +13 -0
  184. package/dist/types/edit/apply-patch/index.d.ts +35 -0
  185. package/dist/types/edit/apply-patch/parser.d.ts +34 -0
  186. package/dist/types/edit/diff.d.ts +59 -0
  187. package/dist/types/edit/file-snapshot-store.d.ts +40 -0
  188. package/dist/types/edit/hashline/block-resolver.d.ts +9 -0
  189. package/dist/types/edit/hashline/diff.d.ts +40 -0
  190. package/dist/types/edit/hashline/execute.d.ts +28 -0
  191. package/dist/types/edit/hashline/filesystem.d.ts +57 -0
  192. package/dist/types/edit/hashline/index.d.ts +5 -0
  193. package/dist/types/edit/hashline/params.d.ts +11 -0
  194. package/dist/types/edit/index.d.ts +62 -0
  195. package/dist/types/edit/modes/apply-patch.d.ts +24 -0
  196. package/dist/types/edit/modes/patch.d.ts +99 -0
  197. package/dist/types/edit/modes/replace.d.ts +142 -0
  198. package/dist/types/edit/normalize.d.ts +31 -0
  199. package/dist/types/edit/notebook.d.ts +23 -0
  200. package/dist/types/edit/read-file.d.ts +2 -0
  201. package/dist/types/edit/renderer.d.ts +109 -0
  202. package/dist/types/edit/streaming.d.ts +75 -0
  203. package/dist/types/eval/__tests__/agent-bridge.test.d.ts +1 -0
  204. package/dist/types/eval/__tests__/bridge-timeout.test.d.ts +1 -0
  205. package/dist/types/eval/__tests__/budget-bridge.test.d.ts +1 -0
  206. package/dist/types/eval/__tests__/idle-timeout.test.d.ts +1 -0
  207. package/dist/types/eval/__tests__/kernel-spawn.test.d.ts +1 -0
  208. package/dist/types/eval/__tests__/llm-bridge.test.d.ts +1 -0
  209. package/dist/types/eval/__tests__/shared-executors.test.d.ts +1 -0
  210. package/dist/types/eval/agent-bridge.d.ts +25 -0
  211. package/dist/types/eval/backend.d.ts +55 -0
  212. package/dist/types/eval/bridge-timeout.d.ts +27 -0
  213. package/dist/types/eval/budget-bridge.d.ts +29 -0
  214. package/dist/types/eval/concurrency-bridge.d.ts +26 -0
  215. package/dist/types/eval/idle-timeout.d.ts +30 -0
  216. package/dist/types/eval/index.d.ts +4 -0
  217. package/dist/types/eval/js/context-manager.d.ts +24 -0
  218. package/dist/types/eval/js/executor.d.ts +36 -0
  219. package/dist/types/eval/js/index.d.ts +10 -0
  220. package/dist/types/eval/js/shared/helpers.d.ts +38 -0
  221. package/dist/types/eval/js/shared/indirect-eval.d.ts +14 -0
  222. package/dist/types/eval/js/shared/local-module-loader.d.ts +16 -0
  223. package/dist/types/eval/js/shared/prelude.d.ts +1 -0
  224. package/dist/types/eval/js/shared/rewrite-imports.d.ts +25 -0
  225. package/dist/types/eval/js/shared/runtime.d.ts +53 -0
  226. package/dist/types/eval/js/shared/types.d.ts +24 -0
  227. package/dist/types/eval/js/tool-bridge.d.ts +20 -0
  228. package/dist/types/eval/js/worker-core.d.ts +5 -0
  229. package/dist/types/eval/js/worker-entry.d.ts +1 -0
  230. package/dist/types/eval/js/worker-protocol.d.ts +77 -0
  231. package/dist/types/eval/llm-bridge.d.ts +25 -0
  232. package/dist/types/eval/py/display.d.ts +25 -0
  233. package/dist/types/eval/py/executor.d.ts +95 -0
  234. package/dist/types/eval/py/index.d.ts +11 -0
  235. package/dist/types/eval/py/kernel.d.ts +70 -0
  236. package/dist/types/eval/py/prelude.d.ts +1 -0
  237. package/dist/types/eval/py/runtime.d.ts +31 -0
  238. package/dist/types/eval/py/spawn-options.d.ts +58 -0
  239. package/dist/types/eval/py/tool-bridge.d.ts +16 -0
  240. package/dist/types/eval/session-id.d.ts +3 -0
  241. package/dist/types/eval/types.d.ts +52 -0
  242. package/dist/types/exa/factory.d.ts +13 -0
  243. package/dist/types/exa/index.d.ts +20 -0
  244. package/dist/types/exa/mcp-client.d.ts +46 -0
  245. package/dist/types/exa/render.d.ts +19 -0
  246. package/dist/types/exa/researcher.d.ts +9 -0
  247. package/dist/types/exa/search.d.ts +9 -0
  248. package/dist/types/exa/types.d.ts +155 -0
  249. package/dist/types/exa/websets.d.ts +9 -0
  250. package/dist/types/exec/bash-executor.d.ts +42 -0
  251. package/dist/types/exec/exec.d.ts +25 -0
  252. package/dist/types/exec/idle-timeout-watchdog.d.ts +18 -0
  253. package/dist/types/exec/non-interactive-env.d.ts +1 -0
  254. package/dist/types/export/custom-share.d.ts +20 -0
  255. package/dist/types/export/html/index.d.ts +10 -0
  256. package/dist/types/export/html/template.generated.d.ts +1 -0
  257. package/dist/types/export/html/template.macro.d.ts +5 -0
  258. package/dist/types/export/ttsr.d.ts +53 -0
  259. package/dist/types/extensibility/custom-commands/bundled/ci-green/index.d.ts +9 -0
  260. package/dist/types/extensibility/custom-commands/bundled/review/index.d.ts +10 -0
  261. package/dist/types/extensibility/custom-commands/index.d.ts +2 -0
  262. package/dist/types/extensibility/custom-commands/loader.d.ts +29 -0
  263. package/dist/types/extensibility/custom-commands/types.d.ts +106 -0
  264. package/dist/types/extensibility/custom-tools/index.d.ts +6 -0
  265. package/dist/types/extensibility/custom-tools/loader.d.ts +69 -0
  266. package/dist/types/extensibility/custom-tools/types.d.ts +232 -0
  267. package/dist/types/extensibility/custom-tools/wrapper.d.ts +23 -0
  268. package/dist/types/extensibility/extensions/compact-handler.d.ts +26 -0
  269. package/dist/types/extensibility/extensions/get-commands-handler.d.ts +29 -0
  270. package/dist/types/extensibility/extensions/index.d.ts +8 -0
  271. package/dist/types/extensibility/extensions/loader.d.ts +43 -0
  272. package/dist/types/extensibility/extensions/runner.d.ts +142 -0
  273. package/dist/types/extensibility/extensions/types.d.ts +892 -0
  274. package/dist/types/extensibility/extensions/wrapper.d.ts +54 -0
  275. package/dist/types/extensibility/hooks/index.d.ts +5 -0
  276. package/dist/types/extensibility/hooks/loader.d.ts +89 -0
  277. package/dist/types/extensibility/hooks/runner.d.ts +126 -0
  278. package/dist/types/extensibility/hooks/tool-wrapper.d.ts +25 -0
  279. package/dist/types/extensibility/hooks/types.d.ts +435 -0
  280. package/dist/types/extensibility/legacy-pi-ai-shim.d.ts +23 -0
  281. package/dist/types/extensibility/legacy-pi-coding-agent-shim.d.ts +14 -0
  282. package/dist/types/extensibility/plugins/doctor.d.ts +3 -0
  283. package/dist/types/extensibility/plugins/git-url.d.ts +44 -0
  284. package/dist/types/extensibility/plugins/index.d.ts +7 -0
  285. package/dist/types/extensibility/plugins/installer.d.ts +5 -0
  286. package/dist/types/extensibility/plugins/legacy-pi-compat.d.ts +29 -0
  287. package/dist/types/extensibility/plugins/loader.d.ts +41 -0
  288. package/dist/types/extensibility/plugins/manager.d.ts +76 -0
  289. package/dist/types/extensibility/plugins/marketplace/cache.d.ts +41 -0
  290. package/dist/types/extensibility/plugins/marketplace/fetcher.d.ts +48 -0
  291. package/dist/types/extensibility/plugins/marketplace/index.d.ts +6 -0
  292. package/dist/types/extensibility/plugins/marketplace/manager.d.ts +57 -0
  293. package/dist/types/extensibility/plugins/marketplace/registry.d.ts +34 -0
  294. package/dist/types/extensibility/plugins/marketplace/source-resolver.d.ts +28 -0
  295. package/dist/types/extensibility/plugins/marketplace/types.d.ts +130 -0
  296. package/dist/types/extensibility/plugins/parser.d.ts +52 -0
  297. package/dist/types/extensibility/plugins/types.d.ts +149 -0
  298. package/dist/types/extensibility/shared-events.d.ts +279 -0
  299. package/dist/types/extensibility/skills.d.ts +57 -0
  300. package/dist/types/extensibility/slash-commands.d.ts +48 -0
  301. package/dist/types/extensibility/tool-proxy.d.ts +4 -0
  302. package/dist/types/extensibility/typebox.d.ts +155 -0
  303. package/dist/types/extensibility/utils.d.ts +12 -0
  304. package/dist/types/gateway/adapters/telegram/access.d.ts +11 -0
  305. package/dist/types/gateway/adapters/telegram/commands.d.ts +41 -0
  306. package/dist/types/gateway/adapters/telegram/expressions.d.ts +60 -0
  307. package/dist/types/gateway/adapters/telegram/media.d.ts +58 -0
  308. package/dist/types/gateway/adapters/telegram/native-command-policy.d.ts +6 -0
  309. package/dist/types/gateway/adapters/telegram/native-commands.d.ts +23 -0
  310. package/dist/types/gateway/adapters/telegram/normalize.d.ts +8 -0
  311. package/dist/types/gateway/adapters/telegram/send-message.d.ts +388 -0
  312. package/dist/types/gateway/adapters/telegram/service.d.ts +65 -0
  313. package/dist/types/gateway/adapters/telegram/setup-api.d.ts +39 -0
  314. package/dist/types/gateway/adapters/telegram/setup-config.d.ts +11 -0
  315. package/dist/types/gateway/adapters/telegram/types.d.ts +187 -0
  316. package/dist/types/gateway/adapters/telegram/webhook.d.ts +53 -0
  317. package/dist/types/gateway/commands.d.ts +33 -0
  318. package/dist/types/gateway/context.d.ts +30 -0
  319. package/dist/types/gateway/native-slash-policy.d.ts +12 -0
  320. package/dist/types/gateway/prometheus-dispatch.d.ts +29 -0
  321. package/dist/types/gateway/router.d.ts +52 -0
  322. package/dist/types/gateway/session-key.d.ts +3 -0
  323. package/dist/types/gateway/types.d.ts +98 -0
  324. package/dist/types/goals/index.d.ts +3 -0
  325. package/dist/types/goals/runtime.d.ts +67 -0
  326. package/dist/types/goals/state.d.ts +35 -0
  327. package/dist/types/goals/tools/goal-tool.d.ts +66 -0
  328. package/dist/types/hindsight/backend.d.ts +13 -0
  329. package/dist/types/hindsight/bank.d.ts +62 -0
  330. package/dist/types/hindsight/client.d.ts +224 -0
  331. package/dist/types/hindsight/config.d.ts +51 -0
  332. package/dist/types/hindsight/content.d.ts +70 -0
  333. package/dist/types/hindsight/index.d.ts +8 -0
  334. package/dist/types/hindsight/mental-models.d.ts +125 -0
  335. package/dist/types/hindsight/state.d.ts +105 -0
  336. package/dist/types/hindsight/transcript.d.ts +28 -0
  337. package/dist/types/index.d.ts +36 -0
  338. package/dist/types/internal-urls/agent-protocol.d.ts +13 -0
  339. package/dist/types/internal-urls/artifact-protocol.d.ts +7 -0
  340. package/dist/types/internal-urls/docs-index.generated.d.ts +2 -0
  341. package/dist/types/internal-urls/index.d.ts +23 -0
  342. package/dist/types/internal-urls/issue-pr-protocol.d.ts +17 -0
  343. package/dist/types/internal-urls/json-query.d.ts +30 -0
  344. package/dist/types/internal-urls/local-protocol.d.ts +50 -0
  345. package/dist/types/internal-urls/mcp-protocol.d.ts +12 -0
  346. package/dist/types/internal-urls/memory-protocol.d.ts +24 -0
  347. package/dist/types/internal-urls/parse.d.ts +19 -0
  348. package/dist/types/internal-urls/prometheus-protocol.d.ts +13 -0
  349. package/dist/types/internal-urls/registry-helpers.d.ts +10 -0
  350. package/dist/types/internal-urls/router.d.ts +21 -0
  351. package/dist/types/internal-urls/rule-protocol.d.ts +7 -0
  352. package/dist/types/internal-urls/skill-protocol.d.ts +14 -0
  353. package/dist/types/internal-urls/types.d.ts +145 -0
  354. package/dist/types/internal-urls/vault-protocol.d.ts +93 -0
  355. package/dist/types/lib/xai-http.d.ts +40 -0
  356. package/dist/types/lsp/client.d.ts +68 -0
  357. package/dist/types/lsp/clients/biome-client.d.ts +16 -0
  358. package/dist/types/lsp/clients/index.d.ts +19 -0
  359. package/dist/types/lsp/clients/lsp-linter-client.d.ts +16 -0
  360. package/dist/types/lsp/clients/swiftlint-client.d.ts +20 -0
  361. package/dist/types/lsp/config.d.ts +65 -0
  362. package/dist/types/lsp/diagnostics-ledger.d.ts +10 -0
  363. package/dist/types/lsp/edits.d.ts +23 -0
  364. package/dist/types/lsp/index.d.ts +140 -0
  365. package/dist/types/lsp/lspmux.d.ts +58 -0
  366. package/dist/types/lsp/render.d.ts +32 -0
  367. package/dist/types/lsp/startup-events.d.ts +11 -0
  368. package/dist/types/lsp/types.d.ts +302 -0
  369. package/dist/types/lsp/utils.d.ts +112 -0
  370. package/dist/types/main.d.ts +56 -0
  371. package/dist/types/mcp/client.d.ts +75 -0
  372. package/dist/types/mcp/config-writer.d.ts +53 -0
  373. package/dist/types/mcp/config.d.ts +75 -0
  374. package/dist/types/mcp/index.d.ts +18 -0
  375. package/dist/types/mcp/json-rpc.d.ts +27 -0
  376. package/dist/types/mcp/loader.d.ts +43 -0
  377. package/dist/types/mcp/manager.d.ts +202 -0
  378. package/dist/types/mcp/oauth-discovery.d.ts +41 -0
  379. package/dist/types/mcp/oauth-flow.d.ts +59 -0
  380. package/dist/types/mcp/render.d.ts +25 -0
  381. package/dist/types/mcp/smithery-auth.d.ts +16 -0
  382. package/dist/types/mcp/smithery-connect.d.ts +38 -0
  383. package/dist/types/mcp/smithery-registry.d.ts +51 -0
  384. package/dist/types/mcp/timeout.d.ts +9 -0
  385. package/dist/types/mcp/tool-bridge.d.ts +90 -0
  386. package/dist/types/mcp/tool-cache.d.ts +8 -0
  387. package/dist/types/mcp/transports/http.d.ts +45 -0
  388. package/dist/types/mcp/transports/index.d.ts +5 -0
  389. package/dist/types/mcp/transports/stdio.d.ts +60 -0
  390. package/dist/types/mcp/types.d.ts +333 -0
  391. package/dist/types/memories/index.d.ts +43 -0
  392. package/dist/types/memories/storage.d.ts +120 -0
  393. package/dist/types/memory-backend/index.d.ts +5 -0
  394. package/dist/types/memory-backend/local-backend.d.ts +9 -0
  395. package/dist/types/memory-backend/off-backend.d.ts +7 -0
  396. package/dist/types/memory-backend/resolve.d.ts +16 -0
  397. package/dist/types/memory-backend/types.d.ts +67 -0
  398. package/dist/types/mnemopi/backend.d.ts +4 -0
  399. package/dist/types/mnemopi/config.d.ts +29 -0
  400. package/dist/types/mnemopi/index.d.ts +3 -0
  401. package/dist/types/mnemopi/state.d.ts +72 -0
  402. package/dist/types/modes/acp/acp-agent.d.ts +61 -0
  403. package/dist/types/modes/acp/acp-client-bridge.d.ts +9 -0
  404. package/dist/types/modes/acp/acp-event-mapper.d.ts +32 -0
  405. package/dist/types/modes/acp/acp-mode.d.ts +5 -0
  406. package/dist/types/modes/acp/index.d.ts +2 -0
  407. package/dist/types/modes/acp/terminal-auth.d.ts +6 -0
  408. package/dist/types/modes/components/agent-dashboard.d.ts +22 -0
  409. package/dist/types/modes/components/assistant-message.d.ts +35 -0
  410. package/dist/types/modes/components/bash-execution.d.ts +29 -0
  411. package/dist/types/modes/components/bordered-loader.d.ts +11 -0
  412. package/dist/types/modes/components/branch-summary-message.d.ts +13 -0
  413. package/dist/types/modes/components/btw-panel.d.ts +16 -0
  414. package/dist/types/modes/components/compaction-summary-message.d.ts +13 -0
  415. package/dist/types/modes/components/copy-selector.d.ts +22 -0
  416. package/dist/types/modes/components/countdown-timer.d.ts +14 -0
  417. package/dist/types/modes/components/custom-editor.d.ts +53 -0
  418. package/dist/types/modes/components/custom-message.d.ts +15 -0
  419. package/dist/types/modes/components/diff.d.ts +11 -0
  420. package/dist/types/modes/components/dynamic-border.d.ts +14 -0
  421. package/dist/types/modes/components/error-banner.d.ts +11 -0
  422. package/dist/types/modes/components/eval-execution.d.ts +23 -0
  423. package/dist/types/modes/components/execution-shared.d.ts +46 -0
  424. package/dist/types/modes/components/extensions/extension-dashboard.d.ts +27 -0
  425. package/dist/types/modes/components/extensions/extension-list.d.ts +39 -0
  426. package/dist/types/modes/components/extensions/index.d.ts +8 -0
  427. package/dist/types/modes/components/extensions/inspector-panel.d.ts +8 -0
  428. package/dist/types/modes/components/extensions/state-manager.d.ts +50 -0
  429. package/dist/types/modes/components/extensions/types.d.ts +147 -0
  430. package/dist/types/modes/components/footer.d.ts +30 -0
  431. package/dist/types/modes/components/history-search.d.ts +7 -0
  432. package/dist/types/modes/components/hook-editor.d.ts +18 -0
  433. package/dist/types/modes/components/hook-input.d.ts +15 -0
  434. package/dist/types/modes/components/hook-message.d.ts +15 -0
  435. package/dist/types/modes/components/hook-selector.d.ts +70 -0
  436. package/dist/types/modes/components/index.d.ts +37 -0
  437. package/dist/types/modes/components/keybinding-hints.d.ts +40 -0
  438. package/dist/types/modes/components/login-dialog.d.ts +32 -0
  439. package/dist/types/modes/components/mcp-add-wizard.d.ts +25 -0
  440. package/dist/types/modes/components/message-frame.d.ts +42 -0
  441. package/dist/types/modes/components/model-selector.d.ts +28 -0
  442. package/dist/types/modes/components/oauth-selector.d.ts +14 -0
  443. package/dist/types/modes/components/omfg-panel.d.ts +19 -0
  444. package/dist/types/modes/components/plugin-selector.d.ts +26 -0
  445. package/dist/types/modes/components/plugin-settings.d.ts +98 -0
  446. package/dist/types/modes/components/queue-mode-selector.d.ts +9 -0
  447. package/dist/types/modes/components/read-tool-group.d.ts +30 -0
  448. package/dist/types/modes/components/segment-track.d.ts +22 -0
  449. package/dist/types/modes/components/session-observer-overlay.d.ts +11 -0
  450. package/dist/types/modes/components/session-selector.d.ts +62 -0
  451. package/dist/types/modes/components/settings-defs.d.ts +50 -0
  452. package/dist/types/modes/components/settings-selector.d.ts +54 -0
  453. package/dist/types/modes/components/show-images-selector.d.ts +9 -0
  454. package/dist/types/modes/components/skill-message.d.ts +9 -0
  455. package/dist/types/modes/components/status-line/context-thresholds.d.ts +10 -0
  456. package/dist/types/modes/components/status-line/git-utils.d.ts +22 -0
  457. package/dist/types/modes/components/status-line/index.d.ts +4 -0
  458. package/dist/types/modes/components/status-line/presets.d.ts +3 -0
  459. package/dist/types/modes/components/status-line/segments.d.ts +5 -0
  460. package/dist/types/modes/components/status-line/separators.d.ts +3 -0
  461. package/dist/types/modes/components/status-line/token-rate.d.ts +10 -0
  462. package/dist/types/modes/components/status-line/types.d.ts +80 -0
  463. package/dist/types/modes/components/status-line.d.ts +77 -0
  464. package/dist/types/modes/components/theme-selector.d.ts +10 -0
  465. package/dist/types/modes/components/thinking-selector.d.ts +10 -0
  466. package/dist/types/modes/components/tiny-title-download-progress.d.ts +11 -0
  467. package/dist/types/modes/components/todo-reminder.d.ts +13 -0
  468. package/dist/types/modes/components/tool-execution.d.ts +77 -0
  469. package/dist/types/modes/components/transcript-container.d.ts +38 -0
  470. package/dist/types/modes/components/tree-selector.d.ts +31 -0
  471. package/dist/types/modes/components/ttsr-notification.d.ts +13 -0
  472. package/dist/types/modes/components/user-message-selector.d.ts +28 -0
  473. package/dist/types/modes/components/user-message.d.ts +8 -0
  474. package/dist/types/modes/components/visual-truncate.d.ts +19 -0
  475. package/dist/types/modes/components/welcome.d.ts +55 -0
  476. package/dist/types/modes/controllers/btw-controller.d.ts +10 -0
  477. package/dist/types/modes/controllers/command-controller-shared.d.ts +47 -0
  478. package/dist/types/modes/controllers/command-controller.d.ts +41 -0
  479. package/dist/types/modes/controllers/event-controller.d.ts +12 -0
  480. package/dist/types/modes/controllers/extension-ui-controller.d.ts +83 -0
  481. package/dist/types/modes/controllers/input-controller.d.ts +34 -0
  482. package/dist/types/modes/controllers/mcp-command-controller.d.ts +10 -0
  483. package/dist/types/modes/controllers/omfg-controller.d.ts +10 -0
  484. package/dist/types/modes/controllers/omfg-rule.d.ts +26 -0
  485. package/dist/types/modes/controllers/selector-controller.d.ts +46 -0
  486. package/dist/types/modes/controllers/ssh-command-controller.d.ts +10 -0
  487. package/dist/types/modes/controllers/todo-command-controller.d.ts +7 -0
  488. package/dist/types/modes/emoji-autocomplete.d.ts +16 -0
  489. package/dist/types/modes/gradient-highlight.d.ts +27 -0
  490. package/dist/types/modes/image-references.d.ts +17 -0
  491. package/dist/types/modes/index.d.ts +9 -0
  492. package/dist/types/modes/interactive-mode.d.ts +291 -0
  493. package/dist/types/modes/internal-url-autocomplete.d.ts +43 -0
  494. package/dist/types/modes/loop-limit.d.ts +22 -0
  495. package/dist/types/modes/magic-keywords.d.ts +14 -0
  496. package/dist/types/modes/markdown-prose.d.ts +27 -0
  497. package/dist/types/modes/oauth-manual-input.d.ts +8 -0
  498. package/dist/types/modes/orchestrate.d.ts +15 -0
  499. package/dist/types/modes/print-mode.d.ts +27 -0
  500. package/dist/types/modes/prompt-action-autocomplete.d.ts +46 -0
  501. package/dist/types/modes/rpc/host-tools.d.ts +16 -0
  502. package/dist/types/modes/rpc/host-uris.d.ts +38 -0
  503. package/dist/types/modes/rpc/rpc-client.d.ts +249 -0
  504. package/dist/types/modes/rpc/rpc-mode.d.ts +17 -0
  505. package/dist/types/modes/rpc/rpc-types.d.ts +586 -0
  506. package/dist/types/modes/runtime-init.d.ts +21 -0
  507. package/dist/types/modes/session-observer-registry.d.ts +26 -0
  508. package/dist/types/modes/setup-wizard/index.d.ts +16 -0
  509. package/dist/types/modes/setup-wizard/scenes/channels.d.ts +14 -0
  510. package/dist/types/modes/setup-wizard/scenes/glyph.d.ts +2 -0
  511. package/dist/types/modes/setup-wizard/scenes/outro.d.ts +2 -0
  512. package/dist/types/modes/setup-wizard/scenes/providers.d.ts +2 -0
  513. package/dist/types/modes/setup-wizard/scenes/sign-in.d.ts +19 -0
  514. package/dist/types/modes/setup-wizard/scenes/splash.d.ts +11 -0
  515. package/dist/types/modes/setup-wizard/scenes/telegram.d.ts +20 -0
  516. package/dist/types/modes/setup-wizard/scenes/theme.d.ts +2 -0
  517. package/dist/types/modes/setup-wizard/scenes/types.d.ts +43 -0
  518. package/dist/types/modes/setup-wizard/scenes/web-search.d.ts +19 -0
  519. package/dist/types/modes/setup-wizard/wizard-overlay.d.ts +14 -0
  520. package/dist/types/modes/shared.d.ts +15 -0
  521. package/dist/types/modes/theme/defaults/index.d.ts +9436 -0
  522. package/dist/types/modes/theme/mermaid-cache.d.ts +9 -0
  523. package/dist/types/modes/theme/shimmer.d.ts +40 -0
  524. package/dist/types/modes/theme/theme.d.ts +305 -0
  525. package/dist/types/modes/turn-budget.d.ts +18 -0
  526. package/dist/types/modes/types.d.ts +294 -0
  527. package/dist/types/modes/ultrathink.d.ts +15 -0
  528. package/dist/types/modes/utils/context-usage.d.ts +47 -0
  529. package/dist/types/modes/utils/copy-targets.d.ts +53 -0
  530. package/dist/types/modes/utils/hotkeys-markdown.d.ts +5 -0
  531. package/dist/types/modes/utils/keybinding-matchers.d.ts +19 -0
  532. package/dist/types/modes/utils/tools-markdown.d.ts +5 -0
  533. package/dist/types/modes/utils/ui-helpers.d.ts +55 -0
  534. package/dist/types/modes/workflow.d.ts +15 -0
  535. package/dist/types/plan-mode/approved-plan.d.ts +49 -0
  536. package/dist/types/plan-mode/plan-handoff.d.ts +20 -0
  537. package/dist/types/plan-mode/plan-protection.d.ts +12 -0
  538. package/dist/types/plan-mode/state.d.ts +6 -0
  539. package/dist/types/registry/agent-registry.d.ts +62 -0
  540. package/dist/types/sdk.d.ts +271 -0
  541. package/dist/types/secrets/index.d.ts +9 -0
  542. package/dist/types/secrets/obfuscator.d.ts +23 -0
  543. package/dist/types/secrets/regex.d.ts +2 -0
  544. package/dist/types/session/agent-session.d.ts +1032 -0
  545. package/dist/types/session/agent-storage.d.ts +94 -0
  546. package/dist/types/session/artifacts.d.ts +61 -0
  547. package/dist/types/session/auth-broker-config.d.ts +13 -0
  548. package/dist/types/session/auth-storage.d.ts +6 -0
  549. package/dist/types/session/blob-store.d.ts +64 -0
  550. package/dist/types/session/client-bridge.d.ts +88 -0
  551. package/dist/types/session/history-storage.d.ts +30 -0
  552. package/dist/types/session/indexed-session-storage.d.ts +59 -0
  553. package/dist/types/session/messages.d.ts +173 -0
  554. package/dist/types/session/redis-session-storage.d.ts +51 -0
  555. package/dist/types/session/session-dump-format.d.ts +22 -0
  556. package/dist/types/session/session-manager.d.ts +594 -0
  557. package/dist/types/session/session-storage.d.ts +86 -0
  558. package/dist/types/session/shake-types.d.ts +24 -0
  559. package/dist/types/session/sql-session-storage.d.ts +72 -0
  560. package/dist/types/session/streaming-output.d.ts +185 -0
  561. package/dist/types/session/tool-choice-queue.d.ts +78 -0
  562. package/dist/types/session/yield-queue.d.ts +24 -0
  563. package/dist/types/slash-commands/acp-builtins.d.ts +18 -0
  564. package/dist/types/slash-commands/builtin-registry.d.ts +23 -0
  565. package/dist/types/slash-commands/headless-plan.d.ts +3 -0
  566. package/dist/types/slash-commands/helpers/context-report.d.ts +7 -0
  567. package/dist/types/slash-commands/helpers/format.d.ts +10 -0
  568. package/dist/types/slash-commands/helpers/marketplace-manager.d.ts +8 -0
  569. package/dist/types/slash-commands/helpers/mcp.d.ts +3 -0
  570. package/dist/types/slash-commands/helpers/parse.d.ts +33 -0
  571. package/dist/types/slash-commands/helpers/ssh.d.ts +3 -0
  572. package/dist/types/slash-commands/helpers/todo.d.ts +3 -0
  573. package/dist/types/slash-commands/helpers/usage-report.d.ts +7 -0
  574. package/dist/types/slash-commands/marketplace-install-parser.d.ts +33 -0
  575. package/dist/types/slash-commands/types.d.ts +129 -0
  576. package/dist/types/ssh/config-writer.d.ts +49 -0
  577. package/dist/types/ssh/connection-manager.d.ts +33 -0
  578. package/dist/types/ssh/ssh-executor.d.ts +37 -0
  579. package/dist/types/ssh/sshfs-mount.d.ts +6 -0
  580. package/dist/types/ssh/utils.d.ts +2 -0
  581. package/dist/types/stt/downloader.d.ts +9 -0
  582. package/dist/types/stt/index.d.ts +3 -0
  583. package/dist/types/stt/recorder.d.ts +13 -0
  584. package/dist/types/stt/setup.d.ts +18 -0
  585. package/dist/types/stt/stt-controller.d.ts +16 -0
  586. package/dist/types/stt/transcriber.d.ts +16 -0
  587. package/dist/types/system-prompt.d.ts +97 -0
  588. package/dist/types/task/agents.d.ts +28 -0
  589. package/dist/types/task/commands.d.ts +35 -0
  590. package/dist/types/task/discovery.d.ts +18 -0
  591. package/dist/types/task/executor.d.ts +121 -0
  592. package/dist/types/task/index.d.ts +42 -0
  593. package/dist/types/task/name-generator.d.ts +16 -0
  594. package/dist/types/task/output-manager.d.ts +27 -0
  595. package/dist/types/task/parallel.d.ts +34 -0
  596. package/dist/types/task/prometheus-command.d.ts +7 -0
  597. package/dist/types/task/render.d.ts +26 -0
  598. package/dist/types/task/repair-args.d.ts +52 -0
  599. package/dist/types/task/simple-mode.d.ts +8 -0
  600. package/dist/types/task/subprocess-tool-registry.d.ts +72 -0
  601. package/dist/types/task/types.d.ts +300 -0
  602. package/dist/types/task/worktree.d.ts +94 -0
  603. package/dist/types/telemetry-export.d.ts +43 -0
  604. package/dist/types/thinking.d.ts +68 -0
  605. package/dist/types/tiny/compiled-runtime.d.ts +35 -0
  606. package/dist/types/tiny/device.d.ts +78 -0
  607. package/dist/types/tiny/dtype.d.ts +85 -0
  608. package/dist/types/tiny/models.d.ts +204 -0
  609. package/dist/types/tiny/text.d.ts +36 -0
  610. package/dist/types/tiny/title-client.d.ts +73 -0
  611. package/dist/types/tiny/title-protocol.d.ts +77 -0
  612. package/dist/types/tiny/worker.d.ts +2 -0
  613. package/dist/types/tool-discovery/mode.d.ts +8 -0
  614. package/dist/types/tool-discovery/tool-index.d.ts +57 -0
  615. package/dist/types/tools/approval.d.ts +46 -0
  616. package/dist/types/tools/archive-reader.d.ts +44 -0
  617. package/dist/types/tools/ask.d.ts +113 -0
  618. package/dist/types/tools/ast-edit.d.ts +79 -0
  619. package/dist/types/tools/ast-grep.d.ts +70 -0
  620. package/dist/types/tools/auto-generated-guard.d.ts +17 -0
  621. package/dist/types/tools/bash-command-fixup.d.ts +11 -0
  622. package/dist/types/tools/bash-interactive.d.ts +16 -0
  623. package/dist/types/tools/bash-interceptor.d.ts +24 -0
  624. package/dist/types/tools/bash-pty-selection.d.ts +7 -0
  625. package/dist/types/tools/bash-skill-urls.d.ts +31 -0
  626. package/dist/types/tools/bash.d.ts +142 -0
  627. package/dist/types/tools/browser/attach.d.ts +34 -0
  628. package/dist/types/tools/browser/launch.d.ts +62 -0
  629. package/dist/types/tools/browser/readable.d.ts +17 -0
  630. package/dist/types/tools/browser/registry.d.ts +42 -0
  631. package/dist/types/tools/browser/render.d.ts +50 -0
  632. package/dist/types/tools/browser/tab-protocol.d.ts +144 -0
  633. package/dist/types/tools/browser/tab-supervisor.d.ts +63 -0
  634. package/dist/types/tools/browser/tab-worker-entry.d.ts +1 -0
  635. package/dist/types/tools/browser/tab-worker.d.ts +19 -0
  636. package/dist/types/tools/browser.d.ts +116 -0
  637. package/dist/types/tools/checkpoint.d.ts +65 -0
  638. package/dist/types/tools/conflict-detect.d.ts +205 -0
  639. package/dist/types/tools/context.d.ts +19 -0
  640. package/dist/types/tools/debug.d.ts +217 -0
  641. package/dist/types/tools/eval-backends.d.ts +12 -0
  642. package/dist/types/tools/eval-render.d.ts +60 -0
  643. package/dist/types/tools/eval.d.ts +77 -0
  644. package/dist/types/tools/fetch.d.ts +80 -0
  645. package/dist/types/tools/file-recorder.d.ts +13 -0
  646. package/dist/types/tools/find.d.ts +103 -0
  647. package/dist/types/tools/fs-cache-invalidation.d.ts +15 -0
  648. package/dist/types/tools/gh-format.d.ts +6 -0
  649. package/dist/types/tools/gh-renderer.d.ts +25 -0
  650. package/dist/types/tools/gh.d.ts +357 -0
  651. package/dist/types/tools/github-cache.d.ts +103 -0
  652. package/dist/types/tools/grouped-file-output.d.ts +36 -0
  653. package/dist/types/tools/image-gen.d.ts +82 -0
  654. package/dist/types/tools/index.d.ts +269 -0
  655. package/dist/types/tools/inspect-image-renderer.d.ts +26 -0
  656. package/dist/types/tools/inspect-image.d.ts +32 -0
  657. package/dist/types/tools/irc.d.ts +80 -0
  658. package/dist/types/tools/job.d.ts +66 -0
  659. package/dist/types/tools/json-tree.d.ts +23 -0
  660. package/dist/types/tools/jtd-to-json-schema.d.ts +29 -0
  661. package/dist/types/tools/jtd-to-typescript.d.ts +26 -0
  662. package/dist/types/tools/jtd-utils.d.ts +42 -0
  663. package/dist/types/tools/list-limit.d.ts +12 -0
  664. package/dist/types/tools/match-line-format.d.ts +11 -0
  665. package/dist/types/tools/memory-edit.d.ts +40 -0
  666. package/dist/types/tools/memory-recall.d.ts +24 -0
  667. package/dist/types/tools/memory-reflect.d.ts +26 -0
  668. package/dist/types/tools/memory-render.d.ts +60 -0
  669. package/dist/types/tools/memory-retain.d.ts +30 -0
  670. package/dist/types/tools/output-meta.d.ts +204 -0
  671. package/dist/types/tools/output-schema-validator.d.ts +64 -0
  672. package/dist/types/tools/path-utils.d.ts +199 -0
  673. package/dist/types/tools/plan-mode-guard.d.ts +17 -0
  674. package/dist/types/tools/read.d.ts +86 -0
  675. package/dist/types/tools/render-mermaid.d.ts +38 -0
  676. package/dist/types/tools/render-utils.d.ts +185 -0
  677. package/dist/types/tools/renderers.d.ts +26 -0
  678. package/dist/types/tools/report-tool-issue.d.ts +123 -0
  679. package/dist/types/tools/resolve.d.ts +91 -0
  680. package/dist/types/tools/review.d.ts +63 -0
  681. package/dist/types/tools/search-tool-bm25.d.ts +66 -0
  682. package/dist/types/tools/search.d.ts +95 -0
  683. package/dist/types/tools/sqlite-reader.d.ts +110 -0
  684. package/dist/types/tools/ssh.d.ts +69 -0
  685. package/dist/types/tools/todo.d.ts +159 -0
  686. package/dist/types/tools/tool-errors.d.ts +33 -0
  687. package/dist/types/tools/tool-result.d.ts +32 -0
  688. package/dist/types/tools/tool-timeouts.d.ts +51 -0
  689. package/dist/types/tools/tts.d.ts +18 -0
  690. package/dist/types/tools/write.d.ts +65 -0
  691. package/dist/types/tools/yield.d.ts +26 -0
  692. package/dist/types/tui/code-cell.d.ts +40 -0
  693. package/dist/types/tui/file-list.d.ts +22 -0
  694. package/dist/types/tui/hyperlink.d.ts +54 -0
  695. package/dist/types/tui/index.d.ts +11 -0
  696. package/dist/types/tui/output-block.d.ts +57 -0
  697. package/dist/types/tui/status-line.d.ts +18 -0
  698. package/dist/types/tui/tree-list.d.ts +19 -0
  699. package/dist/types/tui/types.d.ts +13 -0
  700. package/dist/types/tui/utils.d.ts +36 -0
  701. package/dist/types/utils/changelog.d.ts +25 -0
  702. package/dist/types/utils/clipboard.d.ts +26 -0
  703. package/dist/types/utils/command-args.d.ts +9 -0
  704. package/dist/types/utils/commit-message-generator.d.ts +7 -0
  705. package/dist/types/utils/edit-mode.d.ts +13 -0
  706. package/dist/types/utils/event-bus.d.ts +6 -0
  707. package/dist/types/utils/external-editor.d.ts +17 -0
  708. package/dist/types/utils/file-display-mode.d.ts +27 -0
  709. package/dist/types/utils/file-mentions.d.ts +13 -0
  710. package/dist/types/utils/git.d.ts +361 -0
  711. package/dist/types/utils/image-loading.d.ts +26 -0
  712. package/dist/types/utils/image-resize.d.ts +39 -0
  713. package/dist/types/utils/jj.d.ts +49 -0
  714. package/dist/types/utils/lang-from-path.d.ts +8 -0
  715. package/dist/types/utils/markit.d.ts +7 -0
  716. package/dist/types/utils/open.d.ts +2 -0
  717. package/dist/types/utils/session-color.d.ts +15 -0
  718. package/dist/types/utils/shell-snapshot.d.ts +5 -0
  719. package/dist/types/utils/sixel.d.ts +21 -0
  720. package/dist/types/utils/title-generator.d.ts +34 -0
  721. package/dist/types/utils/tool-choice.d.ts +7 -0
  722. package/dist/types/utils/tools-manager.d.ts +9 -0
  723. package/dist/types/web/kagi.d.ts +99 -0
  724. package/dist/types/web/parallel.d.ts +58 -0
  725. package/dist/types/web/scrapers/artifacthub.d.ts +6 -0
  726. package/dist/types/web/scrapers/arxiv.d.ts +5 -0
  727. package/dist/types/web/scrapers/aur.d.ts +5 -0
  728. package/dist/types/web/scrapers/biorxiv.d.ts +5 -0
  729. package/dist/types/web/scrapers/bluesky.d.ts +5 -0
  730. package/dist/types/web/scrapers/brew.d.ts +5 -0
  731. package/dist/types/web/scrapers/cheatsh.d.ts +8 -0
  732. package/dist/types/web/scrapers/chocolatey.d.ts +5 -0
  733. package/dist/types/web/scrapers/choosealicense.d.ts +2 -0
  734. package/dist/types/web/scrapers/cisa-kev.d.ts +5 -0
  735. package/dist/types/web/scrapers/clojars.d.ts +5 -0
  736. package/dist/types/web/scrapers/coingecko.d.ts +5 -0
  737. package/dist/types/web/scrapers/crates-io.d.ts +5 -0
  738. package/dist/types/web/scrapers/crossref.d.ts +2 -0
  739. package/dist/types/web/scrapers/devto.d.ts +5 -0
  740. package/dist/types/web/scrapers/discogs.d.ts +8 -0
  741. package/dist/types/web/scrapers/discourse.d.ts +5 -0
  742. package/dist/types/web/scrapers/dockerhub.d.ts +5 -0
  743. package/dist/types/web/scrapers/docs-rs.d.ts +2 -0
  744. package/dist/types/web/scrapers/fdroid.d.ts +5 -0
  745. package/dist/types/web/scrapers/firefox-addons.d.ts +2 -0
  746. package/dist/types/web/scrapers/flathub.d.ts +2 -0
  747. package/dist/types/web/scrapers/github-gist.d.ts +5 -0
  748. package/dist/types/web/scrapers/github.d.ts +12 -0
  749. package/dist/types/web/scrapers/gitlab.d.ts +5 -0
  750. package/dist/types/web/scrapers/go-pkg.d.ts +5 -0
  751. package/dist/types/web/scrapers/hackage.d.ts +5 -0
  752. package/dist/types/web/scrapers/hackernews.d.ts +2 -0
  753. package/dist/types/web/scrapers/hex.d.ts +5 -0
  754. package/dist/types/web/scrapers/huggingface.d.ts +2 -0
  755. package/dist/types/web/scrapers/iacr.d.ts +5 -0
  756. package/dist/types/web/scrapers/index.d.ts +84 -0
  757. package/dist/types/web/scrapers/jetbrains-marketplace.d.ts +2 -0
  758. package/dist/types/web/scrapers/lemmy.d.ts +2 -0
  759. package/dist/types/web/scrapers/lobsters.d.ts +5 -0
  760. package/dist/types/web/scrapers/mastodon.d.ts +5 -0
  761. package/dist/types/web/scrapers/maven.d.ts +6 -0
  762. package/dist/types/web/scrapers/mdn.d.ts +2 -0
  763. package/dist/types/web/scrapers/metacpan.d.ts +5 -0
  764. package/dist/types/web/scrapers/musicbrainz.d.ts +5 -0
  765. package/dist/types/web/scrapers/npm.d.ts +5 -0
  766. package/dist/types/web/scrapers/nuget.d.ts +5 -0
  767. package/dist/types/web/scrapers/nvd.d.ts +5 -0
  768. package/dist/types/web/scrapers/ollama.d.ts +2 -0
  769. package/dist/types/web/scrapers/open-vsx.d.ts +5 -0
  770. package/dist/types/web/scrapers/opencorporates.d.ts +5 -0
  771. package/dist/types/web/scrapers/openlibrary.d.ts +5 -0
  772. package/dist/types/web/scrapers/orcid.d.ts +5 -0
  773. package/dist/types/web/scrapers/osv.d.ts +5 -0
  774. package/dist/types/web/scrapers/packagist.d.ts +5 -0
  775. package/dist/types/web/scrapers/pub-dev.d.ts +5 -0
  776. package/dist/types/web/scrapers/pubmed.d.ts +5 -0
  777. package/dist/types/web/scrapers/pypi.d.ts +5 -0
  778. package/dist/types/web/scrapers/rawg.d.ts +2 -0
  779. package/dist/types/web/scrapers/readthedocs.d.ts +2 -0
  780. package/dist/types/web/scrapers/reddit.d.ts +5 -0
  781. package/dist/types/web/scrapers/repology.d.ts +5 -0
  782. package/dist/types/web/scrapers/rfc.d.ts +5 -0
  783. package/dist/types/web/scrapers/rubygems.d.ts +5 -0
  784. package/dist/types/web/scrapers/searchcode.d.ts +2 -0
  785. package/dist/types/web/scrapers/sec-edgar.d.ts +5 -0
  786. package/dist/types/web/scrapers/semantic-scholar.d.ts +2 -0
  787. package/dist/types/web/scrapers/snapcraft.d.ts +2 -0
  788. package/dist/types/web/scrapers/sourcegraph.d.ts +2 -0
  789. package/dist/types/web/scrapers/spdx.d.ts +5 -0
  790. package/dist/types/web/scrapers/spotify.d.ts +8 -0
  791. package/dist/types/web/scrapers/stackoverflow.d.ts +6 -0
  792. package/dist/types/web/scrapers/terraform.d.ts +5 -0
  793. package/dist/types/web/scrapers/tldr.d.ts +7 -0
  794. package/dist/types/web/scrapers/twitter.d.ts +5 -0
  795. package/dist/types/web/scrapers/types.d.ts +78 -0
  796. package/dist/types/web/scrapers/utils.d.ts +26 -0
  797. package/dist/types/web/scrapers/vimeo.d.ts +5 -0
  798. package/dist/types/web/scrapers/vscode-marketplace.d.ts +5 -0
  799. package/dist/types/web/scrapers/w3c.d.ts +2 -0
  800. package/dist/types/web/scrapers/wikidata.d.ts +5 -0
  801. package/dist/types/web/scrapers/wikipedia.d.ts +5 -0
  802. package/dist/types/web/scrapers/youtube.d.ts +5 -0
  803. package/dist/types/web/search/index.d.ts +85 -0
  804. package/dist/types/web/search/provider.d.ts +21 -0
  805. package/dist/types/web/search/providers/anthropic.d.ts +32 -0
  806. package/dist/types/web/search/providers/base.d.ts +81 -0
  807. package/dist/types/web/search/providers/brave.d.ts +27 -0
  808. package/dist/types/web/search/providers/codex.d.ts +35 -0
  809. package/dist/types/web/search/providers/exa.d.ts +67 -0
  810. package/dist/types/web/search/providers/gemini.d.ts +57 -0
  811. package/dist/types/web/search/providers/jina.d.ts +26 -0
  812. package/dist/types/web/search/providers/kagi.d.ts +25 -0
  813. package/dist/types/web/search/providers/kimi.d.ts +27 -0
  814. package/dist/types/web/search/providers/parallel.d.ts +15 -0
  815. package/dist/types/web/search/providers/perplexity.d.ts +38 -0
  816. package/dist/types/web/search/providers/searxng.d.ts +44 -0
  817. package/dist/types/web/search/providers/synthetic.d.ts +21 -0
  818. package/dist/types/web/search/providers/tavily.d.ts +29 -0
  819. package/dist/types/web/search/providers/utils.d.ts +52 -0
  820. package/dist/types/web/search/providers/zai.d.ts +28 -0
  821. package/dist/types/web/search/render.d.ts +34 -0
  822. package/dist/types/web/search/types.d.ts +409 -0
  823. package/dist/types/web/search/utils.d.ts +4 -0
  824. package/dist/types/workspace-tree.d.ts +42 -0
  825. package/examples/README.md +21 -0
  826. package/examples/custom-tools/README.md +104 -0
  827. package/examples/custom-tools/hello/index.ts +20 -0
  828. package/examples/extensions/README.md +142 -0
  829. package/examples/extensions/api-demo.ts +79 -0
  830. package/examples/extensions/chalk-logger.ts +25 -0
  831. package/examples/extensions/hello.ts +31 -0
  832. package/examples/extensions/pirate.ts +43 -0
  833. package/examples/extensions/plan-mode.ts +549 -0
  834. package/examples/extensions/reload-runtime.ts +38 -0
  835. package/examples/extensions/thinking-note.ts +13 -0
  836. package/examples/extensions/tools.ts +144 -0
  837. package/examples/extensions/with-deps/index.ts +36 -0
  838. package/examples/extensions/with-deps/package-lock.json +31 -0
  839. package/examples/extensions/with-deps/package.json +16 -0
  840. package/examples/hooks/README.md +56 -0
  841. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  842. package/examples/hooks/confirm-destructive.ts +58 -0
  843. package/examples/hooks/custom-compaction.ts +116 -0
  844. package/examples/hooks/dirty-repo-guard.ts +51 -0
  845. package/examples/hooks/file-trigger.ts +40 -0
  846. package/examples/hooks/git-checkpoint.ts +52 -0
  847. package/examples/hooks/handoff.ts +150 -0
  848. package/examples/hooks/permission-gate.ts +33 -0
  849. package/examples/hooks/protected-paths.ts +29 -0
  850. package/examples/hooks/qna.ts +119 -0
  851. package/examples/hooks/status-line.ts +39 -0
  852. package/examples/sdk/01-minimal.ts +21 -0
  853. package/examples/sdk/02-custom-model.ts +50 -0
  854. package/examples/sdk/03-custom-prompt.ts +46 -0
  855. package/examples/sdk/04-skills.ts +43 -0
  856. package/examples/sdk/06-extensions.ts +82 -0
  857. package/examples/sdk/06-hooks.ts +61 -0
  858. package/examples/sdk/07-context-files.ts +35 -0
  859. package/examples/sdk/08-prompt-templates.ts +36 -0
  860. package/examples/sdk/08-slash-commands.ts +41 -0
  861. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  862. package/examples/sdk/11-sessions.ts +47 -0
  863. package/examples/sdk/12-redis-sessions.ts +54 -0
  864. package/examples/sdk/13-sql-sessions.ts +61 -0
  865. package/examples/sdk/README.md +172 -0
  866. package/package.json +548 -0
  867. package/scripts/build-binary.ts +105 -0
  868. package/scripts/format-prompts.ts +68 -0
  869. package/scripts/generate-docs-index.ts +40 -0
  870. package/scripts/generate-template.ts +33 -0
  871. package/src/async/index.ts +2 -0
  872. package/src/async/job-manager.ts +571 -0
  873. package/src/async/support.ts +5 -0
  874. package/src/auto-thinking/classifier.ts +180 -0
  875. package/src/autoresearch/command-resume.md +14 -0
  876. package/src/autoresearch/dashboard.ts +436 -0
  877. package/src/autoresearch/git.ts +319 -0
  878. package/src/autoresearch/helpers.ts +218 -0
  879. package/src/autoresearch/index.ts +536 -0
  880. package/src/autoresearch/prompt-setup.md +43 -0
  881. package/src/autoresearch/prompt.md +103 -0
  882. package/src/autoresearch/resume-message.md +10 -0
  883. package/src/autoresearch/state.ts +273 -0
  884. package/src/autoresearch/storage.ts +699 -0
  885. package/src/autoresearch/tools/init-experiment.ts +272 -0
  886. package/src/autoresearch/tools/log-experiment.ts +524 -0
  887. package/src/autoresearch/tools/run-experiment.ts +407 -0
  888. package/src/autoresearch/tools/update-notes.ts +109 -0
  889. package/src/autoresearch/types.ts +168 -0
  890. package/src/bun-imports.d.ts +28 -0
  891. package/src/capability/context-file.ts +58 -0
  892. package/src/capability/extension-module.ts +34 -0
  893. package/src/capability/extension.ts +47 -0
  894. package/src/capability/fs.ts +107 -0
  895. package/src/capability/hook.ts +40 -0
  896. package/src/capability/index.ts +441 -0
  897. package/src/capability/instruction.ts +37 -0
  898. package/src/capability/mcp.ts +74 -0
  899. package/src/capability/prompt.ts +35 -0
  900. package/src/capability/rule-buckets.ts +64 -0
  901. package/src/capability/rule.ts +252 -0
  902. package/src/capability/settings.ts +34 -0
  903. package/src/capability/skill.ts +63 -0
  904. package/src/capability/slash-command.ts +40 -0
  905. package/src/capability/ssh.ts +41 -0
  906. package/src/capability/system-prompt.ts +34 -0
  907. package/src/capability/tool.ts +38 -0
  908. package/src/capability/types.ts +172 -0
  909. package/src/cli/agents-cli.ts +138 -0
  910. package/src/cli/args.ts +335 -0
  911. package/src/cli/auth-broker-cli.ts +898 -0
  912. package/src/cli/auth-gateway-cli.ts +612 -0
  913. package/src/cli/classify-install-target.ts +76 -0
  914. package/src/cli/claude-trace-cli.ts +813 -0
  915. package/src/cli/commands/init-xdg.ts +27 -0
  916. package/src/cli/completion-gen.ts +550 -0
  917. package/src/cli/config-cli.ts +418 -0
  918. package/src/cli/dry-balance-cli.ts +823 -0
  919. package/src/cli/extension-flags.ts +48 -0
  920. package/src/cli/file-processor.ts +133 -0
  921. package/src/cli/gateway-cli.ts +664 -0
  922. package/src/cli/grep-cli.ts +160 -0
  923. package/src/cli/grievances-cli.ts +260 -0
  924. package/src/cli/initial-message.ts +58 -0
  925. package/src/cli/list-models.ts +194 -0
  926. package/src/cli/plugin-cli.ts +996 -0
  927. package/src/cli/read-cli.ts +57 -0
  928. package/src/cli/session-picker.ts +79 -0
  929. package/src/cli/setup-cli.ts +231 -0
  930. package/src/cli/shell-cli.ts +176 -0
  931. package/src/cli/ssh-cli.ts +181 -0
  932. package/src/cli/stats-cli.ts +238 -0
  933. package/src/cli/tiny-models-cli.ts +127 -0
  934. package/src/cli/update-cli.ts +739 -0
  935. package/src/cli/web-search-cli.ts +132 -0
  936. package/src/cli/worktree-cli.ts +291 -0
  937. package/src/cli-commands.ts +49 -0
  938. package/src/cli.ts +123 -0
  939. package/src/commands/acp.ts +24 -0
  940. package/src/commands/agents.ts +57 -0
  941. package/src/commands/auth-broker.ts +99 -0
  942. package/src/commands/auth-gateway.ts +69 -0
  943. package/src/commands/commit.ts +46 -0
  944. package/src/commands/complete.ts +66 -0
  945. package/src/commands/completions.ts +60 -0
  946. package/src/commands/config.ts +51 -0
  947. package/src/commands/dry-balance.ts +43 -0
  948. package/src/commands/gateway.ts +103 -0
  949. package/src/commands/grep.ts +48 -0
  950. package/src/commands/grievances.ts +51 -0
  951. package/src/commands/install.ts +107 -0
  952. package/src/commands/launch.ts +162 -0
  953. package/src/commands/plugin.ts +78 -0
  954. package/src/commands/read.ts +35 -0
  955. package/src/commands/setup.ts +67 -0
  956. package/src/commands/shell.ts +29 -0
  957. package/src/commands/ssh.ts +60 -0
  958. package/src/commands/stats.ts +29 -0
  959. package/src/commands/tiny-models.ts +36 -0
  960. package/src/commands/update.ts +21 -0
  961. package/src/commands/web-search.ts +42 -0
  962. package/src/commands/worktree.ts +56 -0
  963. package/src/commit/agentic/agent.ts +316 -0
  964. package/src/commit/agentic/fallback.ts +96 -0
  965. package/src/commit/agentic/index.ts +355 -0
  966. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  967. package/src/commit/agentic/prompts/session-user.md +25 -0
  968. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  969. package/src/commit/agentic/prompts/system.md +38 -0
  970. package/src/commit/agentic/state.ts +60 -0
  971. package/src/commit/agentic/tools/analyze-file.ts +127 -0
  972. package/src/commit/agentic/tools/git-file-diff.ts +191 -0
  973. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  974. package/src/commit/agentic/tools/git-overview.ts +81 -0
  975. package/src/commit/agentic/tools/index.ts +54 -0
  976. package/src/commit/agentic/tools/propose-changelog.ts +144 -0
  977. package/src/commit/agentic/tools/propose-commit.ts +109 -0
  978. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  979. package/src/commit/agentic/tools/schemas.ts +23 -0
  980. package/src/commit/agentic/tools/split-commit.ts +238 -0
  981. package/src/commit/agentic/topo-sort.ts +44 -0
  982. package/src/commit/agentic/trivial.ts +51 -0
  983. package/src/commit/agentic/validation.ts +183 -0
  984. package/src/commit/analysis/conventional.ts +64 -0
  985. package/src/commit/analysis/index.ts +4 -0
  986. package/src/commit/analysis/scope.ts +242 -0
  987. package/src/commit/analysis/summary.ts +105 -0
  988. package/src/commit/analysis/validation.ts +66 -0
  989. package/src/commit/changelog/detect.ts +40 -0
  990. package/src/commit/changelog/generate.ts +97 -0
  991. package/src/commit/changelog/index.ts +234 -0
  992. package/src/commit/changelog/parse.ts +44 -0
  993. package/src/commit/cli.ts +85 -0
  994. package/src/commit/git/diff.ts +148 -0
  995. package/src/commit/index.ts +5 -0
  996. package/src/commit/map-reduce/index.ts +69 -0
  997. package/src/commit/map-reduce/map-phase.ts +193 -0
  998. package/src/commit/map-reduce/reduce-phase.ts +49 -0
  999. package/src/commit/map-reduce/utils.ts +9 -0
  1000. package/src/commit/message.ts +11 -0
  1001. package/src/commit/model-selection.ts +66 -0
  1002. package/src/commit/pipeline.ts +243 -0
  1003. package/src/commit/prompts/analysis-system.md +148 -0
  1004. package/src/commit/prompts/analysis-user.md +38 -0
  1005. package/src/commit/prompts/changelog-system.md +50 -0
  1006. package/src/commit/prompts/changelog-user.md +18 -0
  1007. package/src/commit/prompts/file-observer-system.md +24 -0
  1008. package/src/commit/prompts/file-observer-user.md +8 -0
  1009. package/src/commit/prompts/reduce-system.md +50 -0
  1010. package/src/commit/prompts/reduce-user.md +17 -0
  1011. package/src/commit/prompts/summary-retry.md +3 -0
  1012. package/src/commit/prompts/summary-system.md +38 -0
  1013. package/src/commit/prompts/summary-user.md +13 -0
  1014. package/src/commit/prompts/types-description.md +2 -0
  1015. package/src/commit/shared-llm.ts +77 -0
  1016. package/src/commit/types.ts +118 -0
  1017. package/src/commit/utils/exclusions.ts +42 -0
  1018. package/src/commit/utils.ts +58 -0
  1019. package/src/config/append-only-context-mode.ts +37 -0
  1020. package/src/config/config-file.ts +317 -0
  1021. package/src/config/file-lock.ts +164 -0
  1022. package/src/config/keybindings.ts +623 -0
  1023. package/src/config/mcp-schema.json +230 -0
  1024. package/src/config/model-equivalence.ts +852 -0
  1025. package/src/config/model-id-affixes.ts +64 -0
  1026. package/src/config/model-registry.ts +2650 -0
  1027. package/src/config/model-resolver.ts +1402 -0
  1028. package/src/config/models-config-schema.ts +177 -0
  1029. package/src/config/prompt-templates.ts +185 -0
  1030. package/src/config/resolve-config-value.ts +94 -0
  1031. package/src/config/settings-schema.ts +3752 -0
  1032. package/src/config/settings.ts +1017 -0
  1033. package/src/config.ts +242 -0
  1034. package/src/cursor.ts +340 -0
  1035. package/src/dap/client.ts +675 -0
  1036. package/src/dap/config.ts +150 -0
  1037. package/src/dap/defaults.json +211 -0
  1038. package/src/dap/index.ts +4 -0
  1039. package/src/dap/session.ts +1339 -0
  1040. package/src/dap/types.ts +600 -0
  1041. package/src/debug/index.ts +525 -0
  1042. package/src/debug/log-formatting.ts +58 -0
  1043. package/src/debug/log-viewer.ts +908 -0
  1044. package/src/debug/profiler.ts +162 -0
  1045. package/src/debug/protocol-probe.ts +267 -0
  1046. package/src/debug/raw-sse-buffer.ts +270 -0
  1047. package/src/debug/raw-sse.ts +292 -0
  1048. package/src/debug/report-bundle.ts +365 -0
  1049. package/src/debug/system-info.ts +111 -0
  1050. package/src/debug/terminal-info.ts +127 -0
  1051. package/src/discovery/agents-md.ts +63 -0
  1052. package/src/discovery/agents.ts +236 -0
  1053. package/src/discovery/builtin-defaults.ts +39 -0
  1054. package/src/discovery/builtin-rules/index.ts +50 -0
  1055. package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
  1056. package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
  1057. package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
  1058. package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
  1059. package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
  1060. package/src/discovery/builtin-rules/rs-result-type.md +19 -0
  1061. package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
  1062. package/src/discovery/builtin-rules/ts-import-type.md +42 -0
  1063. package/src/discovery/builtin-rules/ts-no-any.md +56 -0
  1064. package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
  1065. package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
  1066. package/src/discovery/builtin-rules/ts-no-return-type.md +45 -0
  1067. package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
  1068. package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
  1069. package/src/discovery/builtin-rules/ts-set-map.md +28 -0
  1070. package/src/discovery/builtin.ts +906 -0
  1071. package/src/discovery/claude-plugins.ts +386 -0
  1072. package/src/discovery/claude.ts +584 -0
  1073. package/src/discovery/cline.ts +83 -0
  1074. package/src/discovery/codex.ts +522 -0
  1075. package/src/discovery/context-files.ts +49 -0
  1076. package/src/discovery/cursor.ts +220 -0
  1077. package/src/discovery/gemini.ts +383 -0
  1078. package/src/discovery/github.ts +154 -0
  1079. package/src/discovery/helpers.ts +1015 -0
  1080. package/src/discovery/index.ts +81 -0
  1081. package/src/discovery/mcp-json.ts +171 -0
  1082. package/src/discovery/opencode.ts +398 -0
  1083. package/src/discovery/plugin-dir-roots.ts +28 -0
  1084. package/src/discovery/prometheus-extension-roots.ts +190 -0
  1085. package/src/discovery/prometheus-plugins.ts +385 -0
  1086. package/src/discovery/ssh.ts +153 -0
  1087. package/src/discovery/substitute-plugin-root.ts +29 -0
  1088. package/src/discovery/vscode.ts +105 -0
  1089. package/src/discovery/windsurf.ts +147 -0
  1090. package/src/edit/apply-patch/index.ts +87 -0
  1091. package/src/edit/apply-patch/parser.ts +174 -0
  1092. package/src/edit/diff.ts +812 -0
  1093. package/src/edit/file-snapshot-store.ts +59 -0
  1094. package/src/edit/hashline/block-resolver.ts +14 -0
  1095. package/src/edit/hashline/diff.ts +169 -0
  1096. package/src/edit/hashline/execute.ts +187 -0
  1097. package/src/edit/hashline/filesystem.ts +129 -0
  1098. package/src/edit/hashline/index.ts +5 -0
  1099. package/src/edit/hashline/params.ts +18 -0
  1100. package/src/edit/index.ts +542 -0
  1101. package/src/edit/modes/apply-patch.lark +19 -0
  1102. package/src/edit/modes/apply-patch.ts +53 -0
  1103. package/src/edit/modes/patch.ts +1835 -0
  1104. package/src/edit/modes/replace.ts +1103 -0
  1105. package/src/edit/normalize.ts +345 -0
  1106. package/src/edit/notebook.ts +222 -0
  1107. package/src/edit/read-file.ts +25 -0
  1108. package/src/edit/renderer.ts +660 -0
  1109. package/src/edit/streaming.ts +517 -0
  1110. package/src/eval/__tests__/agent-bridge.test.ts +650 -0
  1111. package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
  1112. package/src/eval/__tests__/budget-bridge.test.ts +69 -0
  1113. package/src/eval/__tests__/idle-timeout.test.ts +80 -0
  1114. package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
  1115. package/src/eval/__tests__/llm-bridge.test.ts +327 -0
  1116. package/src/eval/__tests__/shared-executors.test.ts +609 -0
  1117. package/src/eval/agent-bridge.ts +303 -0
  1118. package/src/eval/backend.ts +58 -0
  1119. package/src/eval/bridge-timeout.ts +44 -0
  1120. package/src/eval/budget-bridge.ts +48 -0
  1121. package/src/eval/concurrency-bridge.ts +34 -0
  1122. package/src/eval/idle-timeout.ts +98 -0
  1123. package/src/eval/index.ts +4 -0
  1124. package/src/eval/js/context-manager.ts +425 -0
  1125. package/src/eval/js/executor.ts +166 -0
  1126. package/src/eval/js/index.ts +47 -0
  1127. package/src/eval/js/shared/helpers.ts +237 -0
  1128. package/src/eval/js/shared/indirect-eval.ts +30 -0
  1129. package/src/eval/js/shared/local-module-loader.ts +342 -0
  1130. package/src/eval/js/shared/prelude.ts +2 -0
  1131. package/src/eval/js/shared/prelude.txt +169 -0
  1132. package/src/eval/js/shared/rewrite-imports.ts +508 -0
  1133. package/src/eval/js/shared/runtime.ts +344 -0
  1134. package/src/eval/js/shared/types.ts +18 -0
  1135. package/src/eval/js/tool-bridge.ts +162 -0
  1136. package/src/eval/js/worker-core.ts +131 -0
  1137. package/src/eval/js/worker-entry.ts +24 -0
  1138. package/src/eval/js/worker-protocol.ts +41 -0
  1139. package/src/eval/llm-bridge.ts +186 -0
  1140. package/src/eval/py/display.ts +71 -0
  1141. package/src/eval/py/executor.ts +698 -0
  1142. package/src/eval/py/index.ts +59 -0
  1143. package/src/eval/py/kernel.ts +709 -0
  1144. package/src/eval/py/prelude.py +620 -0
  1145. package/src/eval/py/prelude.ts +3 -0
  1146. package/src/eval/py/runner.py +1028 -0
  1147. package/src/eval/py/runtime.ts +239 -0
  1148. package/src/eval/py/spawn-options.ts +126 -0
  1149. package/src/eval/py/tool-bridge.ts +182 -0
  1150. package/src/eval/session-id.ts +8 -0
  1151. package/src/eval/types.ts +48 -0
  1152. package/src/exa/factory.ts +60 -0
  1153. package/src/exa/index.ts +27 -0
  1154. package/src/exa/mcp-client.ts +370 -0
  1155. package/src/exa/render.ts +244 -0
  1156. package/src/exa/researcher.ts +36 -0
  1157. package/src/exa/search.ts +47 -0
  1158. package/src/exa/types.ts +166 -0
  1159. package/src/exa/websets.ts +248 -0
  1160. package/src/exec/bash-executor.ts +338 -0
  1161. package/src/exec/exec.ts +53 -0
  1162. package/src/exec/idle-timeout-watchdog.ts +126 -0
  1163. package/src/exec/non-interactive-env.ts +48 -0
  1164. package/src/export/custom-share.ts +65 -0
  1165. package/src/export/html/index.ts +164 -0
  1166. package/src/export/html/template.css +1051 -0
  1167. package/src/export/html/template.generated.ts +2 -0
  1168. package/src/export/html/template.html +46 -0
  1169. package/src/export/html/template.js +2244 -0
  1170. package/src/export/html/template.macro.ts +25 -0
  1171. package/src/export/html/vendor/highlight.min.js +1213 -0
  1172. package/src/export/html/vendor/marked.min.js +6 -0
  1173. package/src/export/ttsr.ts +462 -0
  1174. package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
  1175. package/src/extensibility/custom-commands/bundled/review/index.ts +489 -0
  1176. package/src/extensibility/custom-commands/index.ts +2 -0
  1177. package/src/extensibility/custom-commands/loader.ts +238 -0
  1178. package/src/extensibility/custom-commands/types.ts +113 -0
  1179. package/src/extensibility/custom-tools/index.ts +7 -0
  1180. package/src/extensibility/custom-tools/loader.ts +240 -0
  1181. package/src/extensibility/custom-tools/types.ts +268 -0
  1182. package/src/extensibility/custom-tools/wrapper.ts +47 -0
  1183. package/src/extensibility/extensions/compact-handler.ts +40 -0
  1184. package/src/extensibility/extensions/get-commands-handler.ts +77 -0
  1185. package/src/extensibility/extensions/index.ts +15 -0
  1186. package/src/extensibility/extensions/loader.ts +552 -0
  1187. package/src/extensibility/extensions/runner.ts +917 -0
  1188. package/src/extensibility/extensions/types.ts +1303 -0
  1189. package/src/extensibility/extensions/wrapper.ts +223 -0
  1190. package/src/extensibility/hooks/index.ts +5 -0
  1191. package/src/extensibility/hooks/loader.ts +257 -0
  1192. package/src/extensibility/hooks/runner.ts +425 -0
  1193. package/src/extensibility/hooks/tool-wrapper.ts +107 -0
  1194. package/src/extensibility/hooks/types.ts +606 -0
  1195. package/src/extensibility/legacy-pi-ai-shim.ts +24 -0
  1196. package/src/extensibility/legacy-pi-coding-agent-shim.ts +15 -0
  1197. package/src/extensibility/plugins/doctor.ts +66 -0
  1198. package/src/extensibility/plugins/git-url.ts +367 -0
  1199. package/src/extensibility/plugins/index.ts +9 -0
  1200. package/src/extensibility/plugins/installer.ts +192 -0
  1201. package/src/extensibility/plugins/legacy-pi-compat.ts +661 -0
  1202. package/src/extensibility/plugins/loader.ts +313 -0
  1203. package/src/extensibility/plugins/manager.ts +827 -0
  1204. package/src/extensibility/plugins/marketplace/cache.ts +136 -0
  1205. package/src/extensibility/plugins/marketplace/fetcher.ts +317 -0
  1206. package/src/extensibility/plugins/marketplace/index.ts +6 -0
  1207. package/src/extensibility/plugins/marketplace/manager.ts +770 -0
  1208. package/src/extensibility/plugins/marketplace/registry.ts +196 -0
  1209. package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
  1210. package/src/extensibility/plugins/marketplace/types.ts +191 -0
  1211. package/src/extensibility/plugins/parser.ts +105 -0
  1212. package/src/extensibility/plugins/types.ts +194 -0
  1213. package/src/extensibility/shared-events.ts +343 -0
  1214. package/src/extensibility/skills.ts +312 -0
  1215. package/src/extensibility/slash-commands.ts +227 -0
  1216. package/src/extensibility/tool-proxy.ts +25 -0
  1217. package/src/extensibility/typebox.ts +418 -0
  1218. package/src/extensibility/utils.ts +44 -0
  1219. package/src/gateway/adapters/telegram/access.ts +53 -0
  1220. package/src/gateway/adapters/telegram/commands.ts +206 -0
  1221. package/src/gateway/adapters/telegram/expressions.ts +678 -0
  1222. package/src/gateway/adapters/telegram/media.ts +831 -0
  1223. package/src/gateway/adapters/telegram/native-command-policy.ts +83 -0
  1224. package/src/gateway/adapters/telegram/native-commands.ts +197 -0
  1225. package/src/gateway/adapters/telegram/normalize.ts +593 -0
  1226. package/src/gateway/adapters/telegram/send-message.ts +1465 -0
  1227. package/src/gateway/adapters/telegram/service.ts +577 -0
  1228. package/src/gateway/adapters/telegram/setup-api.ts +196 -0
  1229. package/src/gateway/adapters/telegram/setup-config.ts +72 -0
  1230. package/src/gateway/adapters/telegram/types.ts +206 -0
  1231. package/src/gateway/adapters/telegram/webhook.ts +364 -0
  1232. package/src/gateway/commands.ts +196 -0
  1233. package/src/gateway/context.ts +276 -0
  1234. package/src/gateway/native-slash-policy.ts +19 -0
  1235. package/src/gateway/prometheus-dispatch.ts +362 -0
  1236. package/src/gateway/router.ts +123 -0
  1237. package/src/gateway/session-key.ts +55 -0
  1238. package/src/gateway/types.ts +119 -0
  1239. package/src/goals/index.ts +3 -0
  1240. package/src/goals/runtime.ts +528 -0
  1241. package/src/goals/state.ts +37 -0
  1242. package/src/goals/tools/goal-tool.ts +241 -0
  1243. package/src/hindsight/backend.ts +354 -0
  1244. package/src/hindsight/bank.ts +141 -0
  1245. package/src/hindsight/client.ts +598 -0
  1246. package/src/hindsight/config.ts +175 -0
  1247. package/src/hindsight/content.ts +210 -0
  1248. package/src/hindsight/index.ts +8 -0
  1249. package/src/hindsight/mental-models.ts +382 -0
  1250. package/src/hindsight/seeds.json +32 -0
  1251. package/src/hindsight/state.ts +483 -0
  1252. package/src/hindsight/transcript.ts +71 -0
  1253. package/src/index.ts +59 -0
  1254. package/src/internal-urls/agent-protocol.ts +146 -0
  1255. package/src/internal-urls/artifact-protocol.ts +98 -0
  1256. package/src/internal-urls/docs-index.generated.ts +107 -0
  1257. package/src/internal-urls/index.ts +24 -0
  1258. package/src/internal-urls/issue-pr-protocol.ts +577 -0
  1259. package/src/internal-urls/json-query.ts +126 -0
  1260. package/src/internal-urls/local-protocol.ts +274 -0
  1261. package/src/internal-urls/mcp-protocol.ts +151 -0
  1262. package/src/internal-urls/memory-protocol.ts +196 -0
  1263. package/src/internal-urls/parse.ts +72 -0
  1264. package/src/internal-urls/prometheus-protocol.ts +93 -0
  1265. package/src/internal-urls/registry-helpers.ts +25 -0
  1266. package/src/internal-urls/router.ts +103 -0
  1267. package/src/internal-urls/rule-protocol.ts +45 -0
  1268. package/src/internal-urls/skill-protocol.ts +96 -0
  1269. package/src/internal-urls/types.ts +152 -0
  1270. package/src/internal-urls/vault-protocol.ts +936 -0
  1271. package/src/lib/xai-http.ts +124 -0
  1272. package/src/lsp/client.ts +1052 -0
  1273. package/src/lsp/clients/biome-client.ts +202 -0
  1274. package/src/lsp/clients/index.ts +50 -0
  1275. package/src/lsp/clients/lsp-linter-client.ts +93 -0
  1276. package/src/lsp/clients/swiftlint-client.ts +120 -0
  1277. package/src/lsp/config.ts +492 -0
  1278. package/src/lsp/defaults.json +493 -0
  1279. package/src/lsp/diagnostics-ledger.ts +51 -0
  1280. package/src/lsp/edits.ts +219 -0
  1281. package/src/lsp/index.ts +2340 -0
  1282. package/src/lsp/lspmux.ts +233 -0
  1283. package/src/lsp/render.ts +692 -0
  1284. package/src/lsp/startup-events.ts +13 -0
  1285. package/src/lsp/types.ts +443 -0
  1286. package/src/lsp/utils.ts +699 -0
  1287. package/src/main.ts +1167 -0
  1288. package/src/mcp/client.ts +484 -0
  1289. package/src/mcp/config-writer.ts +225 -0
  1290. package/src/mcp/config.ts +365 -0
  1291. package/src/mcp/index.ts +29 -0
  1292. package/src/mcp/json-rpc.ts +92 -0
  1293. package/src/mcp/loader.ts +124 -0
  1294. package/src/mcp/manager.ts +1274 -0
  1295. package/src/mcp/oauth-discovery.ts +437 -0
  1296. package/src/mcp/oauth-flow.ts +435 -0
  1297. package/src/mcp/render.ts +126 -0
  1298. package/src/mcp/smithery-auth.ts +104 -0
  1299. package/src/mcp/smithery-connect.ts +145 -0
  1300. package/src/mcp/smithery-registry.ts +477 -0
  1301. package/src/mcp/timeout.ts +59 -0
  1302. package/src/mcp/tool-bridge.ts +424 -0
  1303. package/src/mcp/tool-cache.ts +117 -0
  1304. package/src/mcp/transports/http.ts +519 -0
  1305. package/src/mcp/transports/index.ts +6 -0
  1306. package/src/mcp/transports/stdio.ts +386 -0
  1307. package/src/mcp/types.ts +423 -0
  1308. package/src/memories/index.ts +2227 -0
  1309. package/src/memories/storage.ts +617 -0
  1310. package/src/memory-backend/index.ts +5 -0
  1311. package/src/memory-backend/local-backend.ts +30 -0
  1312. package/src/memory-backend/off-backend.ts +16 -0
  1313. package/src/memory-backend/resolve.ts +27 -0
  1314. package/src/memory-backend/types.ts +86 -0
  1315. package/src/mnemopi/backend.ts +374 -0
  1316. package/src/mnemopi/config.ts +160 -0
  1317. package/src/mnemopi/index.ts +3 -0
  1318. package/src/mnemopi/state.ts +549 -0
  1319. package/src/modes/acp/acp-agent.ts +2292 -0
  1320. package/src/modes/acp/acp-client-bridge.ts +154 -0
  1321. package/src/modes/acp/acp-event-mapper.ts +929 -0
  1322. package/src/modes/acp/acp-mode.ts +23 -0
  1323. package/src/modes/acp/index.ts +2 -0
  1324. package/src/modes/acp/terminal-auth.ts +37 -0
  1325. package/src/modes/components/agent-dashboard.ts +1201 -0
  1326. package/src/modes/components/assistant-message.ts +309 -0
  1327. package/src/modes/components/bash-execution.ts +220 -0
  1328. package/src/modes/components/bordered-loader.ts +41 -0
  1329. package/src/modes/components/branch-summary-message.ts +45 -0
  1330. package/src/modes/components/btw-panel.ts +104 -0
  1331. package/src/modes/components/compaction-summary-message.ts +51 -0
  1332. package/src/modes/components/copy-selector.ts +249 -0
  1333. package/src/modes/components/countdown-timer.ts +75 -0
  1334. package/src/modes/components/custom-editor.ts +267 -0
  1335. package/src/modes/components/custom-message.ts +65 -0
  1336. package/src/modes/components/diff.ts +266 -0
  1337. package/src/modes/components/dynamic-border.ts +25 -0
  1338. package/src/modes/components/error-banner.ts +33 -0
  1339. package/src/modes/components/eval-execution.ts +158 -0
  1340. package/src/modes/components/execution-shared.ts +102 -0
  1341. package/src/modes/components/extensions/extension-dashboard.ts +396 -0
  1342. package/src/modes/components/extensions/extension-list.ts +502 -0
  1343. package/src/modes/components/extensions/index.ts +9 -0
  1344. package/src/modes/components/extensions/inspector-panel.ts +317 -0
  1345. package/src/modes/components/extensions/state-manager.ts +632 -0
  1346. package/src/modes/components/extensions/types.ts +186 -0
  1347. package/src/modes/components/footer.ts +272 -0
  1348. package/src/modes/components/history-search.ts +280 -0
  1349. package/src/modes/components/hook-editor.ts +159 -0
  1350. package/src/modes/components/hook-input.ts +79 -0
  1351. package/src/modes/components/hook-message.ts +68 -0
  1352. package/src/modes/components/hook-selector.ts +660 -0
  1353. package/src/modes/components/index.ts +38 -0
  1354. package/src/modes/components/keybinding-hints.ts +65 -0
  1355. package/src/modes/components/login-dialog.ts +164 -0
  1356. package/src/modes/components/mcp-add-wizard.ts +1340 -0
  1357. package/src/modes/components/message-frame.ts +88 -0
  1358. package/src/modes/components/model-selector.ts +1213 -0
  1359. package/src/modes/components/oauth-selector.ts +342 -0
  1360. package/src/modes/components/omfg-panel.ts +141 -0
  1361. package/src/modes/components/plugin-selector.ts +95 -0
  1362. package/src/modes/components/plugin-settings.ts +728 -0
  1363. package/src/modes/components/queue-mode-selector.ts +56 -0
  1364. package/src/modes/components/read-tool-group.ts +267 -0
  1365. package/src/modes/components/segment-track.ts +52 -0
  1366. package/src/modes/components/session-observer-overlay.ts +852 -0
  1367. package/src/modes/components/session-selector.ts +571 -0
  1368. package/src/modes/components/settings-defs.ts +189 -0
  1369. package/src/modes/components/settings-selector.ts +647 -0
  1370. package/src/modes/components/show-images-selector.ts +45 -0
  1371. package/src/modes/components/skill-message.ts +90 -0
  1372. package/src/modes/components/status-line/context-thresholds.ts +79 -0
  1373. package/src/modes/components/status-line/git-utils.ts +42 -0
  1374. package/src/modes/components/status-line/index.ts +4 -0
  1375. package/src/modes/components/status-line/presets.ts +106 -0
  1376. package/src/modes/components/status-line/segments.ts +590 -0
  1377. package/src/modes/components/status-line/separators.ts +55 -0
  1378. package/src/modes/components/status-line/token-rate.ts +66 -0
  1379. package/src/modes/components/status-line/types.ts +93 -0
  1380. package/src/modes/components/status-line.ts +794 -0
  1381. package/src/modes/components/theme-selector.ts +63 -0
  1382. package/src/modes/components/thinking-selector.ts +52 -0
  1383. package/src/modes/components/tiny-title-download-progress.ts +90 -0
  1384. package/src/modes/components/tips.txt +18 -0
  1385. package/src/modes/components/todo-reminder.ts +40 -0
  1386. package/src/modes/components/tool-execution.ts +1017 -0
  1387. package/src/modes/components/transcript-container.ts +181 -0
  1388. package/src/modes/components/tree-selector.ts +978 -0
  1389. package/src/modes/components/ttsr-notification.ts +80 -0
  1390. package/src/modes/components/user-message-selector.ts +227 -0
  1391. package/src/modes/components/user-message.ts +52 -0
  1392. package/src/modes/components/visual-truncate.ts +63 -0
  1393. package/src/modes/components/welcome.ts +435 -0
  1394. package/src/modes/controllers/btw-controller.ts +105 -0
  1395. package/src/modes/controllers/command-controller-shared.ts +108 -0
  1396. package/src/modes/controllers/command-controller.ts +1546 -0
  1397. package/src/modes/controllers/event-controller.ts +926 -0
  1398. package/src/modes/controllers/extension-ui-controller.ts +933 -0
  1399. package/src/modes/controllers/input-controller.ts +947 -0
  1400. package/src/modes/controllers/mcp-command-controller.ts +1938 -0
  1401. package/src/modes/controllers/omfg-controller.ts +283 -0
  1402. package/src/modes/controllers/omfg-rule.ts +647 -0
  1403. package/src/modes/controllers/selector-controller.ts +1117 -0
  1404. package/src/modes/controllers/ssh-command-controller.ts +384 -0
  1405. package/src/modes/controllers/todo-command-controller.ts +485 -0
  1406. package/src/modes/data/emojis.json +1 -0
  1407. package/src/modes/emoji-autocomplete.ts +285 -0
  1408. package/src/modes/gradient-highlight.ts +87 -0
  1409. package/src/modes/image-references.ts +111 -0
  1410. package/src/modes/index.ts +34 -0
  1411. package/src/modes/interactive-mode.ts +3151 -0
  1412. package/src/modes/internal-url-autocomplete.ts +143 -0
  1413. package/src/modes/loop-limit.ts +140 -0
  1414. package/src/modes/magic-keywords.ts +20 -0
  1415. package/src/modes/markdown-prose.ts +247 -0
  1416. package/src/modes/oauth-manual-input.ts +42 -0
  1417. package/src/modes/orchestrate.ts +42 -0
  1418. package/src/modes/print-mode.ts +126 -0
  1419. package/src/modes/prompt-action-autocomplete.ts +260 -0
  1420. package/src/modes/rpc/host-tools.ts +186 -0
  1421. package/src/modes/rpc/host-uris.ts +235 -0
  1422. package/src/modes/rpc/rpc-client.ts +812 -0
  1423. package/src/modes/rpc/rpc-mode.ts +862 -0
  1424. package/src/modes/rpc/rpc-types.ts +378 -0
  1425. package/src/modes/runtime-init.ts +116 -0
  1426. package/src/modes/session-observer-registry.ts +146 -0
  1427. package/src/modes/setup-wizard/index.ts +88 -0
  1428. package/src/modes/setup-wizard/scenes/channels.ts +104 -0
  1429. package/src/modes/setup-wizard/scenes/glyph.ts +96 -0
  1430. package/src/modes/setup-wizard/scenes/outro.ts +35 -0
  1431. package/src/modes/setup-wizard/scenes/providers.ts +70 -0
  1432. package/src/modes/setup-wizard/scenes/sign-in.ts +213 -0
  1433. package/src/modes/setup-wizard/scenes/splash.ts +201 -0
  1434. package/src/modes/setup-wizard/scenes/telegram.ts +695 -0
  1435. package/src/modes/setup-wizard/scenes/theme.ts +299 -0
  1436. package/src/modes/setup-wizard/scenes/types.ts +48 -0
  1437. package/src/modes/setup-wizard/scenes/web-search.ts +128 -0
  1438. package/src/modes/setup-wizard/wizard-overlay.ts +275 -0
  1439. package/src/modes/shared.ts +47 -0
  1440. package/src/modes/theme/dark.json +95 -0
  1441. package/src/modes/theme/defaults/alabaster.json +93 -0
  1442. package/src/modes/theme/defaults/amethyst.json +96 -0
  1443. package/src/modes/theme/defaults/anthracite.json +93 -0
  1444. package/src/modes/theme/defaults/basalt.json +91 -0
  1445. package/src/modes/theme/defaults/birch.json +95 -0
  1446. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  1447. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  1448. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  1449. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  1450. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  1451. package/src/modes/theme/defaults/dark-copper.json +95 -0
  1452. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  1453. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  1454. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  1455. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  1456. package/src/modes/theme/defaults/dark-ember.json +95 -0
  1457. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  1458. package/src/modes/theme/defaults/dark-forest.json +96 -0
  1459. package/src/modes/theme/defaults/dark-github.json +105 -0
  1460. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  1461. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  1462. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  1463. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  1464. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  1465. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  1466. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  1467. package/src/modes/theme/defaults/dark-nord.json +97 -0
  1468. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  1469. package/src/modes/theme/defaults/dark-one.json +100 -0
  1470. package/src/modes/theme/defaults/dark-poimandres.json +142 -0
  1471. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  1472. package/src/modes/theme/defaults/dark-reef.json +91 -0
  1473. package/src/modes/theme/defaults/dark-retro.json +92 -0
  1474. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  1475. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  1476. package/src/modes/theme/defaults/dark-slate.json +95 -0
  1477. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  1478. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  1479. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  1480. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  1481. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  1482. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  1483. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  1484. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  1485. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  1486. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  1487. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  1488. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  1489. package/src/modes/theme/defaults/graphite.json +92 -0
  1490. package/src/modes/theme/defaults/index.ts +199 -0
  1491. package/src/modes/theme/defaults/light-arctic.json +107 -0
  1492. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  1493. package/src/modes/theme/defaults/light-canyon.json +91 -0
  1494. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  1495. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  1496. package/src/modes/theme/defaults/light-coral.json +95 -0
  1497. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  1498. package/src/modes/theme/defaults/light-dawn.json +90 -0
  1499. package/src/modes/theme/defaults/light-dunes.json +91 -0
  1500. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  1501. package/src/modes/theme/defaults/light-forest.json +100 -0
  1502. package/src/modes/theme/defaults/light-frost.json +95 -0
  1503. package/src/modes/theme/defaults/light-github.json +115 -0
  1504. package/src/modes/theme/defaults/light-glacier.json +91 -0
  1505. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  1506. package/src/modes/theme/defaults/light-haze.json +90 -0
  1507. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  1508. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  1509. package/src/modes/theme/defaults/light-lavender.json +95 -0
  1510. package/src/modes/theme/defaults/light-meadow.json +91 -0
  1511. package/src/modes/theme/defaults/light-mint.json +95 -0
  1512. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  1513. package/src/modes/theme/defaults/light-ocean.json +99 -0
  1514. package/src/modes/theme/defaults/light-one.json +99 -0
  1515. package/src/modes/theme/defaults/light-opal.json +91 -0
  1516. package/src/modes/theme/defaults/light-orchard.json +91 -0
  1517. package/src/modes/theme/defaults/light-paper.json +95 -0
  1518. package/src/modes/theme/defaults/light-poimandres.json +142 -0
  1519. package/src/modes/theme/defaults/light-prism.json +90 -0
  1520. package/src/modes/theme/defaults/light-retro.json +98 -0
  1521. package/src/modes/theme/defaults/light-sand.json +95 -0
  1522. package/src/modes/theme/defaults/light-savanna.json +91 -0
  1523. package/src/modes/theme/defaults/light-solarized.json +102 -0
  1524. package/src/modes/theme/defaults/light-soleil.json +90 -0
  1525. package/src/modes/theme/defaults/light-sunset.json +99 -0
  1526. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  1527. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  1528. package/src/modes/theme/defaults/light-wetland.json +91 -0
  1529. package/src/modes/theme/defaults/light-zenith.json +89 -0
  1530. package/src/modes/theme/defaults/limestone.json +94 -0
  1531. package/src/modes/theme/defaults/mahogany.json +97 -0
  1532. package/src/modes/theme/defaults/marble.json +93 -0
  1533. package/src/modes/theme/defaults/obsidian.json +91 -0
  1534. package/src/modes/theme/defaults/onyx.json +91 -0
  1535. package/src/modes/theme/defaults/pearl.json +93 -0
  1536. package/src/modes/theme/defaults/porcelain.json +91 -0
  1537. package/src/modes/theme/defaults/quartz.json +96 -0
  1538. package/src/modes/theme/defaults/sandstone.json +95 -0
  1539. package/src/modes/theme/defaults/titanium.json +90 -0
  1540. package/src/modes/theme/light.json +93 -0
  1541. package/src/modes/theme/mermaid-cache.ts +29 -0
  1542. package/src/modes/theme/shimmer.ts +224 -0
  1543. package/src/modes/theme/theme-schema.json +459 -0
  1544. package/src/modes/theme/theme.ts +2569 -0
  1545. package/src/modes/turn-budget.ts +31 -0
  1546. package/src/modes/types.ts +338 -0
  1547. package/src/modes/ultrathink.ts +41 -0
  1548. package/src/modes/utils/context-usage.ts +335 -0
  1549. package/src/modes/utils/copy-targets.ts +254 -0
  1550. package/src/modes/utils/hotkeys-markdown.ts +60 -0
  1551. package/src/modes/utils/keybinding-matchers.ts +51 -0
  1552. package/src/modes/utils/tools-markdown.ts +27 -0
  1553. package/src/modes/utils/ui-helpers.ts +772 -0
  1554. package/src/modes/workflow.ts +42 -0
  1555. package/src/plan-mode/approved-plan.ts +163 -0
  1556. package/src/plan-mode/plan-handoff.ts +37 -0
  1557. package/src/plan-mode/plan-protection.ts +31 -0
  1558. package/src/plan-mode/state.ts +6 -0
  1559. package/src/priority.json +41 -0
  1560. package/src/prompts/agents/designer.md +66 -0
  1561. package/src/prompts/agents/explore.md +58 -0
  1562. package/src/prompts/agents/frontmatter.md +11 -0
  1563. package/src/prompts/agents/init.md +33 -0
  1564. package/src/prompts/agents/librarian.md +120 -0
  1565. package/src/prompts/agents/oracle.md +55 -0
  1566. package/src/prompts/agents/plan.md +48 -0
  1567. package/src/prompts/agents/reviewer.md +140 -0
  1568. package/src/prompts/agents/task.md +16 -0
  1569. package/src/prompts/ci-green-request.md +38 -0
  1570. package/src/prompts/dry-balance-bench.md +8 -0
  1571. package/src/prompts/goals/goal-budget-limit.md +16 -0
  1572. package/src/prompts/goals/goal-continuation.md +28 -0
  1573. package/src/prompts/goals/goal-mode-active.md +23 -0
  1574. package/src/prompts/memories/consolidation.md +35 -0
  1575. package/src/prompts/memories/identity_review.md +38 -0
  1576. package/src/prompts/memories/post_turn_review.md +37 -0
  1577. package/src/prompts/memories/read-path.md +16 -0
  1578. package/src/prompts/memories/stage_one_input.md +6 -0
  1579. package/src/prompts/memories/stage_one_system.md +21 -0
  1580. package/src/prompts/review-custom-request.md +22 -0
  1581. package/src/prompts/review-headless-request.md +16 -0
  1582. package/src/prompts/review-request.md +69 -0
  1583. package/src/prompts/steering/user-interjection.md +10 -0
  1584. package/src/prompts/system/agent-creation-architect.md +50 -0
  1585. package/src/prompts/system/agent-creation-user.md +6 -0
  1586. package/src/prompts/system/auto-continue.md +1 -0
  1587. package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
  1588. package/src/prompts/system/auto-thinking-difficulty.md +12 -0
  1589. package/src/prompts/system/btw-user.md +8 -0
  1590. package/src/prompts/system/commit-message-system.md +2 -0
  1591. package/src/prompts/system/custom-system-prompt.md +64 -0
  1592. package/src/prompts/system/eager-todo.md +13 -0
  1593. package/src/prompts/system/empty-stop-retry.md +6 -0
  1594. package/src/prompts/system/irc-incoming.md +8 -0
  1595. package/src/prompts/system/memory-consolidation-system.md +8 -0
  1596. package/src/prompts/system/memory-extraction-system.md +26 -0
  1597. package/src/prompts/system/omfg-user.md +51 -0
  1598. package/src/prompts/system/orchestrate-notice.md +40 -0
  1599. package/src/prompts/system/plan-mode-active.md +116 -0
  1600. package/src/prompts/system/plan-mode-approved.md +25 -0
  1601. package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
  1602. package/src/prompts/system/plan-mode-reference.md +11 -0
  1603. package/src/prompts/system/plan-mode-subagent.md +34 -0
  1604. package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
  1605. package/src/prompts/system/project-prompt.md +52 -0
  1606. package/src/prompts/system/subagent-system-prompt.md +69 -0
  1607. package/src/prompts/system/subagent-user-prompt.md +3 -0
  1608. package/src/prompts/system/subagent-yield-reminder.md +12 -0
  1609. package/src/prompts/system/system-prompt.md +275 -0
  1610. package/src/prompts/system/tiny-title-system.md +8 -0
  1611. package/src/prompts/system/title-system.md +3 -0
  1612. package/src/prompts/system/ttsr-interrupt.md +7 -0
  1613. package/src/prompts/system/ttsr-tool-reminder.md +5 -0
  1614. package/src/prompts/system/ultrathink-notice.md +3 -0
  1615. package/src/prompts/system/web-search.md +25 -0
  1616. package/src/prompts/system/workflow-notice.md +70 -0
  1617. package/src/prompts/tools/apply-patch.md +65 -0
  1618. package/src/prompts/tools/ask.md +30 -0
  1619. package/src/prompts/tools/ast-edit.md +39 -0
  1620. package/src/prompts/tools/ast-grep.md +42 -0
  1621. package/src/prompts/tools/async-result.md +8 -0
  1622. package/src/prompts/tools/bash.md +39 -0
  1623. package/src/prompts/tools/browser.md +73 -0
  1624. package/src/prompts/tools/checkpoint.md +16 -0
  1625. package/src/prompts/tools/debug.md +34 -0
  1626. package/src/prompts/tools/eval.md +89 -0
  1627. package/src/prompts/tools/find.md +37 -0
  1628. package/src/prompts/tools/github.md +20 -0
  1629. package/src/prompts/tools/goal.md +18 -0
  1630. package/src/prompts/tools/image-gen.md +7 -0
  1631. package/src/prompts/tools/inspect-image-system.md +20 -0
  1632. package/src/prompts/tools/inspect-image.md +32 -0
  1633. package/src/prompts/tools/irc.md +49 -0
  1634. package/src/prompts/tools/job.md +19 -0
  1635. package/src/prompts/tools/lsp.md +42 -0
  1636. package/src/prompts/tools/memory-edit.md +8 -0
  1637. package/src/prompts/tools/patch.md +70 -0
  1638. package/src/prompts/tools/read.md +86 -0
  1639. package/src/prompts/tools/recall.md +5 -0
  1640. package/src/prompts/tools/reflect.md +5 -0
  1641. package/src/prompts/tools/render-mermaid.md +9 -0
  1642. package/src/prompts/tools/replace.md +36 -0
  1643. package/src/prompts/tools/resolve.md +9 -0
  1644. package/src/prompts/tools/retain.md +6 -0
  1645. package/src/prompts/tools/rewind.md +13 -0
  1646. package/src/prompts/tools/search-tool-bm25.md +40 -0
  1647. package/src/prompts/tools/search.md +25 -0
  1648. package/src/prompts/tools/ssh.md +35 -0
  1649. package/src/prompts/tools/task-summary.md +28 -0
  1650. package/src/prompts/tools/task.md +80 -0
  1651. package/src/prompts/tools/todo.md +58 -0
  1652. package/src/prompts/tools/web-search.md +10 -0
  1653. package/src/prompts/tools/write.md +14 -0
  1654. package/src/registry/agent-registry.ts +140 -0
  1655. package/src/sdk.ts +2374 -0
  1656. package/src/secrets/index.ts +116 -0
  1657. package/src/secrets/obfuscator.ts +277 -0
  1658. package/src/secrets/regex.ts +21 -0
  1659. package/src/session/agent-session.ts +9765 -0
  1660. package/src/session/agent-storage.ts +455 -0
  1661. package/src/session/artifacts.ts +135 -0
  1662. package/src/session/auth-broker-config.ts +102 -0
  1663. package/src/session/auth-storage.ts +27 -0
  1664. package/src/session/blob-store.ts +255 -0
  1665. package/src/session/client-bridge.ts +85 -0
  1666. package/src/session/history-storage.ts +348 -0
  1667. package/src/session/indexed-session-storage.ts +430 -0
  1668. package/src/session/messages.ts +575 -0
  1669. package/src/session/redis-session-storage.ts +170 -0
  1670. package/src/session/session-dump-format.ts +209 -0
  1671. package/src/session/session-manager.ts +3541 -0
  1672. package/src/session/session-storage.ts +529 -0
  1673. package/src/session/shake-types.ts +43 -0
  1674. package/src/session/sql-session-storage.ts +314 -0
  1675. package/src/session/streaming-output.ts +1093 -0
  1676. package/src/session/tool-choice-queue.ts +213 -0
  1677. package/src/session/yield-queue.ts +155 -0
  1678. package/src/slash-commands/acp-builtins.ts +46 -0
  1679. package/src/slash-commands/builtin-registry.ts +1726 -0
  1680. package/src/slash-commands/headless-plan.ts +142 -0
  1681. package/src/slash-commands/helpers/context-report.ts +39 -0
  1682. package/src/slash-commands/helpers/format.ts +46 -0
  1683. package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
  1684. package/src/slash-commands/helpers/mcp.ts +532 -0
  1685. package/src/slash-commands/helpers/parse.ts +85 -0
  1686. package/src/slash-commands/helpers/ssh.ts +195 -0
  1687. package/src/slash-commands/helpers/todo.ts +279 -0
  1688. package/src/slash-commands/helpers/usage-report.ts +93 -0
  1689. package/src/slash-commands/marketplace-install-parser.ts +99 -0
  1690. package/src/slash-commands/types.ts +139 -0
  1691. package/src/ssh/config-writer.ts +183 -0
  1692. package/src/ssh/connection-manager.ts +482 -0
  1693. package/src/ssh/ssh-executor.ts +133 -0
  1694. package/src/ssh/sshfs-mount.ts +140 -0
  1695. package/src/ssh/utils.ts +8 -0
  1696. package/src/stt/downloader.ts +71 -0
  1697. package/src/stt/index.ts +3 -0
  1698. package/src/stt/recorder.ts +351 -0
  1699. package/src/stt/setup.ts +52 -0
  1700. package/src/stt/stt-controller.ts +160 -0
  1701. package/src/stt/transcribe.py +70 -0
  1702. package/src/stt/transcriber.ts +91 -0
  1703. package/src/system-prompt.ts +591 -0
  1704. package/src/task/agents.ts +167 -0
  1705. package/src/task/commands.ts +131 -0
  1706. package/src/task/discovery.ts +129 -0
  1707. package/src/task/executor.ts +1659 -0
  1708. package/src/task/index.ts +1401 -0
  1709. package/src/task/name-generator.ts +1577 -0
  1710. package/src/task/output-manager.ts +99 -0
  1711. package/src/task/parallel.ts +116 -0
  1712. package/src/task/prometheus-command.ts +26 -0
  1713. package/src/task/render.ts +1211 -0
  1714. package/src/task/repair-args.ts +117 -0
  1715. package/src/task/simple-mode.ts +27 -0
  1716. package/src/task/subprocess-tool-registry.ts +88 -0
  1717. package/src/task/types.ts +320 -0
  1718. package/src/task/worktree.ts +506 -0
  1719. package/src/telemetry-export.ts +127 -0
  1720. package/src/thinking.ts +165 -0
  1721. package/src/tiny/compiled-runtime.ts +179 -0
  1722. package/src/tiny/device.ts +111 -0
  1723. package/src/tiny/dtype.ts +101 -0
  1724. package/src/tiny/models.ts +242 -0
  1725. package/src/tiny/text.ts +165 -0
  1726. package/src/tiny/title-client.ts +491 -0
  1727. package/src/tiny/title-protocol.ts +56 -0
  1728. package/src/tiny/worker.ts +566 -0
  1729. package/src/tool-discovery/mode.ts +24 -0
  1730. package/src/tool-discovery/tool-index.ts +256 -0
  1731. package/src/tools/approval.ts +189 -0
  1732. package/src/tools/archive-reader.ts +629 -0
  1733. package/src/tools/ask.ts +879 -0
  1734. package/src/tools/ast-edit.ts +595 -0
  1735. package/src/tools/ast-grep.ts +458 -0
  1736. package/src/tools/auto-generated-guard.ts +305 -0
  1737. package/src/tools/bash-command-fixup.ts +37 -0
  1738. package/src/tools/bash-interactive.ts +388 -0
  1739. package/src/tools/bash-interceptor.ts +67 -0
  1740. package/src/tools/bash-pty-selection.ts +14 -0
  1741. package/src/tools/bash-skill-urls.ts +248 -0
  1742. package/src/tools/bash.ts +1238 -0
  1743. package/src/tools/browser/attach.ts +175 -0
  1744. package/src/tools/browser/launch.ts +651 -0
  1745. package/src/tools/browser/readable.ts +95 -0
  1746. package/src/tools/browser/registry.ts +194 -0
  1747. package/src/tools/browser/render.ts +213 -0
  1748. package/src/tools/browser/tab-protocol.ts +105 -0
  1749. package/src/tools/browser/tab-supervisor.ts +577 -0
  1750. package/src/tools/browser/tab-worker-entry.ts +21 -0
  1751. package/src/tools/browser/tab-worker.ts +1058 -0
  1752. package/src/tools/browser.ts +316 -0
  1753. package/src/tools/checkpoint.ts +136 -0
  1754. package/src/tools/conflict-detect.ts +672 -0
  1755. package/src/tools/context.ts +39 -0
  1756. package/src/tools/debug.ts +1052 -0
  1757. package/src/tools/eval-backends.ts +27 -0
  1758. package/src/tools/eval-render.ts +765 -0
  1759. package/src/tools/eval.ts +568 -0
  1760. package/src/tools/fetch.ts +1586 -0
  1761. package/src/tools/file-recorder.ts +35 -0
  1762. package/src/tools/find.ts +571 -0
  1763. package/src/tools/fs-cache-invalidation.ts +28 -0
  1764. package/src/tools/gh-format.ts +12 -0
  1765. package/src/tools/gh-renderer.ts +428 -0
  1766. package/src/tools/gh.ts +3582 -0
  1767. package/src/tools/github-cache.ts +548 -0
  1768. package/src/tools/grouped-file-output.ts +103 -0
  1769. package/src/tools/image-gen.ts +1448 -0
  1770. package/src/tools/index.ts +516 -0
  1771. package/src/tools/inspect-image-renderer.ts +103 -0
  1772. package/src/tools/inspect-image.ts +166 -0
  1773. package/src/tools/irc.ts +294 -0
  1774. package/src/tools/job.ts +521 -0
  1775. package/src/tools/json-tree.ts +243 -0
  1776. package/src/tools/jtd-to-json-schema.ts +219 -0
  1777. package/src/tools/jtd-to-typescript.ts +136 -0
  1778. package/src/tools/jtd-utils.ts +102 -0
  1779. package/src/tools/list-limit.ts +40 -0
  1780. package/src/tools/match-line-format.ts +20 -0
  1781. package/src/tools/memory-edit.ts +59 -0
  1782. package/src/tools/memory-recall.ts +104 -0
  1783. package/src/tools/memory-reflect.ts +92 -0
  1784. package/src/tools/memory-render.ts +185 -0
  1785. package/src/tools/memory-retain.ts +91 -0
  1786. package/src/tools/output-meta.ts +754 -0
  1787. package/src/tools/output-schema-validator.ts +132 -0
  1788. package/src/tools/path-utils.ts +1031 -0
  1789. package/src/tools/plan-mode-guard.ts +81 -0
  1790. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  1791. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  1792. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  1793. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  1794. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  1795. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  1796. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  1797. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  1798. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  1799. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  1800. package/src/tools/puppeteer/10_stealth_plugins.txt +206 -0
  1801. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  1802. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  1803. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  1804. package/src/tools/read.ts +2557 -0
  1805. package/src/tools/render-mermaid.ts +69 -0
  1806. package/src/tools/render-utils.ts +836 -0
  1807. package/src/tools/renderers.ts +75 -0
  1808. package/src/tools/report-tool-issue.ts +528 -0
  1809. package/src/tools/resolve.ts +259 -0
  1810. package/src/tools/review.ts +253 -0
  1811. package/src/tools/search-tool-bm25.ts +338 -0
  1812. package/src/tools/search.ts +1475 -0
  1813. package/src/tools/sqlite-reader.ts +819 -0
  1814. package/src/tools/ssh.ts +347 -0
  1815. package/src/tools/todo.ts +856 -0
  1816. package/src/tools/tool-errors.ts +62 -0
  1817. package/src/tools/tool-result.ts +94 -0
  1818. package/src/tools/tool-timeouts.ts +30 -0
  1819. package/src/tools/tts.ts +133 -0
  1820. package/src/tools/write.ts +1085 -0
  1821. package/src/tools/yield.ts +260 -0
  1822. package/src/tui/code-cell.ts +221 -0
  1823. package/src/tui/file-list.ts +55 -0
  1824. package/src/tui/hyperlink.ts +161 -0
  1825. package/src/tui/index.ts +12 -0
  1826. package/src/tui/output-block.ts +288 -0
  1827. package/src/tui/status-line.ts +50 -0
  1828. package/src/tui/tree-list.ts +84 -0
  1829. package/src/tui/types.ts +15 -0
  1830. package/src/tui/utils.ts +103 -0
  1831. package/src/utils/changelog.ts +106 -0
  1832. package/src/utils/clipboard.ts +193 -0
  1833. package/src/utils/command-args.ts +76 -0
  1834. package/src/utils/commit-message-generator.ts +142 -0
  1835. package/src/utils/edit-mode.ts +41 -0
  1836. package/src/utils/event-bus.ts +33 -0
  1837. package/src/utils/external-editor.ts +65 -0
  1838. package/src/utils/file-display-mode.ts +45 -0
  1839. package/src/utils/file-mentions.ts +280 -0
  1840. package/src/utils/git.ts +1538 -0
  1841. package/src/utils/image-loading.ts +102 -0
  1842. package/src/utils/image-resize.ts +309 -0
  1843. package/src/utils/jj.ts +248 -0
  1844. package/src/utils/lang-from-path.ts +239 -0
  1845. package/src/utils/markit.ts +89 -0
  1846. package/src/utils/open.ts +55 -0
  1847. package/src/utils/session-color.ts +68 -0
  1848. package/src/utils/shell-snapshot.ts +187 -0
  1849. package/src/utils/sixel.ts +69 -0
  1850. package/src/utils/title-generator.ts +353 -0
  1851. package/src/utils/tool-choice.ts +33 -0
  1852. package/src/utils/tools-manager.ts +363 -0
  1853. package/src/web/kagi.ts +302 -0
  1854. package/src/web/parallel.ts +349 -0
  1855. package/src/web/scrapers/artifacthub.ts +207 -0
  1856. package/src/web/scrapers/arxiv.ts +83 -0
  1857. package/src/web/scrapers/aur.ts +162 -0
  1858. package/src/web/scrapers/biorxiv.ts +133 -0
  1859. package/src/web/scrapers/bluesky.ts +262 -0
  1860. package/src/web/scrapers/brew.ts +172 -0
  1861. package/src/web/scrapers/cheatsh.ts +68 -0
  1862. package/src/web/scrapers/chocolatey.ts +196 -0
  1863. package/src/web/scrapers/choosealicense.ts +95 -0
  1864. package/src/web/scrapers/cisa-kev.ts +87 -0
  1865. package/src/web/scrapers/clojars.ts +154 -0
  1866. package/src/web/scrapers/coingecko.ts +177 -0
  1867. package/src/web/scrapers/crates-io.ts +97 -0
  1868. package/src/web/scrapers/crossref.ts +136 -0
  1869. package/src/web/scrapers/devto.ts +147 -0
  1870. package/src/web/scrapers/discogs.ts +306 -0
  1871. package/src/web/scrapers/discourse.ts +197 -0
  1872. package/src/web/scrapers/dockerhub.ts +138 -0
  1873. package/src/web/scrapers/docs-rs.ts +653 -0
  1874. package/src/web/scrapers/fdroid.ts +134 -0
  1875. package/src/web/scrapers/firefox-addons.ts +191 -0
  1876. package/src/web/scrapers/flathub.ts +223 -0
  1877. package/src/web/scrapers/github-gist.ts +58 -0
  1878. package/src/web/scrapers/github.ts +452 -0
  1879. package/src/web/scrapers/gitlab.ts +401 -0
  1880. package/src/web/scrapers/go-pkg.ts +266 -0
  1881. package/src/web/scrapers/hackage.ts +140 -0
  1882. package/src/web/scrapers/hackernews.ts +189 -0
  1883. package/src/web/scrapers/hex.ts +105 -0
  1884. package/src/web/scrapers/huggingface.ts +321 -0
  1885. package/src/web/scrapers/iacr.ts +89 -0
  1886. package/src/web/scrapers/index.ts +252 -0
  1887. package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
  1888. package/src/web/scrapers/lemmy.ts +203 -0
  1889. package/src/web/scrapers/lobsters.ts +175 -0
  1890. package/src/web/scrapers/mastodon.ts +292 -0
  1891. package/src/web/scrapers/maven.ts +138 -0
  1892. package/src/web/scrapers/mdn.ts +173 -0
  1893. package/src/web/scrapers/metacpan.ts +222 -0
  1894. package/src/web/scrapers/musicbrainz.ts +250 -0
  1895. package/src/web/scrapers/npm.ts +98 -0
  1896. package/src/web/scrapers/nuget.ts +183 -0
  1897. package/src/web/scrapers/nvd.ts +222 -0
  1898. package/src/web/scrapers/ollama.ts +239 -0
  1899. package/src/web/scrapers/open-vsx.ts +106 -0
  1900. package/src/web/scrapers/opencorporates.ts +292 -0
  1901. package/src/web/scrapers/openlibrary.ts +336 -0
  1902. package/src/web/scrapers/orcid.ts +286 -0
  1903. package/src/web/scrapers/osv.ts +176 -0
  1904. package/src/web/scrapers/packagist.ts +160 -0
  1905. package/src/web/scrapers/pub-dev.ts +143 -0
  1906. package/src/web/scrapers/pubmed.ts +211 -0
  1907. package/src/web/scrapers/pypi.ts +112 -0
  1908. package/src/web/scrapers/rawg.ts +110 -0
  1909. package/src/web/scrapers/readthedocs.ts +120 -0
  1910. package/src/web/scrapers/reddit.ts +95 -0
  1911. package/src/web/scrapers/repology.ts +251 -0
  1912. package/src/web/scrapers/rfc.ts +201 -0
  1913. package/src/web/scrapers/rubygems.ts +103 -0
  1914. package/src/web/scrapers/searchcode.ts +189 -0
  1915. package/src/web/scrapers/sec-edgar.ts +261 -0
  1916. package/src/web/scrapers/semantic-scholar.ts +171 -0
  1917. package/src/web/scrapers/snapcraft.ts +187 -0
  1918. package/src/web/scrapers/sourcegraph.ts +336 -0
  1919. package/src/web/scrapers/spdx.ts +108 -0
  1920. package/src/web/scrapers/spotify.ts +198 -0
  1921. package/src/web/scrapers/stackoverflow.ts +120 -0
  1922. package/src/web/scrapers/terraform.ts +277 -0
  1923. package/src/web/scrapers/tldr.ts +47 -0
  1924. package/src/web/scrapers/twitter.ts +93 -0
  1925. package/src/web/scrapers/types.ts +318 -0
  1926. package/src/web/scrapers/utils.ts +109 -0
  1927. package/src/web/scrapers/vimeo.ts +133 -0
  1928. package/src/web/scrapers/vscode-marketplace.ts +187 -0
  1929. package/src/web/scrapers/w3c.ts +156 -0
  1930. package/src/web/scrapers/wikidata.ts +344 -0
  1931. package/src/web/scrapers/wikipedia.ts +84 -0
  1932. package/src/web/scrapers/youtube.ts +319 -0
  1933. package/src/web/search/index.ts +292 -0
  1934. package/src/web/search/provider.ts +157 -0
  1935. package/src/web/search/providers/anthropic.ts +302 -0
  1936. package/src/web/search/providers/base.ts +88 -0
  1937. package/src/web/search/providers/brave.ts +149 -0
  1938. package/src/web/search/providers/codex.ts +557 -0
  1939. package/src/web/search/providers/exa.ts +309 -0
  1940. package/src/web/search/providers/gemini.ts +455 -0
  1941. package/src/web/search/providers/jina.ts +101 -0
  1942. package/src/web/search/providers/kagi.ts +79 -0
  1943. package/src/web/search/providers/kimi.ts +171 -0
  1944. package/src/web/search/providers/parallel.ts +210 -0
  1945. package/src/web/search/providers/perplexity.ts +579 -0
  1946. package/src/web/search/providers/searxng.ts +309 -0
  1947. package/src/web/search/providers/synthetic.ts +110 -0
  1948. package/src/web/search/providers/tavily.ts +173 -0
  1949. package/src/web/search/providers/utils.ts +128 -0
  1950. package/src/web/search/providers/zai.ts +320 -0
  1951. package/src/web/search/render.ts +267 -0
  1952. package/src/web/search/types.ts +478 -0
  1953. package/src/web/search/utils.ts +17 -0
  1954. package/src/workspace-tree.ts +286 -0
@@ -0,0 +1,2557 @@
1
+ import { Database } from "bun:sqlite";
2
+ import * as fs from "node:fs/promises";
3
+ import * as path from "node:path";
4
+ import type { AgentTool, AgentToolContext, AgentToolResult, AgentToolUpdateCallback } from "@prometheus-ai/agent-core";
5
+ import type { ImageContent, TextContent } from "@prometheus-ai/ai";
6
+ import { formatHashlineHeader, formatNumberedLine, formatNumberedLines } from "@prometheus-ai/hashline";
7
+ import { glob, type SummaryResult, summarizeCode } from "@prometheus-ai/natives";
8
+ import type { Component } from "@prometheus-ai/tui";
9
+ import { Text } from "@prometheus-ai/tui";
10
+ import { getRemoteDir, logger, prompt, readImageMetadata, untilAborted } from "@prometheus-ai/utils";
11
+ import * as z from "zod/v4";
12
+ import { getFileSnapshotStore, recordFileSnapshot } from "../edit/file-snapshot-store";
13
+ import { normalizeToLF } from "../edit/normalize";
14
+ import { isNotebookPath, readEditableNotebookText } from "../edit/notebook";
15
+ import type { RenderResultOptions } from "../extensibility/custom-tools/types";
16
+ import { InternalUrlRouter } from "../internal-urls";
17
+ import { parseInternalUrl } from "../internal-urls/parse";
18
+ import type { InternalUrl } from "../internal-urls/types";
19
+ import { getLanguageFromPath, type Theme } from "../modes/theme/theme";
20
+ import readDescription from "../prompts/tools/read.md" with { type: "text" };
21
+ import type { ToolSession } from "../sdk";
22
+ import {
23
+ DEFAULT_MAX_BYTES,
24
+ DEFAULT_MAX_LINES,
25
+ noTruncResult,
26
+ type TruncationResult,
27
+ truncateHead,
28
+ truncateHeadBytes,
29
+ truncateLine,
30
+ } from "../session/streaming-output";
31
+ import { fileHyperlink, renderCodeCell, renderMarkdownCell, renderStatusLine, tryResolveInternalUrlSync } from "../tui";
32
+ import { CachedOutputBlock, markFramedBlockComponent } from "../tui/output-block";
33
+ import { resolveFileDisplayMode } from "../utils/file-display-mode";
34
+ import { ImageInputTooLargeError, loadImageInput, MAX_IMAGE_INPUT_BYTES } from "../utils/image-loading";
35
+ import { convertFileWithMarkit } from "../utils/markit";
36
+ import { buildDirectoryTree, type DirectoryTree } from "../workspace-tree";
37
+ import { type ArchiveReader, openArchive, parseArchivePathCandidates } from "./archive-reader";
38
+ import {
39
+ type ConflictEntry,
40
+ type ConflictScope,
41
+ formatConflictSummary,
42
+ formatConflictWarning,
43
+ getConflictHistory,
44
+ parseConflictUri,
45
+ renderConflictRegion,
46
+ scanConflictLines,
47
+ scanFileForConflicts,
48
+ } from "./conflict-detect";
49
+ import {
50
+ executeReadUrl,
51
+ isReadableUrlPath,
52
+ loadReadUrlCacheEntry,
53
+ parseReadUrlTarget,
54
+ type ReadUrlToolDetails,
55
+ renderReadUrlCall,
56
+ renderReadUrlResult,
57
+ } from "./fetch";
58
+ import { applyListLimit } from "./list-limit";
59
+ import {
60
+ formatFullOutputReference,
61
+ formatStyledTruncationWarning,
62
+ type OutputMeta,
63
+ resolveOutputMaxColumns,
64
+ stripOutputNotice,
65
+ } from "./output-meta";
66
+ import {
67
+ expandPath,
68
+ formatPathRelativeToCwd,
69
+ type LineRange,
70
+ parseLineRanges,
71
+ resolveReadPath,
72
+ splitDelimitedPathEntry,
73
+ splitInternalUrlSel,
74
+ splitPathAndSel,
75
+ } from "./path-utils";
76
+ import { formatBytes, replaceTabs, shortenPath, wrapBrackets } from "./render-utils";
77
+ import {
78
+ executeReadQuery,
79
+ getRowByKey,
80
+ getRowByRowId,
81
+ getTableSchema,
82
+ isSqliteFile,
83
+ listTables,
84
+ parseSqlitePathCandidates,
85
+ parseSqliteSelector,
86
+ queryRows,
87
+ renderRow,
88
+ renderSchema,
89
+ renderTable,
90
+ renderTableList,
91
+ resolveTableRowLookup,
92
+ } from "./sqlite-reader";
93
+ import { ToolAbortError, ToolError, throwIfAborted } from "./tool-errors";
94
+ import { toolResult } from "./tool-result";
95
+
96
+ // Document types converted to markdown via markit.
97
+ const CONVERTIBLE_EXTENSIONS = new Set([".pdf", ".doc", ".docx", ".ppt", ".pptx", ".xls", ".xlsx", ".rtf", ".epub"]);
98
+
99
+ const MAX_SUMMARY_BYTES = 2 * 1024 * 1024;
100
+ const MAX_SUMMARY_LINES = 20_000;
101
+ /**
102
+ * Per-line column cap for file reads. Lines wider than the value of
103
+ * `tools.outputMaxColumns` are ellipsis-truncated at display time; the file
104
+ * on disk is unchanged. Shared with the streaming sink path so one setting
105
+ * covers `bash`/`ssh`/`python`/`js eval` and `read` uniformly.
106
+ */
107
+ const PROSE_SUMMARY_EXTENSIONS = new Set([".md", ".txt"]);
108
+ // Remote mount path prefix (sshfs mounts) - skip fuzzy matching to avoid hangs
109
+ const REMOTE_MOUNT_PREFIX = getRemoteDir() + path.sep;
110
+
111
+ function isRemoteMountPath(absolutePath: string): boolean {
112
+ return absolutePath.startsWith(REMOTE_MOUNT_PREFIX);
113
+ }
114
+
115
+ function prependLineNumbers(text: string, startNum: number): string {
116
+ const textLines = text.split("\n");
117
+ return textLines.map((line, i) => `${startNum + i}|${line}`).join("\n");
118
+ }
119
+
120
+ interface HashlineHeaderContext {
121
+ header: string;
122
+ tag: string;
123
+ fullText?: string;
124
+ }
125
+
126
+ function recordFullHashlineContext(
127
+ session: ToolSession,
128
+ absolutePath: string | undefined,
129
+ displayPath: string,
130
+ fullText: string,
131
+ ): HashlineHeaderContext | undefined {
132
+ if (!absolutePath || !path.isAbsolute(absolutePath)) return undefined;
133
+ const normalized = normalizeToLF(fullText);
134
+ const tag = getFileSnapshotStore(session).record(absolutePath, normalized);
135
+ return {
136
+ header: formatHashlineHeader(displayPath, tag),
137
+ tag,
138
+ fullText: normalized,
139
+ };
140
+ }
141
+
142
+ async function readHashlineHeaderContext(
143
+ session: ToolSession,
144
+ absolutePath: string,
145
+ cwd: string,
146
+ ): Promise<HashlineHeaderContext> {
147
+ const fullText = await Bun.file(absolutePath).text();
148
+ const context = recordFullHashlineContext(
149
+ session,
150
+ absolutePath,
151
+ formatPathRelativeToCwd(absolutePath, cwd),
152
+ fullText,
153
+ );
154
+ if (!context) throw new ToolError(`Cannot record hashline snapshot for non-absolute path: ${absolutePath}`);
155
+ return context;
156
+ }
157
+
158
+ function hashlineHeaderContext(displayPath: string, tag: string): HashlineHeaderContext {
159
+ return { header: formatHashlineHeader(displayPath, tag), tag };
160
+ }
161
+
162
+ function prependHashlineHeader(text: string, context: HashlineHeaderContext | undefined): string {
163
+ return context ? `${context.header}\n${text}` : text;
164
+ }
165
+
166
+ function formatTextWithMode(
167
+ text: string,
168
+ startNum: number,
169
+ shouldAddHashLines: boolean,
170
+ shouldAddLineNumbers: boolean,
171
+ ): string {
172
+ if (shouldAddHashLines) return formatNumberedLines(text, startNum);
173
+ if (shouldAddLineNumbers) return prependLineNumbers(text, startNum);
174
+ return text;
175
+ }
176
+
177
+ const BRACE_PAIRS: Record<string, string> = { "{": "}", "(": ")", "[": "]" };
178
+ const BRACE_TAIL_TRAILING_RE = /^[;,)\]}]*$/;
179
+
180
+ /**
181
+ * Decide whether the kept lines surrounding an elided range collapse to a
182
+ * single brace-pair line in the rendered summary. Returns true when the head
183
+ * line ends with `{` / `(` / `[` and the tail line is the matching closer
184
+ * (optionally followed by terminating punctuation like `;`, `,`, or further
185
+ * closers — e.g. `};`, `})`, `]);`).
186
+ */
187
+ function canMergeBracePair(headLine: string, tailLine: string): boolean {
188
+ const head = headLine.trimEnd();
189
+ const tail = tailLine.trim();
190
+ const opener = head.slice(-1);
191
+ const closer = BRACE_PAIRS[opener];
192
+ if (!closer) return false;
193
+ if (!tail.startsWith(closer)) return false;
194
+ return BRACE_TAIL_TRAILING_RE.test(tail.slice(closer.length));
195
+ }
196
+
197
+ function formatSingleLine(
198
+ line: number,
199
+ text: string,
200
+ shouldAddHashLines: boolean,
201
+ shouldAddLineNumbers: boolean,
202
+ ): string {
203
+ if (shouldAddHashLines) return formatNumberedLine(line, text);
204
+ if (shouldAddLineNumbers) return `${line}|${text}`;
205
+ return text;
206
+ }
207
+
208
+ function formatMergedBraceLine(
209
+ startLine: number,
210
+ endLine: number,
211
+ headText: string,
212
+ tailText: string,
213
+ shouldAddHashLines: boolean,
214
+ shouldAddLineNumbers: boolean,
215
+ ): { model: string; display: string } {
216
+ const merged = `${headText.trimEnd()} .. ${tailText.trim()}`;
217
+ if (shouldAddHashLines) {
218
+ return { model: `${startLine}-${endLine}:${merged}`, display: merged };
219
+ }
220
+ if (shouldAddLineNumbers) {
221
+ return { model: `${startLine}-${endLine}|${merged}`, display: merged };
222
+ }
223
+ return { model: merged, display: merged };
224
+ }
225
+
226
+ function countTextLines(text: string): number {
227
+ if (text.length === 0) return 0;
228
+ return text.split("\n").length;
229
+ }
230
+
231
+ /** Inclusive line range describing one elided span in a structural summary. */
232
+ interface ElidedRange {
233
+ start: number;
234
+ end: number;
235
+ }
236
+
237
+ /** Sample ranges shown in the footer to demonstrate the multi-range syntax. */
238
+ const FOOTER_RANGE_SAMPLES = 2;
239
+
240
+ /**
241
+ * Footer appended to summarized reads telling the model how to recover the
242
+ * elided body. Without this hint, agents either ignore the `...`/`{ .. }`
243
+ * markers or burn a turn guessing the right selector (see issue #1046). The
244
+ * footer demonstrates the multi-range selector syntax with concrete sample
245
+ * ranges drawn from the actual elision so the model re-reads only what it
246
+ * needs instead of falling back to `:raw` or whole-file reads.
247
+ */
248
+ function formatSummaryElisionFooter(
249
+ readPath: string,
250
+ elidedRanges: ReadonlyArray<ElidedRange>,
251
+ elidedLines: number,
252
+ ): string {
253
+ if (elidedRanges.length === 0) return "";
254
+ const lineWord = elidedLines === 1 ? "line" : "lines";
255
+ const sampleCount = Math.min(elidedRanges.length, FOOTER_RANGE_SAMPLES);
256
+ const selector = elidedRanges
257
+ .slice(0, sampleCount)
258
+ .map(r => `${r.start}-${r.end}`)
259
+ .join(",");
260
+ const example = `${readPath}:${selector}`;
261
+ const tail = elidedRanges.length > sampleCount ? `, e.g. ${example}` : ` with ${example}`;
262
+ return `[${elidedLines} ${lineWord} elided; re-read needed ranges${tail}]`;
263
+ }
264
+ const READ_CHUNK_SIZE = 8 * 1024;
265
+
266
+ /**
267
+ * Context lines added around an explicit range read. Anchor-stale failures
268
+ * cluster on edits whose anchors land just outside the most recent read
269
+ * window, but the data (`scripts/session-stats/analyze_selector_reads.py`)
270
+ * shows most follow-up reads are disjoint hops, not adjacent extensions —
271
+ * so symmetric padding rarely pays for itself.
272
+ *
273
+ * Leading=1 catches accidental single-line reads where the anchor is the
274
+ * line immediately above the requested start. Trailing=3 buffers the
275
+ * common case where the agent asks for a narrow range and then needs the
276
+ * next few lines to disambiguate an anchor.
277
+ */
278
+ const RANGE_LEADING_CONTEXT_LINES = 1;
279
+ const RANGE_TRAILING_CONTEXT_LINES = 3;
280
+
281
+ /**
282
+ * Expand a [start, end) range with leading/trailing context lines on the
283
+ * sides where the user actually constrained the range. A start of 0 (no
284
+ * explicit offset) does not get leading context — that's already an
285
+ * open-ended read from the top.
286
+ */
287
+ function expandRangeWithContext(
288
+ requestedStart: number,
289
+ requestedEnd: number,
290
+ totalLines: number,
291
+ expandStart: boolean,
292
+ expandEnd: boolean,
293
+ ): { startLine: number; endLine: number } {
294
+ return {
295
+ startLine: expandStart ? Math.max(0, requestedStart - RANGE_LEADING_CONTEXT_LINES) : requestedStart,
296
+ endLine: expandEnd ? Math.min(totalLines, requestedEnd + RANGE_TRAILING_CONTEXT_LINES) : requestedEnd,
297
+ };
298
+ }
299
+
300
+ async function streamLinesFromFile(
301
+ filePath: string,
302
+ startLine: number,
303
+ maxLinesToCollect: number,
304
+ maxBytes: number,
305
+ selectedLineLimit: number | null,
306
+ signal?: AbortSignal,
307
+ ): Promise<{
308
+ lines: string[];
309
+ totalFileLines: number;
310
+ collectedBytes: number;
311
+ stoppedByByteLimit: boolean;
312
+ firstLinePreview?: { text: string; bytes: number };
313
+ firstLineByteLength?: number;
314
+ selectedBytesTotal: number;
315
+ }> {
316
+ const bufferChunk = Buffer.allocUnsafe(READ_CHUNK_SIZE);
317
+ const collectedLines: string[] = [];
318
+ let lineIndex = 0;
319
+ let collectedBytes = 0;
320
+ let stoppedByByteLimit = false;
321
+ let doneCollecting = false;
322
+ let fileHandle: fs.FileHandle | null = null;
323
+ let currentLineLength = 0;
324
+ let currentLineChunks: Buffer[] = [];
325
+ let sawAnyByte = false;
326
+ let endedWithNewline = false;
327
+ let firstLinePreviewBytes = 0;
328
+ const firstLinePreviewChunks: Buffer[] = [];
329
+ let firstLineByteLength: number | undefined;
330
+ let selectedBytesTotal = 0;
331
+ let selectedLinesSeen = 0;
332
+ let captureLine = false;
333
+ let discardLineChunks = false;
334
+ let lineCaptureLimit = 0;
335
+
336
+ const setupLineState = () => {
337
+ captureLine = !doneCollecting && lineIndex >= startLine;
338
+ discardLineChunks = !captureLine;
339
+ if (captureLine) {
340
+ const separatorBytes = collectedLines.length > 0 ? 1 : 0;
341
+ lineCaptureLimit = maxBytes - collectedBytes - separatorBytes;
342
+ if (lineCaptureLimit <= 0) {
343
+ discardLineChunks = true;
344
+ }
345
+ } else {
346
+ lineCaptureLimit = 0;
347
+ }
348
+ };
349
+
350
+ const decodeLine = (): string => {
351
+ if (currentLineLength === 0) return "";
352
+ if (currentLineChunks.length === 1 && currentLineChunks[0]?.length === currentLineLength) {
353
+ return currentLineChunks[0].toString("utf-8");
354
+ }
355
+ return Buffer.concat(currentLineChunks, currentLineLength).toString("utf-8");
356
+ };
357
+
358
+ const maybeCapturePreview = (segment: Uint8Array) => {
359
+ if (doneCollecting || lineIndex < startLine || collectedLines.length !== 0) return;
360
+ if (firstLinePreviewBytes >= maxBytes || segment.length === 0) return;
361
+ const remaining = maxBytes - firstLinePreviewBytes;
362
+ const slice = segment.length > remaining ? segment.subarray(0, remaining) : segment;
363
+ if (slice.length === 0) return;
364
+ firstLinePreviewChunks.push(Buffer.from(slice));
365
+ firstLinePreviewBytes += slice.length;
366
+ };
367
+
368
+ const appendSegment = (segment: Uint8Array) => {
369
+ currentLineLength += segment.length;
370
+ maybeCapturePreview(segment);
371
+ if (!captureLine || discardLineChunks || segment.length === 0) return;
372
+ if (currentLineLength <= lineCaptureLimit) {
373
+ currentLineChunks.push(Buffer.from(segment));
374
+ } else {
375
+ discardLineChunks = true;
376
+ }
377
+ };
378
+
379
+ const finalizeLine = () => {
380
+ if (lineIndex >= startLine && (selectedLineLimit === null || selectedLinesSeen < selectedLineLimit)) {
381
+ selectedBytesTotal += currentLineLength + (selectedLinesSeen > 0 ? 1 : 0);
382
+ selectedLinesSeen++;
383
+ }
384
+
385
+ if (!doneCollecting && lineIndex >= startLine) {
386
+ const separatorBytes = collectedLines.length > 0 ? 1 : 0;
387
+ if (collectedLines.length >= maxLinesToCollect) {
388
+ doneCollecting = true;
389
+ } else if (collectedLines.length === 0 && currentLineLength > maxBytes) {
390
+ stoppedByByteLimit = true;
391
+ doneCollecting = true;
392
+ if (firstLineByteLength === undefined) {
393
+ firstLineByteLength = currentLineLength;
394
+ }
395
+ } else if (collectedLines.length > 0 && collectedBytes + separatorBytes + currentLineLength > maxBytes) {
396
+ stoppedByByteLimit = true;
397
+ doneCollecting = true;
398
+ } else {
399
+ const lineText = decodeLine();
400
+ collectedLines.push(lineText);
401
+ collectedBytes += separatorBytes + currentLineLength;
402
+ if (firstLineByteLength === undefined) {
403
+ firstLineByteLength = currentLineLength;
404
+ }
405
+ if (collectedBytes > maxBytes) {
406
+ stoppedByByteLimit = true;
407
+ doneCollecting = true;
408
+ } else if (collectedLines.length >= maxLinesToCollect) {
409
+ doneCollecting = true;
410
+ }
411
+ }
412
+ } else if (lineIndex >= startLine && firstLineByteLength === undefined) {
413
+ firstLineByteLength = currentLineLength;
414
+ }
415
+
416
+ lineIndex++;
417
+ currentLineLength = 0;
418
+ currentLineChunks = [];
419
+ setupLineState();
420
+ };
421
+
422
+ setupLineState();
423
+
424
+ try {
425
+ fileHandle = await fs.open(filePath, "r");
426
+
427
+ while (true) {
428
+ throwIfAborted(signal);
429
+ const { bytesRead } = await fileHandle.read(bufferChunk, 0, bufferChunk.length, null);
430
+ if (bytesRead === 0) break;
431
+
432
+ sawAnyByte = true;
433
+ const chunk = bufferChunk.subarray(0, bytesRead);
434
+ endedWithNewline = chunk[bytesRead - 1] === 0x0a;
435
+
436
+ let start = 0;
437
+ for (let i = 0; i < chunk.length; i++) {
438
+ if (chunk[i] === 0x0a) {
439
+ const segment = chunk.subarray(start, i);
440
+ if (segment.length > 0) {
441
+ appendSegment(segment);
442
+ }
443
+ finalizeLine();
444
+ start = i + 1;
445
+ }
446
+ }
447
+
448
+ if (start < chunk.length) {
449
+ appendSegment(chunk.subarray(start));
450
+ }
451
+ }
452
+ } finally {
453
+ if (fileHandle) {
454
+ await fileHandle.close();
455
+ }
456
+ }
457
+
458
+ if (endedWithNewline || currentLineLength > 0 || !sawAnyByte) {
459
+ finalizeLine();
460
+ }
461
+
462
+ let firstLinePreview: { text: string; bytes: number } | undefined;
463
+ if (firstLinePreviewBytes > 0) {
464
+ const { text, bytes } = truncateHeadBytes(Buffer.concat(firstLinePreviewChunks, firstLinePreviewBytes), maxBytes);
465
+ firstLinePreview = { text, bytes };
466
+ }
467
+
468
+ return {
469
+ lines: collectedLines,
470
+ totalFileLines: lineIndex,
471
+ collectedBytes,
472
+ stoppedByByteLimit,
473
+ firstLinePreview,
474
+ firstLineByteLength,
475
+ selectedBytesTotal,
476
+ };
477
+ }
478
+
479
+ // Maximum image file size (20MB) - larger images will be rejected to prevent OOM during serialization
480
+ const MAX_IMAGE_SIZE = MAX_IMAGE_INPUT_BYTES;
481
+ const GLOB_TIMEOUT_MS = 5000;
482
+
483
+ function isNotFoundError(error: unknown): boolean {
484
+ if (!error || typeof error !== "object") return false;
485
+ const code = (error as { code?: string }).code;
486
+ return code === "ENOENT" || code === "ENOTDIR";
487
+ }
488
+
489
+ /**
490
+ * Attempt to resolve a non-existent path by finding a unique suffix match within the workspace.
491
+ * Uses a glob suffix pattern so the native engine handles matching directly.
492
+ * Returns null when 0 or >1 candidates match (ambiguous = no auto-resolution).
493
+ */
494
+ async function findUniqueSuffixMatch(
495
+ rawPath: string,
496
+ cwd: string,
497
+ signal?: AbortSignal,
498
+ ): Promise<{ absolutePath: string; displayPath: string } | null> {
499
+ const normalized = rawPath.replace(/\\/g, "/").replace(/^\.\//, "").replace(/\/+$/, "");
500
+ if (!normalized) return null;
501
+
502
+ const timeoutSignal = AbortSignal.timeout(GLOB_TIMEOUT_MS);
503
+ const combinedSignal = signal ? AbortSignal.any([signal, timeoutSignal]) : timeoutSignal;
504
+
505
+ let matches: string[];
506
+ try {
507
+ const result = await untilAborted(combinedSignal, () =>
508
+ glob({
509
+ pattern: `**/${normalized}`,
510
+ path: cwd,
511
+ // No fileType filter: matches both files and directories
512
+ hidden: true,
513
+ }),
514
+ );
515
+ matches = result.matches.map(m => m.path);
516
+ } catch (error) {
517
+ if (error instanceof Error && error.name === "AbortError") {
518
+ if (!signal?.aborted) return null; // timeout — give up silently
519
+ throw new ToolAbortError();
520
+ }
521
+ return null;
522
+ }
523
+
524
+ if (matches.length !== 1) return null;
525
+
526
+ return {
527
+ absolutePath: path.resolve(cwd, matches[0]),
528
+ displayPath: matches[0],
529
+ };
530
+ }
531
+
532
+ function decodeUtf8Text(bytes: Uint8Array): string | null {
533
+ for (const byte of bytes) {
534
+ if (byte === 0) return null;
535
+ }
536
+
537
+ try {
538
+ return new TextDecoder("utf-8", { fatal: true }).decode(bytes);
539
+ } catch {
540
+ return null;
541
+ }
542
+ }
543
+
544
+ function prependSuffixResolutionNotice(text: string, suffixResolution?: { from: string; to: string }): string {
545
+ if (!suffixResolution) return text;
546
+
547
+ const notice = `[Path '${suffixResolution.from}' not found; resolved to '${suffixResolution.to}' via suffix match]`;
548
+ return text ? `${notice}\n${text}` : notice;
549
+ }
550
+
551
+ const readSchema = z
552
+ .object({
553
+ path: z.string().describe('path or url; append :<sel> for line ranges or raw mode (e.g. "src/foo.ts:50-100")'),
554
+ })
555
+ .strict();
556
+
557
+ export type ReadToolInput = z.infer<typeof readSchema>;
558
+
559
+ export interface ReadToolDetails {
560
+ kind?: "file" | "url";
561
+ truncation?: TruncationResult;
562
+ isDirectory?: boolean;
563
+ resolvedPath?: string;
564
+ suffixResolution?: { from: string; to: string };
565
+ url?: string;
566
+ finalUrl?: string;
567
+ contentType?: string;
568
+ method?: string;
569
+ notes?: string[];
570
+ meta?: OutputMeta;
571
+ /** Raw text + start line for user-visible TUI rendering, set when content is text-like.
572
+ * Mirrors the same lines the model receives but without hashline/line-number prefixes,
573
+ * so the TUI can render the file content with its own gutter without re-parsing the formatted text. */
574
+ displayContent?: { text: string; startLine: number };
575
+ summary?: { lines: number; elidedSpans: number; elidedLines: number };
576
+ /** Number of unresolved git conflicts surfaced by this read (TUI uses for inline `⚠ N` badge). */
577
+ conflictCount?: number;
578
+ }
579
+
580
+ type ReadParams = ReadToolInput;
581
+
582
+ /** Parsed representation of a path-embedded selector. */
583
+ type ParsedSelector =
584
+ | { kind: "none" }
585
+ | { kind: "raw" }
586
+ | { kind: "conflicts" }
587
+ | { kind: "lines"; ranges: [LineRange, ...LineRange[]]; raw?: boolean };
588
+
589
+ /** Returns true when the selector requested verbatim/raw output (alone or combined with a range). */
590
+ function isRawSelector(parsed: ParsedSelector): boolean {
591
+ return parsed.kind === "raw" || (parsed.kind === "lines" && parsed.raw === true);
592
+ }
593
+
594
+ /** Returns true when the selector requested multiple line ranges. */
595
+ function isMultiRange(parsed: ParsedSelector): boolean {
596
+ return parsed.kind === "lines" && parsed.ranges.length > 1;
597
+ }
598
+
599
+ function parseSel(sel: string | undefined): ParsedSelector {
600
+ if (!sel || sel.length === 0) return { kind: "none" };
601
+
602
+ // Compound selector: `1-50:raw` or `raw:1-50`. Split into chunks and accept
603
+ // any combination of one line range (possibly multi) and the literal `raw`.
604
+ if (sel.includes(":")) {
605
+ const chunks = sel.split(":");
606
+ if (chunks.length === 2) {
607
+ const [a, b] = chunks as [string, string];
608
+ const aIsRaw = a.toLowerCase() === "raw";
609
+ const bIsRaw = b.toLowerCase() === "raw";
610
+ const rangeChunk = aIsRaw ? b : bIsRaw ? a : null;
611
+ const rawChunk = aIsRaw ? a : bIsRaw ? b : null;
612
+ if (rangeChunk !== null && rawChunk !== null) {
613
+ const ranges = parseLineRanges(rangeChunk);
614
+ if (ranges) {
615
+ return { kind: "lines", ranges, raw: true };
616
+ }
617
+ }
618
+ }
619
+ // Unrecognized compound — fall through (sqlite/archive/url consume their own colon syntax).
620
+ return { kind: "none" };
621
+ }
622
+
623
+ if (sel.toLowerCase() === "raw") return { kind: "raw" };
624
+ if (sel.toLowerCase() === "conflicts") return { kind: "conflicts" };
625
+ const ranges = parseLineRanges(sel);
626
+ if (ranges) {
627
+ return { kind: "lines", ranges };
628
+ }
629
+ // Unrecognized selectors fall through; sqlite/archive/url readers consume their own colon syntax.
630
+ return { kind: "none" };
631
+ }
632
+
633
+ /**
634
+ * Convert a single-range selector to the offset/limit pair used by internal pagination.
635
+ * Returns the FIRST range only — multi-range callers MUST branch on `isMultiRange` before
636
+ * calling this helper.
637
+ */
638
+ function selToOffsetLimit(parsed: ParsedSelector): { offset?: number; limit?: number } {
639
+ if (parsed.kind === "lines") {
640
+ const first = parsed.ranges[0];
641
+ const limit = first.endLine !== undefined ? first.endLine - first.startLine + 1 : undefined;
642
+ return { offset: first.startLine, limit };
643
+ }
644
+ return {};
645
+ }
646
+
647
+ interface ResolvedArchiveReadPath {
648
+ absolutePath: string;
649
+ archiveSubPath: string;
650
+ suffixResolution?: { from: string; to: string };
651
+ }
652
+
653
+ interface ResolvedSqliteReadPath {
654
+ absolutePath: string;
655
+ sqliteSubPath: string;
656
+ queryString: string;
657
+ suffixResolution?: { from: string; to: string };
658
+ }
659
+
660
+ /**
661
+ * Read tool implementation.
662
+ *
663
+ * Reads files with support for images, converted documents (via markit), and text.
664
+ * Directories return a formatted listing with modification times.
665
+ */
666
+ export class ReadTool implements AgentTool<typeof readSchema, ReadToolDetails> {
667
+ readonly name = "read";
668
+ readonly approval = "read" as const;
669
+ readonly label = "Read";
670
+ readonly loadMode = "essential";
671
+ readonly description: string;
672
+ readonly parameters = readSchema;
673
+ readonly nonAbortable = true;
674
+ readonly strict = true;
675
+
676
+ readonly #autoResizeImages: boolean;
677
+ readonly #defaultLimit: number;
678
+ readonly #inspectImageEnabled: boolean;
679
+
680
+ constructor(private readonly session: ToolSession) {
681
+ const displayMode = resolveFileDisplayMode(session);
682
+ this.#autoResizeImages = session.settings.get("images.autoResize");
683
+ this.#defaultLimit = Math.max(
684
+ 1,
685
+ Math.min(session.settings.get("read.defaultLimit") ?? DEFAULT_MAX_LINES, DEFAULT_MAX_LINES),
686
+ );
687
+ this.#inspectImageEnabled = session.settings.get("inspect_image.enabled");
688
+ this.description = prompt.render(readDescription, {
689
+ DEFAULT_LIMIT: String(this.#defaultLimit),
690
+ DEFAULT_MAX_LINES: String(DEFAULT_MAX_LINES),
691
+ IS_HL_MODE: displayMode.hashLines,
692
+ IS_LINE_NUMBER_MODE: !displayMode.hashLines && displayMode.lineNumbers,
693
+ INSPECT_IMAGE_ENABLED: this.#inspectImageEnabled,
694
+ });
695
+ }
696
+
697
+ async #tryReadDelimitedPaths(
698
+ readPath: string,
699
+ signal?: AbortSignal,
700
+ ): Promise<AgentToolResult<ReadToolDetails> | null> {
701
+ const parts = await splitDelimitedPathEntry(readPath, this.session.cwd);
702
+ if (!parts) return null;
703
+
704
+ const notice = `Note: interpreted as ${parts.length} paths: ${parts.join(", ")}`;
705
+ const notes = [notice];
706
+ const content: Array<TextContent | ImageContent> = [];
707
+ let pendingText = notice;
708
+ const flushText = () => {
709
+ if (pendingText.length === 0) return;
710
+ content.push({ type: "text", text: pendingText });
711
+ pendingText = "";
712
+ };
713
+ const appendText = (text: string) => {
714
+ pendingText = pendingText.length > 0 ? `${pendingText}\n\n${text}` : text;
715
+ };
716
+
717
+ for (const part of parts) {
718
+ try {
719
+ const result = await this.execute("read-delimited-part", { path: part }, signal);
720
+ for (const block of result.content) {
721
+ if (block.type === "text") {
722
+ appendText(block.text);
723
+ continue;
724
+ }
725
+ flushText();
726
+ content.push(block);
727
+ }
728
+ } catch (error) {
729
+ if (error instanceof ToolAbortError || signal?.aborted) throw error;
730
+ const message = error instanceof Error ? error.message : String(error);
731
+ const errorNote = `Could not read ${part}: ${message}`;
732
+ notes.push(errorNote);
733
+ appendText(`[${errorNote}]`);
734
+ }
735
+ }
736
+ flushText();
737
+
738
+ return toolResult<ReadToolDetails>({ notes }).content(content).done();
739
+ }
740
+
741
+ async #resolveArchiveReadPath(readPath: string, signal?: AbortSignal): Promise<ResolvedArchiveReadPath | null> {
742
+ const candidates = parseArchivePathCandidates(readPath);
743
+ for (const candidate of candidates) {
744
+ let absolutePath = resolveReadPath(candidate.archivePath, this.session.cwd);
745
+ let suffixResolution: { from: string; to: string } | undefined;
746
+
747
+ try {
748
+ const stat = await Bun.file(absolutePath).stat();
749
+ if (stat.isDirectory()) continue;
750
+ return {
751
+ absolutePath,
752
+ archiveSubPath: candidate.archivePath === readPath ? "" : candidate.subPath,
753
+ suffixResolution,
754
+ };
755
+ } catch (error) {
756
+ if (!isNotFoundError(error) || isRemoteMountPath(absolutePath)) continue;
757
+
758
+ const suffixMatch = await findUniqueSuffixMatch(candidate.archivePath, this.session.cwd, signal);
759
+ if (!suffixMatch) continue;
760
+
761
+ try {
762
+ const retryStat = await Bun.file(suffixMatch.absolutePath).stat();
763
+ if (retryStat.isDirectory()) continue;
764
+
765
+ absolutePath = suffixMatch.absolutePath;
766
+ suffixResolution = { from: candidate.archivePath, to: suffixMatch.displayPath };
767
+ return {
768
+ absolutePath,
769
+ archiveSubPath: candidate.archivePath === readPath ? "" : candidate.subPath,
770
+ suffixResolution,
771
+ };
772
+ } catch (retryError) {
773
+ if (!isNotFoundError(retryError)) {
774
+ throw retryError;
775
+ }
776
+ }
777
+ }
778
+ }
779
+
780
+ return null;
781
+ }
782
+
783
+ async #resolveSqliteReadPath(readPath: string, signal?: AbortSignal): Promise<ResolvedSqliteReadPath | null> {
784
+ const candidates = parseSqlitePathCandidates(readPath);
785
+ for (const candidate of candidates) {
786
+ let absolutePath = resolveReadPath(candidate.sqlitePath, this.session.cwd);
787
+ let suffixResolution: { from: string; to: string } | undefined;
788
+
789
+ try {
790
+ const stat = await Bun.file(absolutePath).stat();
791
+ if (stat.isDirectory()) continue;
792
+ if (!(await isSqliteFile(absolutePath))) continue;
793
+
794
+ return {
795
+ absolutePath,
796
+ sqliteSubPath: candidate.subPath,
797
+ queryString: candidate.queryString,
798
+ suffixResolution,
799
+ };
800
+ } catch (error) {
801
+ if (!isNotFoundError(error) || isRemoteMountPath(absolutePath)) continue;
802
+
803
+ const suffixMatch = await findUniqueSuffixMatch(candidate.sqlitePath, this.session.cwd, signal);
804
+ if (!suffixMatch) continue;
805
+
806
+ try {
807
+ const retryStat = await Bun.file(suffixMatch.absolutePath).stat();
808
+ if (retryStat.isDirectory()) continue;
809
+ if (!(await isSqliteFile(suffixMatch.absolutePath))) continue;
810
+
811
+ absolutePath = suffixMatch.absolutePath;
812
+ suffixResolution = { from: candidate.sqlitePath, to: suffixMatch.displayPath };
813
+ return {
814
+ absolutePath,
815
+ sqliteSubPath: candidate.subPath,
816
+ queryString: candidate.queryString,
817
+ suffixResolution,
818
+ };
819
+ } catch (retryError) {
820
+ if (!isNotFoundError(retryError)) {
821
+ throw retryError;
822
+ }
823
+ }
824
+ }
825
+ }
826
+
827
+ return null;
828
+ }
829
+
830
+ #buildInMemoryTextResult(
831
+ text: string,
832
+ offset: number | undefined,
833
+ limit: number | undefined,
834
+ options: {
835
+ details?: ReadToolDetails;
836
+ sourcePath?: string;
837
+ sourceUrl?: string;
838
+ sourceInternal?: string;
839
+ entityLabel: string;
840
+ ignoreResultLimits?: boolean;
841
+ raw?: boolean;
842
+ immutable?: boolean;
843
+ },
844
+ ): AgentToolResult<ReadToolDetails> {
845
+ const displayMode = resolveFileDisplayMode(this.session, { raw: options.raw, immutable: options.immutable });
846
+ const details = options.details ?? {};
847
+ const allLines = text.split("\n");
848
+ const totalLines = allLines.length;
849
+ // User-requested 0-indexed range start. Lines BEFORE this are leading
850
+ // context (added below if offset is explicit).
851
+ const requestedStart = offset ? Math.max(0, offset - 1) : 0;
852
+ const ignoreResultLimits = options.ignoreResultLimits ?? false;
853
+ const requestedEnd = limit !== undefined ? Math.min(requestedStart + limit, allLines.length) : allLines.length;
854
+ // Expand only on sides the user actually constrained: leading context
855
+ // when offset>1, trailing context when a finite limit was set.
856
+ const expanded = expandRangeWithContext(
857
+ requestedStart,
858
+ requestedEnd,
859
+ allLines.length,
860
+ offset !== undefined && offset > 1,
861
+ limit !== undefined,
862
+ );
863
+ const startLine = expanded.startLine;
864
+ const endLineExpanded = expanded.endLine;
865
+ const startLineDisplay = startLine + 1;
866
+
867
+ const resultBuilder = toolResult(details);
868
+ if (options.sourcePath) {
869
+ resultBuilder.sourcePath(options.sourcePath);
870
+ }
871
+ if (options.sourceUrl) {
872
+ resultBuilder.sourceUrl(options.sourceUrl);
873
+ }
874
+ if (options.sourceInternal) {
875
+ resultBuilder.sourceInternal(options.sourceInternal);
876
+ }
877
+
878
+ if (requestedStart >= allLines.length) {
879
+ const suggestion =
880
+ allLines.length === 0
881
+ ? `The ${options.entityLabel} is empty.`
882
+ : `Use :1 to read from the start, or :${allLines.length} to read the last line.`;
883
+ return resultBuilder
884
+ .text(
885
+ `Line ${requestedStart + 1} is beyond end of ${options.entityLabel} (${allLines.length} lines total). ${suggestion}`,
886
+ )
887
+ .done();
888
+ }
889
+
890
+ const endLine = endLineExpanded;
891
+ const selectedContent = allLines.slice(startLine, endLine).join("\n");
892
+ const userLimitedLines = limit !== undefined ? endLine - startLine : undefined;
893
+ const truncation = ignoreResultLimits ? noTruncResult(selectedContent) : truncateHead(selectedContent);
894
+
895
+ const shouldAddHashLines = displayMode.hashLines;
896
+ const shouldAddLineNumbers = shouldAddHashLines ? false : displayMode.lineNumbers;
897
+ const hashContext =
898
+ shouldAddHashLines && options.sourcePath
899
+ ? recordFullHashlineContext(
900
+ this.session,
901
+ options.sourcePath,
902
+ formatPathRelativeToCwd(options.sourcePath, this.session.cwd),
903
+ text,
904
+ )
905
+ : undefined;
906
+ let emittedHashlineHeader = false;
907
+ const formatText = (content: string, startNum: number): string => {
908
+ details.displayContent = { text: content, startLine: startNum };
909
+ const formatted = formatTextWithMode(content, startNum, shouldAddHashLines, shouldAddLineNumbers);
910
+ if (!hashContext || emittedHashlineHeader) return formatted;
911
+ emittedHashlineHeader = true;
912
+ return prependHashlineHeader(formatted, hashContext);
913
+ };
914
+
915
+ let outputText: string;
916
+ let truncationInfo:
917
+ | { result: TruncationResult; options: { direction: "head"; startLine?: number; totalFileLines?: number } }
918
+ | undefined;
919
+
920
+ if (truncation.firstLineExceedsLimit) {
921
+ const firstLine = allLines[startLine] ?? "";
922
+ const firstLineBytes = Buffer.byteLength(firstLine, "utf-8");
923
+ const snippet = truncateHeadBytes(firstLine, DEFAULT_MAX_BYTES);
924
+
925
+ if (shouldAddHashLines) {
926
+ outputText = `[Line ${startLineDisplay} is ${formatBytes(
927
+ firstLineBytes,
928
+ )}, exceeds ${formatBytes(DEFAULT_MAX_BYTES)} limit. Hashline output requires full lines; cannot emit an editable numbered preview for a truncated line.]`;
929
+ } else {
930
+ outputText = formatText(snippet.text, startLineDisplay);
931
+ }
932
+
933
+ if (snippet.text.length === 0) {
934
+ outputText = `[Line ${startLineDisplay} is ${formatBytes(
935
+ firstLineBytes,
936
+ )}, exceeds ${formatBytes(DEFAULT_MAX_BYTES)} limit. Unable to display a valid UTF-8 snippet.]`;
937
+ }
938
+
939
+ details.truncation = truncation;
940
+ truncationInfo = {
941
+ result: truncation,
942
+ options: { direction: "head", startLine: startLineDisplay, totalFileLines: totalLines },
943
+ };
944
+ } else if (truncation.truncated) {
945
+ outputText = formatText(truncation.content, startLineDisplay);
946
+ details.truncation = truncation;
947
+ truncationInfo = {
948
+ result: truncation,
949
+ options: { direction: "head", startLine: startLineDisplay, totalFileLines: totalLines },
950
+ };
951
+ } else if (userLimitedLines !== undefined && startLine + userLimitedLines < allLines.length) {
952
+ const remaining = allLines.length - (startLine + userLimitedLines);
953
+ const nextOffset = startLine + userLimitedLines + 1;
954
+
955
+ outputText = formatText(selectedContent, startLineDisplay);
956
+ outputText += `\n\n[${remaining} more lines in ${options.entityLabel}. Use :${nextOffset} to continue]`;
957
+ } else {
958
+ outputText = formatText(truncation.content, startLineDisplay);
959
+ }
960
+
961
+ resultBuilder.text(outputText);
962
+ if (truncationInfo) {
963
+ resultBuilder.truncation(truncationInfo.result, truncationInfo.options);
964
+ }
965
+ return resultBuilder.done();
966
+ }
967
+
968
+ /**
969
+ * Render a multi-range read against in-memory text. Each range emits a
970
+ * formatted block with its own anchors / line numbers, blocks are joined
971
+ * with an elision separator, and ranges past EOF surface as `[…]` notices
972
+ * so the model can correct the next call. No leading/trailing context is
973
+ * added — multi-range callers always specify exact bounds.
974
+ */
975
+ #buildInMemoryMultiRangeResult(
976
+ text: string,
977
+ ranges: readonly LineRange[],
978
+ options: {
979
+ details?: ReadToolDetails;
980
+ sourcePath?: string;
981
+ sourceUrl?: string;
982
+ sourceInternal?: string;
983
+ entityLabel: string;
984
+ raw?: boolean;
985
+ immutable?: boolean;
986
+ },
987
+ ): AgentToolResult<ReadToolDetails> {
988
+ const displayMode = resolveFileDisplayMode(this.session, { raw: options.raw, immutable: options.immutable });
989
+ const details = options.details ?? {};
990
+ const allLines = text.split("\n");
991
+ const totalLines = allLines.length;
992
+ const shouldAddHashLines = displayMode.hashLines;
993
+ const shouldAddLineNumbers = shouldAddHashLines ? false : displayMode.lineNumbers;
994
+ const hashContext =
995
+ shouldAddHashLines && options.sourcePath
996
+ ? recordFullHashlineContext(
997
+ this.session,
998
+ options.sourcePath,
999
+ formatPathRelativeToCwd(options.sourcePath, this.session.cwd),
1000
+ text,
1001
+ )
1002
+ : undefined;
1003
+ let emittedHashlineHeader = false;
1004
+
1005
+ const resultBuilder = toolResult(details);
1006
+ if (options.sourcePath) resultBuilder.sourcePath(options.sourcePath);
1007
+ if (options.sourceUrl) resultBuilder.sourceUrl(options.sourceUrl);
1008
+ if (options.sourceInternal) resultBuilder.sourceInternal(options.sourceInternal);
1009
+
1010
+ const parts: string[] = [];
1011
+ const outOfBounds: LineRange[] = [];
1012
+ for (const range of ranges) {
1013
+ if (range.startLine > totalLines) {
1014
+ outOfBounds.push(range);
1015
+ continue;
1016
+ }
1017
+ const effectiveEnd = Math.min(range.endLine ?? totalLines, totalLines);
1018
+ const sliced = allLines.slice(range.startLine - 1, effectiveEnd).join("\n");
1019
+ const formatted = formatTextWithMode(sliced, range.startLine, shouldAddHashLines, shouldAddLineNumbers);
1020
+ parts.push(hashContext && !emittedHashlineHeader ? prependHashlineHeader(formatted, hashContext) : formatted);
1021
+ if (hashContext) emittedHashlineHeader = true;
1022
+ }
1023
+
1024
+ const outputText = parts.length > 0 ? parts.join("\n\n…\n\n") : "";
1025
+ const notices: string[] = [];
1026
+ for (const range of outOfBounds) {
1027
+ const bound = range.endLine !== undefined ? `${range.startLine}-${range.endLine}` : `${range.startLine}`;
1028
+ notices.push(`[Range ${bound} is beyond end of ${options.entityLabel} (${totalLines} lines total); skipped]`);
1029
+ }
1030
+ const finalText =
1031
+ notices.length > 0 ? (outputText ? `${outputText}\n${notices.join("\n")}` : notices.join("\n")) : outputText;
1032
+ resultBuilder.text(finalText);
1033
+ return resultBuilder.done();
1034
+ }
1035
+
1036
+ /**
1037
+ * Stream multiple non-contiguous ranges from a local file. ACP bridge takes
1038
+ * priority when present (editor buffer is source of truth); otherwise each
1039
+ * range is streamed independently with its own line/byte budget. Out-of-bounds
1040
+ * ranges surface as inline notices rather than aborting the read.
1041
+ */
1042
+ async #readLocalFileMultiRange(
1043
+ absolutePath: string,
1044
+ ranges: readonly LineRange[],
1045
+ parsed: ParsedSelector,
1046
+ displayMode: { hashLines: boolean; lineNumbers: boolean },
1047
+ suffixResolution: { from: string; to: string } | undefined,
1048
+ signal: AbortSignal | undefined,
1049
+ ): Promise<{
1050
+ outputText: string;
1051
+ columnTruncated: number;
1052
+ bridgeResult?: AgentToolResult<ReadToolDetails>;
1053
+ }> {
1054
+ const rawSelector = isRawSelector(parsed);
1055
+
1056
+ // ACP bridge first — the editor's in-memory buffer is source of truth.
1057
+ const bridgePromise = this.#routeReadThroughBridge(absolutePath);
1058
+ if (bridgePromise !== undefined) {
1059
+ try {
1060
+ const bridgeText = await bridgePromise;
1061
+ const bridgeResult = this.#buildInMemoryMultiRangeResult(bridgeText, ranges, {
1062
+ details: { resolvedPath: absolutePath, suffixResolution },
1063
+ sourcePath: absolutePath,
1064
+ entityLabel: "file",
1065
+ raw: rawSelector,
1066
+ });
1067
+ if (suffixResolution) {
1068
+ const notice = `[Path '${suffixResolution.from}' not found; resolved to '${suffixResolution.to}' via suffix match]`;
1069
+ const firstText = bridgeResult.content.find((c): c is TextContent => c.type === "text");
1070
+ if (firstText) firstText.text = `${notice}\n${firstText.text}`;
1071
+ }
1072
+ return { outputText: "", columnTruncated: 0, bridgeResult };
1073
+ } catch (error) {
1074
+ logger.warn("ACP fs readTextFile failed; falling back to disk", { path: absolutePath, error });
1075
+ }
1076
+ }
1077
+
1078
+ const shouldAddHashLines = !rawSelector && displayMode.hashLines;
1079
+ const shouldAddLineNumbers = rawSelector ? false : shouldAddHashLines ? false : displayMode.lineNumbers;
1080
+ const maxColumns = resolveOutputMaxColumns(this.session.settings);
1081
+
1082
+ const blocks: string[] = [];
1083
+ const notices: string[] = [];
1084
+ let columnTruncated = 0;
1085
+
1086
+ for (const range of ranges) {
1087
+ const rangeStart = range.startLine - 1; // 0-indexed
1088
+ const requestedLength = range.endLine !== undefined ? range.endLine - range.startLine + 1 : this.#defaultLimit;
1089
+ const maxLines = Math.min(requestedLength, DEFAULT_MAX_LINES);
1090
+ const maxBytesForRead = Math.max(DEFAULT_MAX_BYTES, maxLines * 512);
1091
+
1092
+ const streamResult = await streamLinesFromFile(
1093
+ absolutePath,
1094
+ rangeStart,
1095
+ maxLines,
1096
+ maxBytesForRead,
1097
+ maxLines,
1098
+ signal,
1099
+ );
1100
+ const totalFileLines = streamResult.totalFileLines;
1101
+
1102
+ if (rangeStart >= totalFileLines) {
1103
+ const bound = range.endLine !== undefined ? `${range.startLine}-${range.endLine}` : `${range.startLine}`;
1104
+ notices.push(`[Range ${bound} is beyond end of file (${totalFileLines} lines total); skipped]`);
1105
+ continue;
1106
+ }
1107
+
1108
+ const collectedLines = streamResult.lines;
1109
+ // Column truncation is display-only; clone before stamping ellipsis so
1110
+ // the original on-disk lines stay intact for display reconstruction.
1111
+ let displayLines: string[] = collectedLines;
1112
+ if (!rawSelector && maxColumns > 0) {
1113
+ let cloned: string[] | undefined;
1114
+ for (let i = 0; i < collectedLines.length; i++) {
1115
+ const { text, wasTruncated } = truncateLine(collectedLines[i], maxColumns);
1116
+ if (wasTruncated) {
1117
+ if (!cloned) cloned = collectedLines.slice();
1118
+ cloned[i] = text;
1119
+ columnTruncated = maxColumns;
1120
+ }
1121
+ }
1122
+ if (cloned) displayLines = cloned;
1123
+ }
1124
+ const blockText = displayLines.join("\n");
1125
+ blocks.push(formatTextWithMode(blockText, range.startLine, shouldAddHashLines, shouldAddLineNumbers));
1126
+ }
1127
+
1128
+ let outputText = blocks.join("\n\n…\n\n");
1129
+ if (shouldAddHashLines && outputText) {
1130
+ const tag = await recordFileSnapshot(this.session, absolutePath);
1131
+ if (tag) {
1132
+ outputText = `${formatHashlineHeader(formatPathRelativeToCwd(absolutePath, this.session.cwd), tag)}\n${outputText}`;
1133
+ }
1134
+ }
1135
+ if (notices.length > 0) {
1136
+ outputText = outputText ? `${outputText}\n${notices.join("\n")}` : notices.join("\n");
1137
+ }
1138
+ return { outputText, columnTruncated };
1139
+ }
1140
+
1141
+ async #readArchiveDirectory(
1142
+ archive: ArchiveReader,
1143
+ archivePath: string,
1144
+ subPath: string,
1145
+ limit: number | undefined,
1146
+ details: ReadToolDetails,
1147
+ signal?: AbortSignal,
1148
+ ): Promise<AgentToolResult<ReadToolDetails>> {
1149
+ const DEFAULT_LIMIT = 500;
1150
+ const effectiveLimit = limit ?? DEFAULT_LIMIT;
1151
+ const entries = archive.listDirectory(subPath);
1152
+
1153
+ const listLimit = applyListLimit(entries, { limit: effectiveLimit });
1154
+ const limitedEntries = listLimit.items;
1155
+ const limitMeta = listLimit.meta;
1156
+
1157
+ const results: string[] = [];
1158
+ for (const entry of limitedEntries) {
1159
+ throwIfAborted(signal);
1160
+ if (entry.isDirectory) {
1161
+ results.push(`${entry.name}/`);
1162
+ continue;
1163
+ }
1164
+
1165
+ const sizeSuffix = entry.size > 0 ? ` (${formatBytes(entry.size)})` : "";
1166
+ results.push(`${entry.name}${sizeSuffix}`);
1167
+ }
1168
+
1169
+ const output = results.length > 0 ? results.join("\n") : "(empty archive directory)";
1170
+ const text = prependSuffixResolutionNotice(output, details.suffixResolution);
1171
+ const truncation = truncateHead(text, { maxLines: Number.MAX_SAFE_INTEGER });
1172
+ const directoryDetails: ReadToolDetails = { ...details, isDirectory: true };
1173
+ const resultBuilder = toolResult<ReadToolDetails>(directoryDetails).text(truncation.content);
1174
+ resultBuilder.sourcePath(archivePath).limits({ resultLimit: limitMeta.resultLimit?.reached });
1175
+ if (truncation.truncated) {
1176
+ directoryDetails.truncation = truncation;
1177
+ resultBuilder.truncation(truncation, { direction: "head" });
1178
+ }
1179
+ return resultBuilder.done();
1180
+ }
1181
+
1182
+ async #readArchive(
1183
+ readPath: string,
1184
+ parsedSel: ParsedSelector,
1185
+ resolvedArchivePath: ResolvedArchiveReadPath,
1186
+ signal?: AbortSignal,
1187
+ ): Promise<AgentToolResult<ReadToolDetails>> {
1188
+ throwIfAborted(signal);
1189
+ const archive = await openArchive(resolvedArchivePath.absolutePath);
1190
+ throwIfAborted(signal);
1191
+
1192
+ const details: ReadToolDetails = {
1193
+ resolvedPath: resolvedArchivePath.absolutePath,
1194
+ suffixResolution: resolvedArchivePath.suffixResolution,
1195
+ };
1196
+
1197
+ const node = archive.getNode(resolvedArchivePath.archiveSubPath);
1198
+ if (!node) {
1199
+ throw new ToolError(`Path '${readPath}' not found inside archive`);
1200
+ }
1201
+
1202
+ if (node.isDirectory) {
1203
+ if (isMultiRange(parsedSel)) {
1204
+ throw new ToolError("Multi-range line selectors are not supported for archive directory listings.");
1205
+ }
1206
+ const { limit } = selToOffsetLimit(parsedSel);
1207
+ return this.#readArchiveDirectory(
1208
+ archive,
1209
+ resolvedArchivePath.absolutePath,
1210
+ resolvedArchivePath.archiveSubPath,
1211
+ limit,
1212
+ details,
1213
+ signal,
1214
+ );
1215
+ }
1216
+
1217
+ const entry = await archive.readFile(resolvedArchivePath.archiveSubPath);
1218
+ const text = decodeUtf8Text(entry.bytes);
1219
+ if (text === null) {
1220
+ return toolResult<ReadToolDetails>(details)
1221
+ .text(
1222
+ prependSuffixResolutionNotice(
1223
+ `[Cannot read binary archive entry '${entry.path}' (${formatBytes(entry.size)})]`,
1224
+ resolvedArchivePath.suffixResolution,
1225
+ ),
1226
+ )
1227
+ .sourcePath(resolvedArchivePath.absolutePath)
1228
+ .done();
1229
+ }
1230
+
1231
+ const raw = isRawSelector(parsedSel);
1232
+ const result =
1233
+ isMultiRange(parsedSel) && parsedSel.kind === "lines"
1234
+ ? this.#buildInMemoryMultiRangeResult(text, parsedSel.ranges, {
1235
+ details,
1236
+ sourcePath: resolvedArchivePath.absolutePath,
1237
+ entityLabel: "archive entry",
1238
+ raw,
1239
+ })
1240
+ : this.#buildInMemoryTextResult(
1241
+ text,
1242
+ selToOffsetLimit(parsedSel).offset,
1243
+ selToOffsetLimit(parsedSel).limit,
1244
+ {
1245
+ details,
1246
+ sourcePath: resolvedArchivePath.absolutePath,
1247
+ entityLabel: "archive entry",
1248
+ raw,
1249
+ },
1250
+ );
1251
+ const firstText = result.content.find((content): content is TextContent => content.type === "text");
1252
+ if (firstText) {
1253
+ firstText.text = prependSuffixResolutionNotice(firstText.text, resolvedArchivePath.suffixResolution);
1254
+ }
1255
+ return result;
1256
+ }
1257
+
1258
+ async #readSqlite(
1259
+ resolvedSqlitePath: ResolvedSqliteReadPath,
1260
+ signal?: AbortSignal,
1261
+ ): Promise<AgentToolResult<ReadToolDetails>> {
1262
+ throwIfAborted(signal);
1263
+
1264
+ const selectorInput = {
1265
+ subPath: resolvedSqlitePath.sqliteSubPath,
1266
+ queryString: resolvedSqlitePath.queryString,
1267
+ };
1268
+ const selector = parseSqliteSelector(selectorInput.subPath, selectorInput.queryString);
1269
+ const details: ReadToolDetails = {
1270
+ resolvedPath: resolvedSqlitePath.absolutePath,
1271
+ suffixResolution: resolvedSqlitePath.suffixResolution,
1272
+ };
1273
+
1274
+ let db: Database | null = null;
1275
+ try {
1276
+ db = new Database(resolvedSqlitePath.absolutePath, { readonly: true, strict: true });
1277
+ db.run("PRAGMA busy_timeout = 3000");
1278
+ throwIfAborted(signal);
1279
+
1280
+ switch (selector.kind) {
1281
+ case "list": {
1282
+ const listLimit = applyListLimit(listTables(db), { limit: 500 });
1283
+ const output = prependSuffixResolutionNotice(
1284
+ renderTableList(listLimit.items),
1285
+ resolvedSqlitePath.suffixResolution,
1286
+ );
1287
+ const truncation = truncateHead(output, { maxLines: Number.MAX_SAFE_INTEGER });
1288
+ details.truncation = truncation.truncated ? truncation : undefined;
1289
+ const resultBuilder = toolResult<ReadToolDetails>(details)
1290
+ .text(truncation.content)
1291
+ .sourcePath(resolvedSqlitePath.absolutePath)
1292
+ .limits({ resultLimit: listLimit.meta.resultLimit?.reached });
1293
+ if (truncation.truncated) {
1294
+ resultBuilder.truncation(truncation, { direction: "head" });
1295
+ }
1296
+ return resultBuilder.done();
1297
+ }
1298
+ case "schema": {
1299
+ const sampleRows = queryRows(db, selector.table, { limit: selector.sampleLimit, offset: 0 });
1300
+ let output = renderSchema(getTableSchema(db, selector.table), {
1301
+ columns: sampleRows.columns,
1302
+ rows: sampleRows.rows,
1303
+ });
1304
+ if (sampleRows.rows.length < sampleRows.totalCount) {
1305
+ const remaining = sampleRows.totalCount - sampleRows.rows.length;
1306
+ output += `\n[${remaining} more rows; append :${selector.table}?limit=20&offset=${sampleRows.rows.length} to the database path to continue]`;
1307
+ }
1308
+ return toolResult<ReadToolDetails>(details)
1309
+ .text(prependSuffixResolutionNotice(output, resolvedSqlitePath.suffixResolution))
1310
+ .sourcePath(resolvedSqlitePath.absolutePath)
1311
+ .done();
1312
+ }
1313
+ case "row": {
1314
+ const lookup = resolveTableRowLookup(db, selector.table);
1315
+ const row =
1316
+ lookup.kind === "pk"
1317
+ ? getRowByKey(db, selector.table, lookup, selector.key)
1318
+ : getRowByRowId(db, selector.table, selector.key);
1319
+ if (!row) {
1320
+ return toolResult<ReadToolDetails>(details)
1321
+ .text(
1322
+ prependSuffixResolutionNotice(
1323
+ `No row found in table '${selector.table}' for key '${selector.key}'.`,
1324
+ resolvedSqlitePath.suffixResolution,
1325
+ ),
1326
+ )
1327
+ .sourcePath(resolvedSqlitePath.absolutePath)
1328
+ .done();
1329
+ }
1330
+ return toolResult<ReadToolDetails>(details)
1331
+ .text(prependSuffixResolutionNotice(renderRow(row), resolvedSqlitePath.suffixResolution))
1332
+ .sourcePath(resolvedSqlitePath.absolutePath)
1333
+ .done();
1334
+ }
1335
+ case "query": {
1336
+ const page = queryRows(db, selector.table, selector);
1337
+ return toolResult<ReadToolDetails>(details)
1338
+ .text(
1339
+ prependSuffixResolutionNotice(
1340
+ renderTable(page.columns, page.rows, {
1341
+ totalCount: page.totalCount,
1342
+ offset: selector.offset,
1343
+ limit: selector.limit,
1344
+ table: selector.table,
1345
+ dbPath: resolvedSqlitePath.absolutePath,
1346
+ }),
1347
+ resolvedSqlitePath.suffixResolution,
1348
+ ),
1349
+ )
1350
+ .sourcePath(resolvedSqlitePath.absolutePath)
1351
+ .done();
1352
+ }
1353
+ case "raw": {
1354
+ const result = executeReadQuery(db, selector.sql);
1355
+ return toolResult<ReadToolDetails>(details)
1356
+ .text(
1357
+ prependSuffixResolutionNotice(
1358
+ renderTable(result.columns, result.rows, {
1359
+ totalCount: result.rows.length,
1360
+ offset: 0,
1361
+ limit: result.rows.length || DEFAULT_MAX_LINES,
1362
+ table: "query",
1363
+ dbPath: resolvedSqlitePath.absolutePath,
1364
+ }),
1365
+ resolvedSqlitePath.suffixResolution,
1366
+ ),
1367
+ )
1368
+ .sourcePath(resolvedSqlitePath.absolutePath)
1369
+ .done();
1370
+ }
1371
+ }
1372
+
1373
+ throw new ToolError("Unsupported SQLite selector");
1374
+ } catch (error) {
1375
+ if (error instanceof ToolError) {
1376
+ throw error;
1377
+ }
1378
+ throw new ToolError(error instanceof Error ? error.message : String(error));
1379
+ } finally {
1380
+ db?.close();
1381
+ }
1382
+ }
1383
+
1384
+ #routeReadThroughBridge(
1385
+ absolutePath: string,
1386
+ options?: { line?: number; limit?: number },
1387
+ ): Promise<string> | undefined {
1388
+ const bridge = this.session.getClientBridge?.();
1389
+ if (!bridge?.capabilities.readTextFile || !bridge.readTextFile) return undefined;
1390
+ return bridge.readTextFile({ path: absolutePath, ...options });
1391
+ }
1392
+
1393
+ async #trySummarize(absolutePath: string, fileSize: number, signal?: AbortSignal): Promise<SummaryResult | null> {
1394
+ if (fileSize > MAX_SUMMARY_BYTES) return null;
1395
+
1396
+ try {
1397
+ throwIfAborted(signal);
1398
+ const bridgePromise = this.#routeReadThroughBridge(absolutePath);
1399
+ const code =
1400
+ bridgePromise !== undefined
1401
+ ? await bridgePromise.catch(() => Bun.file(absolutePath).text())
1402
+ : await Bun.file(absolutePath).text();
1403
+ throwIfAborted(signal);
1404
+ const lineCount = countTextLines(code);
1405
+ if (lineCount > MAX_SUMMARY_LINES) return null;
1406
+ if (lineCount < this.session.settings.get("read.summarize.minTotalLines")) return null;
1407
+
1408
+ const result = summarizeCode({
1409
+ code,
1410
+ path: absolutePath,
1411
+ minBodyLines: this.session.settings.get("read.summarize.minBodyLines"),
1412
+ minCommentLines: this.session.settings.get("read.summarize.minCommentLines"),
1413
+ unfoldUntilLines: this.session.settings.get("read.summarize.unfoldUntil"),
1414
+ unfoldLimitLines: this.session.settings.get("read.summarize.unfoldLimit"),
1415
+ });
1416
+ return result;
1417
+ } catch {
1418
+ return null;
1419
+ }
1420
+ }
1421
+
1422
+ #renderSummary(summary: SummaryResult): {
1423
+ text: string;
1424
+ displayText: string;
1425
+ elidedRanges: ElidedRange[];
1426
+ elidedLines: number;
1427
+ } {
1428
+ const displayMode = resolveFileDisplayMode(this.session);
1429
+ const shouldAddHashLines = displayMode.hashLines;
1430
+ const shouldAddLineNumbers = shouldAddHashLines ? false : displayMode.lineNumbers;
1431
+
1432
+ // Flatten segments into per-line units so we can merge a kept-head /
1433
+ // elided / kept-tail sandwich into a single brace-pair line when the
1434
+ // boundary lines look like `… {` and `}` (or matching variants).
1435
+ type Unit =
1436
+ | { kind: "line"; line: number; text: string }
1437
+ | { kind: "elided"; startLine: number; endLine: number }
1438
+ | {
1439
+ kind: "merged";
1440
+ startLine: number;
1441
+ endLine: number;
1442
+ headText: string;
1443
+ tailText: string;
1444
+ };
1445
+
1446
+ const raw: Unit[] = [];
1447
+ for (const segment of summary.segments) {
1448
+ if (segment.kind === "elided") {
1449
+ raw.push({ kind: "elided", startLine: segment.startLine, endLine: segment.endLine });
1450
+ continue;
1451
+ }
1452
+ const text = segment.text ?? "";
1453
+ if (text.length === 0) continue;
1454
+ const lines = text.split("\n");
1455
+ for (let i = 0; i < lines.length; i++) {
1456
+ raw.push({ kind: "line", line: segment.startLine + i, text: lines[i] });
1457
+ }
1458
+ }
1459
+
1460
+ const units: Unit[] = [];
1461
+ let i = 0;
1462
+ while (i < raw.length) {
1463
+ const cur = raw[i];
1464
+ if (cur.kind === "elided") {
1465
+ const prev = units.length > 0 ? units[units.length - 1] : null;
1466
+ const next = i + 1 < raw.length ? raw[i + 1] : null;
1467
+ if (prev?.kind === "line" && next?.kind === "line" && canMergeBracePair(prev.text, next.text)) {
1468
+ units.pop();
1469
+ units.push({
1470
+ kind: "merged",
1471
+ startLine: prev.line,
1472
+ endLine: next.line,
1473
+ headText: prev.text,
1474
+ tailText: next.text,
1475
+ });
1476
+ i += 2;
1477
+ continue;
1478
+ }
1479
+ }
1480
+ units.push(cur);
1481
+ i++;
1482
+ }
1483
+
1484
+ const modelParts: string[] = [];
1485
+ const displayParts: string[] = [];
1486
+ const elidedRanges: ElidedRange[] = [];
1487
+ let elidedLines = 0;
1488
+ for (const unit of units) {
1489
+ if (unit.kind === "elided") {
1490
+ modelParts.push("...");
1491
+ displayParts.push("...");
1492
+ elidedRanges.push({ start: unit.startLine, end: unit.endLine });
1493
+ elidedLines += unit.endLine - unit.startLine + 1;
1494
+ continue;
1495
+ }
1496
+ if (unit.kind === "merged") {
1497
+ const formatted = formatMergedBraceLine(
1498
+ unit.startLine,
1499
+ unit.endLine,
1500
+ unit.headText,
1501
+ unit.tailText,
1502
+ shouldAddHashLines,
1503
+ shouldAddLineNumbers,
1504
+ );
1505
+ modelParts.push(formatted.model);
1506
+ displayParts.push(formatted.display);
1507
+ // Suggest the full brace range so re-reading shows both braces
1508
+ // plus the elided body in one shot.
1509
+ elidedRanges.push({ start: unit.startLine, end: unit.endLine });
1510
+ // Merged brace pair encloses (start+1)..(end-1) as elided.
1511
+ elidedLines += Math.max(0, unit.endLine - unit.startLine - 1);
1512
+ continue;
1513
+ }
1514
+ modelParts.push(formatSingleLine(unit.line, unit.text, shouldAddHashLines, shouldAddLineNumbers));
1515
+ displayParts.push(unit.text);
1516
+ }
1517
+
1518
+ return { text: modelParts.join("\n"), displayText: displayParts.join("\n"), elidedRanges, elidedLines };
1519
+ }
1520
+
1521
+ async execute(
1522
+ _toolCallId: string,
1523
+ params: ReadParams,
1524
+ signal?: AbortSignal,
1525
+ _onUpdate?: AgentToolUpdateCallback<ReadToolDetails>,
1526
+ _toolContext?: AgentToolContext,
1527
+ ): Promise<AgentToolResult<ReadToolDetails>> {
1528
+ let { path: readPath } = params;
1529
+ if (readPath.startsWith("file://")) {
1530
+ readPath = expandPath(readPath);
1531
+ }
1532
+
1533
+ const conflictUri = parseConflictUri(readPath);
1534
+ if (conflictUri) {
1535
+ if (conflictUri.id === "*") {
1536
+ throw new ToolError(
1537
+ "Reading `conflict://*` is not supported — wildcards are write-only. Use the `<path>:conflicts` read selector for the full list of conflicts in a file, or read `conflict://<N>` to inspect a single block.",
1538
+ );
1539
+ }
1540
+ return this.#readConflictRegion(conflictUri.id, conflictUri.scope);
1541
+ }
1542
+ const displayMode = resolveFileDisplayMode(this.session);
1543
+
1544
+ const parsedUrlTarget = parseReadUrlTarget(readPath);
1545
+ if (parsedUrlTarget) {
1546
+ if (!this.session.settings.get("fetch.enabled")) {
1547
+ throw new ToolError("URL reads are disabled by settings.");
1548
+ }
1549
+ if (parsedUrlTarget.ranges !== undefined) {
1550
+ const cached = await loadReadUrlCacheEntry(
1551
+ this.session,
1552
+ { path: parsedUrlTarget.path, raw: parsedUrlTarget.raw },
1553
+ signal,
1554
+ { ensureArtifact: true, preferCached: true },
1555
+ );
1556
+ return this.#buildInMemoryMultiRangeResult(cached.output, parsedUrlTarget.ranges, {
1557
+ details: { ...cached.details },
1558
+ sourceUrl: cached.details.finalUrl,
1559
+ entityLabel: "URL output",
1560
+ raw: parsedUrlTarget.raw,
1561
+ immutable: true,
1562
+ });
1563
+ }
1564
+ if (parsedUrlTarget.offset !== undefined || parsedUrlTarget.limit !== undefined) {
1565
+ const cached = await loadReadUrlCacheEntry(
1566
+ this.session,
1567
+ { path: parsedUrlTarget.path, raw: parsedUrlTarget.raw },
1568
+ signal,
1569
+ {
1570
+ ensureArtifact: true,
1571
+ preferCached: true,
1572
+ },
1573
+ );
1574
+ return this.#buildInMemoryTextResult(cached.output, parsedUrlTarget.offset, parsedUrlTarget.limit, {
1575
+ details: { ...cached.details },
1576
+ sourceUrl: cached.details.finalUrl,
1577
+ entityLabel: "URL output",
1578
+ raw: parsedUrlTarget.raw,
1579
+ immutable: true,
1580
+ });
1581
+ }
1582
+ return executeReadUrl(this.session, { path: parsedUrlTarget.path, raw: parsedUrlTarget.raw }, signal);
1583
+ }
1584
+
1585
+ // Handle internal URLs (agent://, artifact://, memory://, skill://, rule://, local://, mcp://, prometheus://, issue://, pr://).
1586
+ // Use the internal-URL-aware splitter so malformed selectors are peeled
1587
+ // off the URL and surfaced via parseSel rather than confusing handlers.
1588
+ const internalRouter = InternalUrlRouter.instance();
1589
+ if (internalRouter.canHandle(readPath)) {
1590
+ const internalTarget = splitInternalUrlSel(readPath);
1591
+ const parsed = parseSel(internalTarget.sel);
1592
+ return this.#handleInternalUrl(internalTarget.path, parsed, signal);
1593
+ }
1594
+
1595
+ const archivePath = await this.#resolveArchiveReadPath(readPath, signal);
1596
+ if (archivePath) {
1597
+ const archiveSubPath = splitPathAndSel(archivePath.archiveSubPath);
1598
+ const archiveParsed = parseSel(archiveSubPath.sel);
1599
+ return this.#readArchive(
1600
+ readPath,
1601
+ archiveParsed,
1602
+ { ...archivePath, archiveSubPath: archiveSubPath.path },
1603
+ signal,
1604
+ );
1605
+ }
1606
+
1607
+ const sqlitePath = await this.#resolveSqliteReadPath(readPath, signal);
1608
+ if (sqlitePath) {
1609
+ return this.#readSqlite(sqlitePath, signal);
1610
+ }
1611
+
1612
+ const localTarget = splitPathAndSel(readPath);
1613
+ const localReadPath = localTarget.path;
1614
+ const parsed = parseSel(localTarget.sel);
1615
+
1616
+ let absolutePath = resolveReadPath(localReadPath, this.session.cwd);
1617
+ let suffixResolution: { from: string; to: string } | undefined;
1618
+
1619
+ let isDirectory = false;
1620
+ let fileSize = 0;
1621
+ try {
1622
+ const stat = await Bun.file(absolutePath).stat();
1623
+ fileSize = stat.size;
1624
+ isDirectory = stat.isDirectory();
1625
+ } catch (error) {
1626
+ if (isNotFoundError(error)) {
1627
+ // Attempt unique suffix resolution before falling back to fuzzy suggestions
1628
+ if (!isRemoteMountPath(absolutePath)) {
1629
+ const suffixMatch = await findUniqueSuffixMatch(localReadPath, this.session.cwd, signal);
1630
+ if (suffixMatch) {
1631
+ try {
1632
+ const retryStat = await Bun.file(suffixMatch.absolutePath).stat();
1633
+ absolutePath = suffixMatch.absolutePath;
1634
+ fileSize = retryStat.size;
1635
+ isDirectory = retryStat.isDirectory();
1636
+ suffixResolution = { from: localReadPath, to: suffixMatch.displayPath };
1637
+ } catch {
1638
+ // Suffix match candidate no longer stats — fall through to error path
1639
+ }
1640
+ }
1641
+ }
1642
+
1643
+ if (!suffixResolution) {
1644
+ const delimitedResult = await this.#tryReadDelimitedPaths(readPath, signal);
1645
+ if (delimitedResult) return delimitedResult;
1646
+ throw new ToolError(`Path '${localReadPath}' not found`);
1647
+ }
1648
+ } else {
1649
+ throw error;
1650
+ }
1651
+ }
1652
+
1653
+ if (isDirectory) {
1654
+ if (isMultiRange(parsed)) {
1655
+ throw new ToolError("Multi-range line selectors are not supported for directory listings.");
1656
+ }
1657
+ const { offset, limit } = selToOffsetLimit(parsed);
1658
+ const dirResult = await this.#readDirectory(absolutePath, offset, limit, signal);
1659
+ if (suffixResolution) {
1660
+ dirResult.details ??= {};
1661
+ dirResult.details.suffixResolution = suffixResolution;
1662
+ }
1663
+ return dirResult;
1664
+ }
1665
+
1666
+ if (parsed.kind === "conflicts") {
1667
+ return this.#readFileConflicts(absolutePath, suffixResolution, signal);
1668
+ }
1669
+
1670
+ const imageMetadata = await readImageMetadata(absolutePath);
1671
+ const mimeType = imageMetadata?.mimeType;
1672
+ const ext = path.extname(absolutePath).toLowerCase();
1673
+ const shouldConvertWithMarkit = CONVERTIBLE_EXTENSIONS.has(ext);
1674
+ // Read the file based on type
1675
+ let content: Array<TextContent | ImageContent> | undefined;
1676
+ let details: ReadToolDetails = {};
1677
+ let sourcePath: string | undefined;
1678
+ let columnTruncated = 0;
1679
+ let truncationInfo:
1680
+ | { result: TruncationResult; options: { direction: "head"; startLine?: number; totalFileLines?: number } }
1681
+ | undefined;
1682
+
1683
+ if (mimeType) {
1684
+ if (this.#inspectImageEnabled) {
1685
+ const metadata = imageMetadata;
1686
+ const outputMime = metadata?.mimeType ?? mimeType;
1687
+ const outputBytes = fileSize;
1688
+ const metadataLines = [
1689
+ "Image metadata:",
1690
+ `- MIME: ${outputMime}`,
1691
+ `- Bytes: ${outputBytes} (${formatBytes(outputBytes)})`,
1692
+ metadata?.width !== undefined && metadata.height !== undefined
1693
+ ? `- Dimensions: ${metadata.width}x${metadata.height}`
1694
+ : "- Dimensions: unknown",
1695
+ metadata?.channels !== undefined ? `- Channels: ${metadata.channels}` : "- Channels: unknown",
1696
+ metadata?.hasAlpha === true
1697
+ ? "- Alpha: yes"
1698
+ : metadata?.hasAlpha === false
1699
+ ? "- Alpha: no"
1700
+ : "- Alpha: unknown",
1701
+ "",
1702
+ `If you want to analyze the image, call inspect_image with path="${formatPathRelativeToCwd(
1703
+ absolutePath,
1704
+ this.session.cwd,
1705
+ )}" and a question describing what to inspect and the desired output format.`,
1706
+ ];
1707
+ content = [{ type: "text", text: metadataLines.join("\n") }];
1708
+ details = {};
1709
+ sourcePath = absolutePath;
1710
+ } else {
1711
+ if (fileSize > MAX_IMAGE_SIZE) {
1712
+ const sizeStr = formatBytes(fileSize);
1713
+ const maxStr = formatBytes(MAX_IMAGE_SIZE);
1714
+ throw new ToolError(`Image file too large: ${sizeStr} exceeds ${maxStr} limit.`);
1715
+ }
1716
+ try {
1717
+ const imageInput = await loadImageInput({
1718
+ path: readPath,
1719
+ cwd: this.session.cwd,
1720
+ autoResize: this.#autoResizeImages,
1721
+ maxBytes: MAX_IMAGE_SIZE,
1722
+ resolvedPath: absolutePath,
1723
+ detectedMimeType: mimeType,
1724
+ });
1725
+ if (!imageInput) {
1726
+ throw new ToolError(`Read image file [${mimeType}] failed: unsupported image format.`);
1727
+ }
1728
+ content = [
1729
+ { type: "text", text: imageInput.textNote },
1730
+ { type: "image", data: imageInput.data, mimeType: imageInput.mimeType },
1731
+ ];
1732
+ details = {};
1733
+ sourcePath = imageInput.resolvedPath;
1734
+ } catch (error) {
1735
+ if (error instanceof ImageInputTooLargeError) {
1736
+ throw new ToolError(error.message);
1737
+ }
1738
+ throw error;
1739
+ }
1740
+ }
1741
+ } else if (isNotebookPath(absolutePath) && !isRawSelector(parsed)) {
1742
+ const notebookText = await readEditableNotebookText(absolutePath, localReadPath);
1743
+ if (isMultiRange(parsed) && parsed.kind === "lines") {
1744
+ return this.#buildInMemoryMultiRangeResult(notebookText, parsed.ranges, {
1745
+ details: { resolvedPath: absolutePath },
1746
+ sourcePath: absolutePath,
1747
+ entityLabel: "notebook",
1748
+ });
1749
+ }
1750
+ const { offset, limit } = selToOffsetLimit(parsed);
1751
+ return this.#buildInMemoryTextResult(notebookText, offset, limit, {
1752
+ details: { resolvedPath: absolutePath },
1753
+ sourcePath: absolutePath,
1754
+ entityLabel: "notebook",
1755
+ });
1756
+ } else if (shouldConvertWithMarkit) {
1757
+ // Convert document via markit.
1758
+ const result = await convertFileWithMarkit(absolutePath, signal);
1759
+ if (result.ok) {
1760
+ // Apply truncation to converted content
1761
+ const truncation = truncateHead(result.content);
1762
+ const outputText = truncation.content;
1763
+
1764
+ details = { truncation };
1765
+ sourcePath = absolutePath;
1766
+ truncationInfo = { result: truncation, options: { direction: "head", startLine: 1 } };
1767
+
1768
+ content = [{ type: "text", text: outputText }];
1769
+ } else if (result.error) {
1770
+ content = [{ type: "text", text: `[Cannot read ${ext} file: ${result.error || "conversion failed"}]` }];
1771
+ } else {
1772
+ content = [{ type: "text", text: `[Cannot read ${ext} file: conversion failed]` }];
1773
+ }
1774
+ } else {
1775
+ if (
1776
+ parsed.kind === "none" &&
1777
+ this.session.settings.get("read.summarize.enabled") &&
1778
+ (this.session.settings.get("read.summarize.prose") || !PROSE_SUMMARY_EXTENSIONS.has(ext))
1779
+ ) {
1780
+ const summary = await this.#trySummarize(absolutePath, fileSize, signal);
1781
+ if (summary?.parsed && summary.elided) {
1782
+ const renderedSummary = this.#renderSummary(summary);
1783
+ const footer = formatSummaryElisionFooter(
1784
+ localReadPath,
1785
+ renderedSummary.elidedRanges,
1786
+ renderedSummary.elidedLines,
1787
+ );
1788
+ const summaryHashContext = displayMode.hashLines
1789
+ ? await readHashlineHeaderContext(this.session, absolutePath, this.session.cwd)
1790
+ : undefined;
1791
+ const bodyText = footer ? `${renderedSummary.text}\n\n${footer}` : renderedSummary.text;
1792
+ const modelText = prependHashlineHeader(bodyText, summaryHashContext);
1793
+ details = {
1794
+ displayContent: { text: renderedSummary.displayText, startLine: 1 },
1795
+ summary: {
1796
+ lines: countTextLines(renderedSummary.text),
1797
+ elidedSpans: renderedSummary.elidedRanges.length,
1798
+ elidedLines: renderedSummary.elidedLines,
1799
+ },
1800
+ };
1801
+
1802
+ sourcePath = absolutePath;
1803
+ content = [{ type: "text", text: modelText }];
1804
+ }
1805
+ }
1806
+
1807
+ if (!content) {
1808
+ if (isMultiRange(parsed) && parsed.kind === "lines") {
1809
+ const multiResult = await this.#readLocalFileMultiRange(
1810
+ absolutePath,
1811
+ parsed.ranges,
1812
+ parsed,
1813
+ displayMode,
1814
+ suffixResolution,
1815
+ signal,
1816
+ );
1817
+ if (multiResult.bridgeResult) return multiResult.bridgeResult;
1818
+ content = [{ type: "text", text: multiResult.outputText }];
1819
+ sourcePath = absolutePath;
1820
+ details = {};
1821
+ if (multiResult.columnTruncated > 0) {
1822
+ columnTruncated = multiResult.columnTruncated;
1823
+ }
1824
+ } else {
1825
+ // Raw text or line-range mode
1826
+ const { offset, limit } = selToOffsetLimit(parsed);
1827
+ // Try ACP bridge first — editor's in-memory buffer is source of truth.
1828
+ // Request full text so local range rendering keeps normal context and line numbers.
1829
+ const bridgePromise = this.#routeReadThroughBridge(absolutePath);
1830
+ if (bridgePromise !== undefined) {
1831
+ try {
1832
+ const bridgeText = await bridgePromise;
1833
+ const bridgeResult = this.#buildInMemoryTextResult(bridgeText, offset, limit, {
1834
+ details: { resolvedPath: absolutePath, suffixResolution },
1835
+ sourcePath: absolutePath,
1836
+ entityLabel: "file",
1837
+ raw: isRawSelector(parsed),
1838
+ });
1839
+ if (suffixResolution) {
1840
+ const notice = `[Path '${suffixResolution.from}' not found; resolved to '${suffixResolution.to}' via suffix match]`;
1841
+ const firstText = bridgeResult.content.find((c): c is TextContent => c.type === "text");
1842
+ if (firstText) firstText.text = `${notice}\n${firstText.text}`;
1843
+ }
1844
+ return bridgeResult;
1845
+ } catch (error) {
1846
+ logger.warn("ACP fs readTextFile failed; falling back to disk", { path: absolutePath, error });
1847
+ }
1848
+ }
1849
+
1850
+ // User-requested 0-indexed range start. Lines BEFORE this become
1851
+ // leading context (added below if offset is explicit).
1852
+ const requestedStart = offset ? Math.max(0, offset - 1) : 0;
1853
+ const expandStart = offset !== undefined && offset > 1;
1854
+ const expandEnd = limit !== undefined;
1855
+ const leadingContext = expandStart ? Math.min(requestedStart, RANGE_LEADING_CONTEXT_LINES) : 0;
1856
+ const trailingContext = expandEnd ? RANGE_TRAILING_CONTEXT_LINES : 0;
1857
+ const startLine = requestedStart - leadingContext;
1858
+ const startLineDisplay = startLine + 1;
1859
+
1860
+ const DEFAULT_LIMIT = this.#defaultLimit;
1861
+ const effectiveLimit = limit ?? DEFAULT_LIMIT;
1862
+ const maxLinesToCollect = Math.min(effectiveLimit + leadingContext + trailingContext, DEFAULT_MAX_LINES);
1863
+ const selectedLineLimit = effectiveLimit + leadingContext + trailingContext;
1864
+ // Scale byte budget with line limit so the configured line count actually fits.
1865
+ // Assume ~512 bytes/line average; never go below the shared default.
1866
+ const maxBytesForRead = Math.max(DEFAULT_MAX_BYTES, maxLinesToCollect * 512);
1867
+
1868
+ const streamResult = await streamLinesFromFile(
1869
+ absolutePath,
1870
+ startLine,
1871
+ maxLinesToCollect,
1872
+ maxBytesForRead,
1873
+ selectedLineLimit,
1874
+ signal,
1875
+ );
1876
+
1877
+ const {
1878
+ lines: collectedLines,
1879
+ totalFileLines,
1880
+ collectedBytes,
1881
+ stoppedByByteLimit,
1882
+ firstLinePreview,
1883
+ firstLineByteLength,
1884
+ } = streamResult;
1885
+
1886
+ // Check if offset is out of bounds - return graceful message instead of throwing
1887
+ if (requestedStart >= totalFileLines) {
1888
+ const suggestion =
1889
+ totalFileLines === 0
1890
+ ? "The file is empty."
1891
+ : `Use :1 to read from the start, or :${totalFileLines} to read the last line.`;
1892
+ return toolResult<ReadToolDetails>({ resolvedPath: absolutePath, suffixResolution })
1893
+ .text(
1894
+ `Line ${requestedStart + 1} is beyond end of file (${totalFileLines} lines total). ${suggestion}`,
1895
+ )
1896
+ .done();
1897
+ }
1898
+
1899
+ // Per-line column cap. Skipped in raw mode so `:raw` always returns
1900
+ // verbatim bytes for paste-back-into-tool workflows. Total byte/line
1901
+ // counts in `truncation` keep reflecting the source, not the trimmed
1902
+ // view — column truncation surfaces separately via `.limits()`.
1903
+ const rawSelector = isRawSelector(parsed);
1904
+ const maxColumns = resolveOutputMaxColumns(this.session.settings);
1905
+ // Column truncation is display-only. `collectedLines` MUST stay
1906
+ // byte-for-byte with the on-disk content so the snapshot recorded
1907
+ // below can be verified against the live file. Mutating it with
1908
+ // ellipsis-truncated text made every long-line file uneditable on
1909
+ // the next edit attempt.
1910
+ let displayLines: string[] = collectedLines;
1911
+ if (!rawSelector && maxColumns > 0) {
1912
+ let cloned: string[] | undefined;
1913
+ for (let i = 0; i < collectedLines.length; i++) {
1914
+ const { text, wasTruncated } = truncateLine(collectedLines[i], maxColumns);
1915
+ if (wasTruncated) {
1916
+ if (!cloned) cloned = collectedLines.slice();
1917
+ cloned[i] = text;
1918
+ columnTruncated = maxColumns;
1919
+ }
1920
+ }
1921
+ if (cloned) displayLines = cloned;
1922
+ }
1923
+
1924
+ const selectedContent = displayLines.join("\n");
1925
+ const userLimitedLines = collectedLines.length;
1926
+
1927
+ const totalSelectedLines = totalFileLines - startLine;
1928
+ const totalSelectedBytes = collectedBytes;
1929
+ const wasTruncated = collectedLines.length < totalSelectedLines || stoppedByByteLimit;
1930
+ const firstLineExceedsLimit = firstLineByteLength !== undefined && firstLineByteLength > maxBytesForRead;
1931
+
1932
+ const truncation: TruncationResult = {
1933
+ content: selectedContent,
1934
+ truncated: wasTruncated,
1935
+ truncatedBy: stoppedByByteLimit ? "bytes" : wasTruncated ? "lines" : undefined,
1936
+ totalLines: totalSelectedLines,
1937
+ totalBytes: totalSelectedBytes,
1938
+ outputLines: collectedLines.length,
1939
+ outputBytes: collectedBytes,
1940
+ lastLinePartial: false,
1941
+ firstLineExceedsLimit,
1942
+ };
1943
+
1944
+ const shouldAddHashLines = !rawSelector && displayMode.hashLines;
1945
+ const shouldAddLineNumbers = rawSelector ? false : shouldAddHashLines ? false : displayMode.lineNumbers;
1946
+ let hashContext: HashlineHeaderContext | undefined;
1947
+ if (shouldAddHashLines && collectedLines.length > 0 && !firstLineExceedsLimit) {
1948
+ // The tag is a content hash of the WHOLE file. A whole-file read
1949
+ // already holds every line in memory; a range read re-reads the
1950
+ // file (bounded by SNAPSHOT_MAX_BYTES) so the tag fingerprints the
1951
+ // full file and any anchor validates while the file is unchanged.
1952
+ const isWholeFile = offset === undefined && limit === undefined && !wasTruncated;
1953
+ const tag = isWholeFile
1954
+ ? getFileSnapshotStore(this.session).record(absolutePath, normalizeToLF(collectedLines.join("\n")))
1955
+ : await recordFileSnapshot(this.session, absolutePath);
1956
+ if (tag) {
1957
+ hashContext = hashlineHeaderContext(formatPathRelativeToCwd(absolutePath, this.session.cwd), tag);
1958
+ }
1959
+ }
1960
+
1961
+ let capturedDisplayContent: { text: string; startLine: number } | undefined;
1962
+ let emittedHashlineHeader = false;
1963
+ const formatText = (text: string, startNum: number): string => {
1964
+ capturedDisplayContent = { text, startLine: startNum };
1965
+ const formatted = formatTextWithMode(text, startNum, shouldAddHashLines, shouldAddLineNumbers);
1966
+ if (!hashContext || emittedHashlineHeader) return formatted;
1967
+ emittedHashlineHeader = true;
1968
+ return prependHashlineHeader(formatted, hashContext);
1969
+ };
1970
+
1971
+ let outputText: string;
1972
+
1973
+ if (truncation.firstLineExceedsLimit) {
1974
+ const firstLineBytes = firstLineByteLength ?? 0;
1975
+ const snippet = firstLinePreview ?? { text: "", bytes: 0 };
1976
+
1977
+ if (shouldAddHashLines) {
1978
+ outputText = `[Line ${startLineDisplay} is ${formatBytes(
1979
+ firstLineBytes,
1980
+ )}, exceeds ${formatBytes(maxBytesForRead)} limit. Hashline output requires full lines; cannot emit an editable numbered preview for a truncated line.]`;
1981
+ } else {
1982
+ outputText = formatText(snippet.text, startLineDisplay);
1983
+ }
1984
+ if (snippet.text.length === 0) {
1985
+ outputText = `[Line ${startLineDisplay} is ${formatBytes(
1986
+ firstLineBytes,
1987
+ )}, exceeds ${formatBytes(maxBytesForRead)} limit. Unable to display a valid UTF-8 snippet.]`;
1988
+ }
1989
+ details = { truncation };
1990
+ sourcePath = absolutePath;
1991
+ truncationInfo = {
1992
+ result: truncation,
1993
+ options: { direction: "head", startLine: startLineDisplay, totalFileLines },
1994
+ };
1995
+ } else if (truncation.truncated) {
1996
+ outputText = formatText(truncation.content, startLineDisplay);
1997
+ details = { truncation };
1998
+ sourcePath = absolutePath;
1999
+ truncationInfo = {
2000
+ result: truncation,
2001
+ options: { direction: "head", startLine: startLineDisplay, totalFileLines },
2002
+ };
2003
+ } else if (startLine + userLimitedLines < totalFileLines) {
2004
+ const remaining = totalFileLines - (startLine + userLimitedLines);
2005
+ const nextOffset = startLine + userLimitedLines + 1;
2006
+
2007
+ outputText = formatText(truncation.content, startLineDisplay);
2008
+ outputText += `\n\n[${remaining} more lines in file. Use :${nextOffset} to continue]`;
2009
+ details = {};
2010
+ sourcePath = absolutePath;
2011
+ } else {
2012
+ // No truncation, no user limit exceeded
2013
+ outputText = formatText(truncation.content, startLineDisplay);
2014
+ details = {};
2015
+ sourcePath = absolutePath;
2016
+ }
2017
+
2018
+ if (capturedDisplayContent) {
2019
+ details.displayContent = capturedDisplayContent;
2020
+ }
2021
+
2022
+ if (!firstLineExceedsLimit && collectedLines.length > 0) {
2023
+ const blocks = scanConflictLines(collectedLines, startLineDisplay);
2024
+ if (blocks.length > 0) {
2025
+ const history = getConflictHistory(this.session);
2026
+ const displayPathForWarning = formatPathRelativeToCwd(absolutePath, this.session.cwd);
2027
+ const entries = blocks.map(block =>
2028
+ history.register({
2029
+ absolutePath,
2030
+ displayPath: displayPathForWarning,
2031
+ ...block,
2032
+ }),
2033
+ );
2034
+ // Cheap full-file scan only when the window already showed
2035
+ // at least one conflict — otherwise pay nothing on clean files.
2036
+ let totalInFile = entries.length;
2037
+ let scanTruncated = false;
2038
+ try {
2039
+ const fileScan = await scanFileForConflicts(absolutePath);
2040
+ totalInFile = Math.max(entries.length, fileScan.blocks.length);
2041
+ scanTruncated = fileScan.scanTruncated;
2042
+ } catch {
2043
+ // Best-effort enrichment; fall back to window-only count.
2044
+ }
2045
+ outputText += formatConflictWarning(entries, {
2046
+ totalInFile,
2047
+ displayPath: displayPathForWarning,
2048
+ scanTruncated,
2049
+ });
2050
+ details.conflictCount = entries.length;
2051
+ }
2052
+ }
2053
+
2054
+ content = [{ type: "text", text: outputText }];
2055
+ }
2056
+ }
2057
+ }
2058
+
2059
+ if (suffixResolution) {
2060
+ details.suffixResolution = suffixResolution;
2061
+ // Inline resolution notice into first text block so the model sees the actual path
2062
+ const notice = `[Path '${suffixResolution.from}' not found; resolved to '${suffixResolution.to}' via suffix match]`;
2063
+ const firstText = content.find((c): c is TextContent => c.type === "text");
2064
+ if (firstText) {
2065
+ firstText.text = `${notice}\n${firstText.text}`;
2066
+ } else {
2067
+ content = [{ type: "text", text: notice }, ...content];
2068
+ }
2069
+ }
2070
+ const resultBuilder = toolResult(details).content(content);
2071
+ if (sourcePath) {
2072
+ resultBuilder.sourcePath(sourcePath);
2073
+ }
2074
+ if (truncationInfo) {
2075
+ resultBuilder.truncation(truncationInfo.result, truncationInfo.options);
2076
+ }
2077
+ if (columnTruncated > 0) {
2078
+ resultBuilder.limits({ columnMax: columnTruncated });
2079
+ }
2080
+ return resultBuilder.done();
2081
+ }
2082
+
2083
+ /**
2084
+ * Render a `conflict://<N>` (or `conflict://<N>/<scope>`) region as
2085
+ * regular file content. The lines are emitted with their original
2086
+ * file line numbers so hashline anchors line up with the source
2087
+ * file, and no truncation footer is appended.
2088
+ */
2089
+ async #readConflictRegion(id: number, scope: ConflictScope | undefined): Promise<AgentToolResult<ReadToolDetails>> {
2090
+ const entry: ConflictEntry | undefined = getConflictHistory(this.session).get(id);
2091
+ if (!entry) {
2092
+ throw new ToolError(
2093
+ `Conflict #${id} not found. Conflict ids are registered when \`read\` surfaces a marker block; re-read the file to get a current id.`,
2094
+ );
2095
+ }
2096
+
2097
+ const region = renderConflictRegion(entry, scope);
2098
+ const displayMode = resolveFileDisplayMode(this.session);
2099
+ const shouldAddHashLines = displayMode.hashLines;
2100
+ const shouldAddLineNumbers = shouldAddHashLines ? false : displayMode.lineNumbers;
2101
+
2102
+ const rawText = region.lines.join("\n");
2103
+ const tag = shouldAddHashLines ? await recordFileSnapshot(this.session, entry.absolutePath) : undefined;
2104
+ const hashContext = tag
2105
+ ? hashlineHeaderContext(formatPathRelativeToCwd(entry.absolutePath, this.session.cwd), tag)
2106
+ : undefined;
2107
+ const formattedBody = formatTextWithMode(rawText, region.startLine, shouldAddHashLines, shouldAddLineNumbers);
2108
+ const formattedText = prependHashlineHeader(formattedBody, hashContext);
2109
+
2110
+ const details: ReadToolDetails = {
2111
+ resolvedPath: entry.absolutePath,
2112
+ displayContent: { text: rawText, startLine: region.startLine },
2113
+ };
2114
+ return toolResult<ReadToolDetails>(details).text(formattedText).sourcePath(entry.absolutePath).done();
2115
+ }
2116
+
2117
+ /**
2118
+ * Implement the `<path>:conflicts` read selector: scan the whole file once, register
2119
+ * every block in the session's conflict history, and return a compact
2120
+ * `#N L_a-L_b` index instead of file content. Designed for heavily
2121
+ * conflicted files where dumping every body would be wasteful.
2122
+ */
2123
+ async #readFileConflicts(
2124
+ absolutePath: string,
2125
+ suffixResolution: { from: string; to: string } | undefined,
2126
+ signal: AbortSignal | undefined,
2127
+ ): Promise<AgentToolResult<ReadToolDetails>> {
2128
+ throwIfAborted(signal);
2129
+ const scan = await scanFileForConflicts(absolutePath);
2130
+ const displayPath = formatPathRelativeToCwd(absolutePath, this.session.cwd);
2131
+ const history = getConflictHistory(this.session);
2132
+ const entries = scan.blocks.map(block =>
2133
+ history.register({
2134
+ absolutePath,
2135
+ displayPath,
2136
+ ...block,
2137
+ }),
2138
+ );
2139
+
2140
+ const summary =
2141
+ entries.length === 0
2142
+ ? `No unresolved git merge conflicts in ${displayPath}.`
2143
+ : formatConflictSummary(entries, { displayPath, scanTruncated: scan.scanTruncated });
2144
+
2145
+ const details: ReadToolDetails = {
2146
+ resolvedPath: absolutePath,
2147
+ suffixResolution,
2148
+ conflictCount: entries.length,
2149
+ };
2150
+ return toolResult<ReadToolDetails>(details).text(summary).sourcePath(absolutePath).done();
2151
+ }
2152
+
2153
+ /**
2154
+ * Handle internal URLs (agent://, artifact://, memory://, skill://, rule://, local://, mcp://).
2155
+ * Supports pagination via offset/limit but rejects them when query extraction is used.
2156
+ */
2157
+ async #handleInternalUrl(
2158
+ url: string,
2159
+ parsedSel: ParsedSelector,
2160
+ signal?: AbortSignal,
2161
+ ): Promise<AgentToolResult<ReadToolDetails>> {
2162
+ const internalRouter = InternalUrlRouter.instance();
2163
+
2164
+ // Check if URL has query extraction (agent:// only).
2165
+ // Use parseInternalUrl which handles colons in host (namespaced skills).
2166
+ let urlMeta: InternalUrl;
2167
+ try {
2168
+ urlMeta = parseInternalUrl(url);
2169
+ } catch (e) {
2170
+ throw new ToolError(e instanceof Error ? e.message : String(e));
2171
+ }
2172
+ const scheme = urlMeta.protocol.replace(/:$/, "").toLowerCase();
2173
+ let hasExtraction = false;
2174
+ if (scheme === "agent") {
2175
+ const hasPathExtraction = urlMeta.pathname && urlMeta.pathname !== "/" && urlMeta.pathname !== "";
2176
+ const queryParam = urlMeta.searchParams.get("q");
2177
+ const hasQueryExtraction = queryParam !== null && queryParam !== "";
2178
+ hasExtraction = hasPathExtraction || hasQueryExtraction;
2179
+ }
2180
+
2181
+ // Reject line selectors when query extraction is used
2182
+ if (hasExtraction && parsedSel.kind !== "none" && parsedSel.kind !== "raw") {
2183
+ throw new ToolError("Cannot combine query extraction with line selectors");
2184
+ }
2185
+
2186
+ // Resolve the internal URL
2187
+ const resource = await internalRouter.resolve(url, {
2188
+ cwd: this.session.cwd,
2189
+ settings: this.session.settings,
2190
+ signal,
2191
+ localProtocolOptions: this.session.localProtocolOptions,
2192
+ });
2193
+ const details: ReadToolDetails = { resolvedPath: resource.sourcePath, contentType: resource.contentType };
2194
+
2195
+ // If extraction was used, return directly (no pagination)
2196
+ if (hasExtraction) {
2197
+ return toolResult(details).text(resource.content).sourceInternal(url).done();
2198
+ }
2199
+
2200
+ const raw = isRawSelector(parsedSel);
2201
+ if (isMultiRange(parsedSel) && parsedSel.kind === "lines") {
2202
+ return this.#buildInMemoryMultiRangeResult(resource.content, parsedSel.ranges, {
2203
+ details,
2204
+ sourcePath: resource.sourcePath,
2205
+ sourceInternal: url,
2206
+ entityLabel: "resource",
2207
+ immutable: resource.immutable,
2208
+ raw,
2209
+ });
2210
+ }
2211
+
2212
+ const { offset, limit } = selToOffsetLimit(parsedSel);
2213
+ return this.#buildInMemoryTextResult(resource.content, offset, limit, {
2214
+ details,
2215
+ sourcePath: resource.sourcePath,
2216
+ sourceInternal: url,
2217
+ entityLabel: "resource",
2218
+ ignoreResultLimits: scheme === "skill",
2219
+ immutable: resource.immutable,
2220
+ raw,
2221
+ });
2222
+ }
2223
+
2224
+ /** Read directory contents as a formatted listing */
2225
+ async #readDirectory(
2226
+ absolutePath: string,
2227
+ offset: number | undefined,
2228
+ limit: number | undefined,
2229
+ signal?: AbortSignal,
2230
+ ): Promise<AgentToolResult<ReadToolDetails>> {
2231
+ const READ_DIRECTORY_MAX_DEPTH = 2;
2232
+ const READ_DIRECTORY_CHILD_LIMIT = 12;
2233
+
2234
+ throwIfAborted(signal);
2235
+ let tree: DirectoryTree;
2236
+ try {
2237
+ tree = await buildDirectoryTree(absolutePath, {
2238
+ maxDepth: READ_DIRECTORY_MAX_DEPTH,
2239
+ perDirLimit: READ_DIRECTORY_CHILD_LIMIT,
2240
+ rootLimit: null,
2241
+ // `lineCap` truncates the rendered tree itself, so apply it only when the caller
2242
+ // did not request an offset — otherwise we'd cap the first N lines before slicing.
2243
+ lineCap: offset === undefined && limit !== undefined ? limit : null,
2244
+ });
2245
+ } catch (error) {
2246
+ const message = error instanceof Error ? error.message : String(error);
2247
+ throw new ToolError(`Cannot read directory: ${message}`);
2248
+ }
2249
+ throwIfAborted(signal);
2250
+
2251
+ const output = tree.totalLines <= 1 ? "(empty directory)" : tree.rendered;
2252
+ const details: ReadToolDetails = {
2253
+ isDirectory: true,
2254
+ resolvedPath: tree.rootPath,
2255
+ };
2256
+
2257
+ // Slice the rendered listing when the caller passed an offset/limit. We do this
2258
+ // instead of passing the selector down to `buildDirectoryTree` because the tree
2259
+ // builder lays out entries hierarchically (per-dir caps, recent-then-elided
2260
+ // summaries); line-based slicing operates on the formatted text and matches what
2261
+ // users expect from `:N-M` on long listings.
2262
+ const wantsSlice = offset !== undefined || limit !== undefined;
2263
+ if (wantsSlice) {
2264
+ const allLines = output.split("\n");
2265
+ const start = offset ? Math.max(0, offset - 1) : 0;
2266
+ if (start >= allLines.length) {
2267
+ const suggestion =
2268
+ allLines.length === 0
2269
+ ? "The listing is empty."
2270
+ : `Use :1 to read from the start, or :${allLines.length} to read the last line.`;
2271
+ return toolResult(details)
2272
+ .text(`Line ${start + 1} is beyond end of listing (${allLines.length} lines total). ${suggestion}`)
2273
+ .sourcePath(tree.rootPath)
2274
+ .done();
2275
+ }
2276
+ const end = limit !== undefined ? Math.min(start + limit, allLines.length) : allLines.length;
2277
+ const sliced = allLines.slice(start, end).join("\n");
2278
+ const resultBuilder = toolResult(details).sourcePath(tree.rootPath);
2279
+ let text = sliced;
2280
+ if (end < allLines.length) {
2281
+ const remaining = allLines.length - end;
2282
+ text += `\n\n[${remaining} more lines in listing. Use :${end + 1} to continue]`;
2283
+ }
2284
+ resultBuilder.text(text);
2285
+ if (tree.truncated) {
2286
+ resultBuilder.limits({ resultLimit: 1 });
2287
+ }
2288
+ return resultBuilder.done();
2289
+ }
2290
+
2291
+ const truncation = truncateHead(output, { maxLines: Number.MAX_SAFE_INTEGER });
2292
+ const resultBuilder = toolResult(details).text(truncation.content).sourcePath(tree.rootPath);
2293
+ if (tree.truncated) {
2294
+ resultBuilder.limits({ resultLimit: 1 });
2295
+ }
2296
+ if (truncation.truncated) {
2297
+ resultBuilder.truncation(truncation, { direction: "head" });
2298
+ details.truncation = truncation;
2299
+ }
2300
+
2301
+ return resultBuilder.done();
2302
+ }
2303
+ }
2304
+
2305
+ // =============================================================================
2306
+ // TUI Renderer
2307
+ // =============================================================================
2308
+
2309
+ interface ReadRenderArgs {
2310
+ path?: string;
2311
+ file_path?: string;
2312
+ sel?: string;
2313
+ // Legacy fields from old schema — tolerated for in-flight tool calls during transition
2314
+ offset?: number;
2315
+ limit?: number;
2316
+ raw?: boolean;
2317
+ }
2318
+
2319
+ const INTERNAL_URL_LIKE_RE = /^[a-z][a-z0-9+.-]*:\/\//i;
2320
+
2321
+ function splitReadRenderPath(rawPath: string): { path: string; sel?: string } {
2322
+ if (INTERNAL_URL_LIKE_RE.test(rawPath)) {
2323
+ const internal = splitInternalUrlSel(rawPath);
2324
+ if (internal.sel) return internal;
2325
+ }
2326
+ return splitPathAndSel(rawPath);
2327
+ }
2328
+
2329
+ function firstReadSelectorLine(sel: string | undefined): number | undefined {
2330
+ if (!sel) return undefined;
2331
+ try {
2332
+ const parsed = parseSel(sel);
2333
+ if (parsed.kind !== "lines") return undefined;
2334
+ return parsed.ranges[0].startLine;
2335
+ } catch {
2336
+ return undefined;
2337
+ }
2338
+ }
2339
+
2340
+ function formatReadPathLink(
2341
+ rawPath: string,
2342
+ options: {
2343
+ resolvedPath?: string;
2344
+ suffixResolution?: { from: string; to: string };
2345
+ offset?: number;
2346
+ fallbackLabel?: string;
2347
+ },
2348
+ ): string {
2349
+ const split = splitReadRenderPath(rawPath);
2350
+ const basePath = split.path || rawPath;
2351
+ const selectorSuffix = split.sel ? `:${split.sel}` : "";
2352
+ const plainDisplayPath = options.suffixResolution
2353
+ ? shortenPath(options.suffixResolution.to)
2354
+ : shortenPath(basePath || options.resolvedPath || options.fallbackLabel || rawPath);
2355
+ const target = options.resolvedPath ?? tryResolveInternalUrlSync(basePath);
2356
+ const line = firstReadSelectorLine(split.sel) ?? options.offset;
2357
+ const linkOptions = line !== undefined ? { line } : undefined;
2358
+ const displayPath = target ? fileHyperlink(target, plainDisplayPath, linkOptions) : plainDisplayPath;
2359
+ return `${displayPath}${selectorSuffix}`;
2360
+ }
2361
+
2362
+ export const readToolRenderer = {
2363
+ renderCall(args: ReadRenderArgs, _options: RenderResultOptions, uiTheme: Theme): Component {
2364
+ if (isReadableUrlPath(args.file_path || args.path || "")) {
2365
+ return renderReadUrlCall(args, _options, uiTheme);
2366
+ }
2367
+
2368
+ const rawPath = args.file_path || args.path || "";
2369
+ const offset = args.offset;
2370
+ const limit = args.limit;
2371
+
2372
+ let pathDisplay = formatReadPathLink(rawPath, { offset, fallbackLabel: "…" }) || "…";
2373
+ if (offset !== undefined || limit !== undefined) {
2374
+ const startLine = offset ?? 1;
2375
+ const endLine = limit !== undefined ? startLine + limit - 1 : "";
2376
+ pathDisplay += `:${startLine}${endLine ? `-${endLine}` : ""}`;
2377
+ }
2378
+
2379
+ const text = renderStatusLine({ icon: "pending", title: "Read", description: pathDisplay }, uiTheme);
2380
+ return new Text(text, 0, 0);
2381
+ },
2382
+
2383
+ renderResult(
2384
+ result: { content: Array<{ type: string; text?: string }>; details?: ReadToolDetails; isError?: boolean },
2385
+ options: RenderResultOptions,
2386
+ uiTheme: Theme,
2387
+ args?: ReadRenderArgs,
2388
+ ): Component {
2389
+ const urlDetails = result.details as ReadUrlToolDetails | undefined;
2390
+ if (urlDetails?.kind === "url" || isReadableUrlPath(args?.file_path || args?.path || "")) {
2391
+ return renderReadUrlResult(
2392
+ result as {
2393
+ content: Array<{ type: string; text?: string }>;
2394
+ details?: ReadUrlToolDetails;
2395
+ isError?: boolean;
2396
+ },
2397
+ options,
2398
+ uiTheme,
2399
+ );
2400
+ }
2401
+
2402
+ if (result.isError) {
2403
+ const rawErrorText = result.content?.find(c => c.type === "text")?.text ?? "";
2404
+ const errorText = (rawErrorText || "Unknown error").replace(/^Error:\s*/, "");
2405
+ const rawPath = args?.file_path || args?.path || "";
2406
+ const filePath = formatReadPathLink(rawPath, { offset: args?.offset }) || shortenPath(rawPath);
2407
+ let title = filePath ? `Read ${filePath}` : "Read";
2408
+ if (args?.offset !== undefined || args?.limit !== undefined) {
2409
+ const startLine = args.offset ?? 1;
2410
+ const endLine = args.limit !== undefined ? startLine + args.limit - 1 : "";
2411
+ title += `:${startLine}${endLine ? `-${endLine}` : ""}`;
2412
+ }
2413
+ const header = renderStatusLine({ icon: "error", title }, uiTheme);
2414
+ const errorLines = errorText.split("\n").map(line => uiTheme.fg("error", replaceTabs(line)));
2415
+ const outputBlock = new CachedOutputBlock();
2416
+ return markFramedBlockComponent({
2417
+ render: (width: number) =>
2418
+ outputBlock.render({ header, state: "error", sections: [{ lines: errorLines }], width }, uiTheme),
2419
+ invalidate: () => outputBlock.invalidate(),
2420
+ });
2421
+ }
2422
+ const details = result.details;
2423
+ const rawText = result.content?.find(c => c.type === "text")?.text ?? "";
2424
+ // Prefer structured `displayContent` from details when available so the TUI
2425
+ // shows clean file content (no model-only hashline anchors) without parsing the formatted text.
2426
+ // Fall back to the raw text, but strip the LLM-facing notice so it doesn't
2427
+ // echo next to the styled warning line below.
2428
+ const contentText = details?.displayContent?.text ?? stripOutputNotice(rawText, details?.meta);
2429
+ const imageContent = result.content?.find(c => c.type === "image");
2430
+ const rawPath = args?.file_path || args?.path || "";
2431
+ const renderPath = splitReadRenderPath(rawPath);
2432
+ const lang = getLanguageFromPath(renderPath.path);
2433
+
2434
+ const warningLines: string[] = [];
2435
+ const truncation = details?.meta?.truncation;
2436
+ const fallback = details?.truncation;
2437
+ if (details?.resolvedPath) {
2438
+ warningLines.push(uiTheme.fg("dim", wrapBrackets(`Resolved path: ${details.resolvedPath}`, uiTheme)));
2439
+ }
2440
+ if (truncation) {
2441
+ if (fallback?.firstLineExceedsLimit) {
2442
+ let warning = `First line exceeds ${formatBytes(fallback.outputBytes ?? fallback.totalBytes)} limit`;
2443
+ if (truncation.artifactId) {
2444
+ warning += `. ${formatFullOutputReference(truncation.artifactId)}`;
2445
+ }
2446
+ warningLines.push(uiTheme.fg("warning", wrapBrackets(warning, uiTheme)));
2447
+ } else {
2448
+ const warning = formatStyledTruncationWarning(details?.meta, uiTheme);
2449
+ if (warning) warningLines.push(warning);
2450
+ }
2451
+ }
2452
+
2453
+ if (imageContent) {
2454
+ const suffix = details?.suffixResolution;
2455
+ const displayPath = formatReadPathLink(rawPath, {
2456
+ resolvedPath: details?.resolvedPath,
2457
+ suffixResolution: suffix,
2458
+ fallbackLabel: "image",
2459
+ });
2460
+ const correction = suffix ? ` ${uiTheme.fg("dim", `(corrected from ${shortenPath(suffix.from)})`)}` : "";
2461
+ const header = renderStatusLine(
2462
+ { icon: suffix ? "warning" : "success", title: "Read", description: `${displayPath}${correction}` },
2463
+ uiTheme,
2464
+ );
2465
+ const detailLines = contentText ? contentText.split("\n").map(line => uiTheme.fg("toolOutput", line)) : [];
2466
+ const lines = [...detailLines, ...warningLines];
2467
+ const outputBlock = new CachedOutputBlock();
2468
+ return markFramedBlockComponent({
2469
+ render: (width: number) =>
2470
+ outputBlock.render(
2471
+ {
2472
+ header,
2473
+ state: "success",
2474
+ sections: [
2475
+ {
2476
+ label: uiTheme.fg("toolTitle", "Details"),
2477
+ lines: lines.length > 0 ? lines : [uiTheme.fg("dim", "(image)")],
2478
+ },
2479
+ ],
2480
+ width,
2481
+ },
2482
+ uiTheme,
2483
+ ),
2484
+ invalidate: () => outputBlock.invalidate(),
2485
+ });
2486
+ }
2487
+
2488
+ const suffix = details?.suffixResolution;
2489
+ // resolvedPath is the absolute fs path for fs-backed reads (regular files plus
2490
+ // local:// / memory:// / skill:// / artifact:// resources). Fall back to a sync
2491
+ // resolver for fs-backed internal URLs so the title is clickable even before the
2492
+ // result lands or if the handler didn't populate resolvedPath.
2493
+ const displayPath = formatReadPathLink(rawPath, {
2494
+ resolvedPath: details?.resolvedPath,
2495
+ suffixResolution: suffix,
2496
+ offset: args?.offset,
2497
+ });
2498
+ const correction = suffix ? ` ${uiTheme.fg("dim", `(corrected from ${shortenPath(suffix.from)})`)}` : "";
2499
+ let title = displayPath ? `Read ${displayPath}${correction}` : "Read";
2500
+ if (args?.offset !== undefined || args?.limit !== undefined) {
2501
+ const startLine = args.offset ?? 1;
2502
+ const endLine = args.limit !== undefined ? startLine + args.limit - 1 : "";
2503
+ title += `:${startLine}${endLine ? `-${endLine}` : ""}`;
2504
+ }
2505
+ if (details?.summary) {
2506
+ title += ` (summary: ${details.summary.elidedSpans} elided span${details.summary.elidedSpans === 1 ? "" : "s"})`;
2507
+ }
2508
+ if (details?.conflictCount && details.conflictCount > 0) {
2509
+ const n = details.conflictCount;
2510
+ title += ` ${uiTheme.fg("warning", `(⚠ ${n} conflict${n === 1 ? "" : "s"})`)}`;
2511
+ }
2512
+ const rawRequested = args?.raw === true || isRawSelector(parseSel(renderPath.sel));
2513
+ const isMarkdown = details?.contentType === "text/markdown" && !rawRequested;
2514
+ let cachedWidth: number | undefined;
2515
+ let cachedExpanded: boolean | undefined;
2516
+ let cachedLines: string[] | undefined;
2517
+ return markFramedBlockComponent({
2518
+ render: (width: number) => {
2519
+ const expanded = options.expanded;
2520
+ if (cachedLines && cachedWidth === width && cachedExpanded === expanded) return cachedLines;
2521
+ cachedLines = isMarkdown
2522
+ ? renderMarkdownCell(
2523
+ {
2524
+ content: contentText,
2525
+ title,
2526
+ status: "complete",
2527
+ output: warningLines.length > 0 ? warningLines.join("\n") : undefined,
2528
+ expanded,
2529
+ width,
2530
+ },
2531
+ uiTheme,
2532
+ )
2533
+ : renderCodeCell(
2534
+ {
2535
+ code: contentText,
2536
+ language: lang,
2537
+ title,
2538
+ status: "complete",
2539
+ output: warningLines.length > 0 ? warningLines.join("\n") : undefined,
2540
+ expanded,
2541
+ width,
2542
+ },
2543
+ uiTheme,
2544
+ );
2545
+ cachedWidth = width;
2546
+ cachedExpanded = expanded;
2547
+ return cachedLines;
2548
+ },
2549
+ invalidate: () => {
2550
+ cachedWidth = undefined;
2551
+ cachedExpanded = undefined;
2552
+ cachedLines = undefined;
2553
+ },
2554
+ });
2555
+ },
2556
+ mergeCallAndResult: true,
2557
+ };