oma-coding-agent 1.1.4

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 (1298) hide show
  1. package/CHANGELOG.md +12164 -0
  2. package/README.md +35 -0
  3. package/dist/cli.js +18266 -0
  4. package/examples/README.md +21 -0
  5. package/examples/custom-tools/README.md +104 -0
  6. package/examples/custom-tools/hello/index.ts +20 -0
  7. package/examples/extensions/README.md +142 -0
  8. package/examples/extensions/api-demo.ts +79 -0
  9. package/examples/extensions/chalk-logger.ts +25 -0
  10. package/examples/extensions/hello.ts +31 -0
  11. package/examples/extensions/pirate.ts +43 -0
  12. package/examples/extensions/plan-mode.ts +549 -0
  13. package/examples/extensions/reload-runtime.ts +38 -0
  14. package/examples/extensions/thinking-note.ts +13 -0
  15. package/examples/extensions/tools.ts +145 -0
  16. package/examples/extensions/with-deps/index.ts +36 -0
  17. package/examples/extensions/with-deps/package-lock.json +31 -0
  18. package/examples/extensions/with-deps/package.json +17 -0
  19. package/examples/hooks/README.md +56 -0
  20. package/examples/hooks/auto-commit-on-exit.ts +48 -0
  21. package/examples/hooks/confirm-destructive.ts +58 -0
  22. package/examples/hooks/custom-compaction.ts +115 -0
  23. package/examples/hooks/dirty-repo-guard.ts +51 -0
  24. package/examples/hooks/file-trigger.ts +40 -0
  25. package/examples/hooks/git-checkpoint.ts +52 -0
  26. package/examples/hooks/handoff.ts +149 -0
  27. package/examples/hooks/permission-gate.ts +33 -0
  28. package/examples/hooks/protected-paths.ts +29 -0
  29. package/examples/hooks/qna.ts +118 -0
  30. package/examples/hooks/status-line.ts +39 -0
  31. package/examples/sdk/01-minimal.ts +21 -0
  32. package/examples/sdk/02-custom-model.ts +49 -0
  33. package/examples/sdk/03-custom-prompt.ts +46 -0
  34. package/examples/sdk/04-skills.ts +43 -0
  35. package/examples/sdk/06-extensions.ts +82 -0
  36. package/examples/sdk/06-hooks.ts +61 -0
  37. package/examples/sdk/07-context-files.ts +35 -0
  38. package/examples/sdk/08-prompt-templates.ts +41 -0
  39. package/examples/sdk/08-slash-commands.ts +46 -0
  40. package/examples/sdk/09-api-keys-and-oauth.ts +54 -0
  41. package/examples/sdk/11-sessions.ts +47 -0
  42. package/examples/sdk/12-redis-sessions.ts +54 -0
  43. package/examples/sdk/13-sql-sessions.ts +61 -0
  44. package/examples/sdk/README.md +172 -0
  45. package/package.json +573 -0
  46. package/scripts/bench-guard.ts +71 -0
  47. package/scripts/build-binary.ts +108 -0
  48. package/scripts/bundle-dist.ts +110 -0
  49. package/scripts/embed-mupdf-wasm.ts +67 -0
  50. package/scripts/format-prompts.ts +68 -0
  51. package/scripts/generate-docs-index.ts +56 -0
  52. package/scripts/generate-share-viewer.ts +34 -0
  53. package/scripts/measure-prompt-tokens.ts +63 -0
  54. package/scripts/omp +42 -0
  55. package/scripts/omp.ts +19 -0
  56. package/src/advisor/__tests__/advisor.test.ts +915 -0
  57. package/src/advisor/advise-tool.ts +165 -0
  58. package/src/advisor/index.ts +4 -0
  59. package/src/advisor/runtime.ts +270 -0
  60. package/src/advisor/transcript-recorder.ts +136 -0
  61. package/src/advisor/watchdog.ts +83 -0
  62. package/src/async/index.ts +1 -0
  63. package/src/async/job-manager.ts +674 -0
  64. package/src/auto-thinking/classifier.ts +190 -0
  65. package/src/autolearn/controller.ts +139 -0
  66. package/src/autolearn/managed-skills.ts +255 -0
  67. package/src/autoresearch/command-resume.md +14 -0
  68. package/src/autoresearch/dashboard.ts +436 -0
  69. package/src/autoresearch/git.ts +319 -0
  70. package/src/autoresearch/helpers.ts +218 -0
  71. package/src/autoresearch/index.ts +536 -0
  72. package/src/autoresearch/prompt-setup.md +43 -0
  73. package/src/autoresearch/prompt.md +103 -0
  74. package/src/autoresearch/resume-message.md +10 -0
  75. package/src/autoresearch/state.ts +273 -0
  76. package/src/autoresearch/storage.ts +700 -0
  77. package/src/autoresearch/tools/init-experiment.ts +269 -0
  78. package/src/autoresearch/tools/log-experiment.ts +521 -0
  79. package/src/autoresearch/tools/run-experiment.ts +407 -0
  80. package/src/autoresearch/tools/update-notes.ts +109 -0
  81. package/src/autoresearch/types.ts +168 -0
  82. package/src/capability/context-file.ts +44 -0
  83. package/src/capability/extension-module.ts +34 -0
  84. package/src/capability/extension.ts +47 -0
  85. package/src/capability/fs.ts +117 -0
  86. package/src/capability/hook.ts +40 -0
  87. package/src/capability/index.ts +436 -0
  88. package/src/capability/instruction.ts +37 -0
  89. package/src/capability/mcp.ts +76 -0
  90. package/src/capability/prompt.ts +35 -0
  91. package/src/capability/rule-buckets.ts +66 -0
  92. package/src/capability/rule.ts +261 -0
  93. package/src/capability/settings.ts +34 -0
  94. package/src/capability/skill.ts +63 -0
  95. package/src/capability/slash-command.ts +40 -0
  96. package/src/capability/ssh.ts +41 -0
  97. package/src/capability/system-prompt.ts +34 -0
  98. package/src/capability/tool.ts +38 -0
  99. package/src/capability/types.ts +168 -0
  100. package/src/cli/agents-cli.ts +138 -0
  101. package/src/cli/args.ts +361 -0
  102. package/src/cli/auth-broker-cli.ts +893 -0
  103. package/src/cli/auth-gateway-cli.ts +608 -0
  104. package/src/cli/bench-cli.ts +552 -0
  105. package/src/cli/classify-install-target.ts +76 -0
  106. package/src/cli/claude-trace-cli.ts +795 -0
  107. package/src/cli/commands/init-xdg.ts +27 -0
  108. package/src/cli/completion-gen.ts +550 -0
  109. package/src/cli/config-cli.ts +418 -0
  110. package/src/cli/dry-balance-cli.ts +858 -0
  111. package/src/cli/extension-flags.ts +48 -0
  112. package/src/cli/file-processor.ts +133 -0
  113. package/src/cli/flag-tables.ts +280 -0
  114. package/src/cli/gallery-cli.ts +231 -0
  115. package/src/cli/gallery-fixtures/agentic.ts +407 -0
  116. package/src/cli/gallery-fixtures/codeintel.ts +187 -0
  117. package/src/cli/gallery-fixtures/edit.ts +194 -0
  118. package/src/cli/gallery-fixtures/fs.ts +220 -0
  119. package/src/cli/gallery-fixtures/index.ts +40 -0
  120. package/src/cli/gallery-fixtures/interaction.ts +49 -0
  121. package/src/cli/gallery-fixtures/memory.ts +81 -0
  122. package/src/cli/gallery-fixtures/misc.ts +250 -0
  123. package/src/cli/gallery-fixtures/search.ts +213 -0
  124. package/src/cli/gallery-fixtures/shell.ts +167 -0
  125. package/src/cli/gallery-fixtures/types.ts +57 -0
  126. package/src/cli/gallery-fixtures/web.ts +158 -0
  127. package/src/cli/gallery-screenshot.ts +279 -0
  128. package/src/cli/grep-cli.ts +160 -0
  129. package/src/cli/grievances-cli.ts +256 -0
  130. package/src/cli/initial-message.ts +58 -0
  131. package/src/cli/models-cli.ts +427 -0
  132. package/src/cli/plugin-cli.ts +996 -0
  133. package/src/cli/profile-alias.ts +338 -0
  134. package/src/cli/profile-bootstrap.ts +243 -0
  135. package/src/cli/read-cli.ts +57 -0
  136. package/src/cli/session-picker.ts +80 -0
  137. package/src/cli/setup-cli.ts +332 -0
  138. package/src/cli/setup-model-picker.ts +43 -0
  139. package/src/cli/shell-cli.ts +176 -0
  140. package/src/cli/ssh-cli.ts +179 -0
  141. package/src/cli/startup-cwd.ts +58 -0
  142. package/src/cli/stats-cli.ts +229 -0
  143. package/src/cli/tiny-models-cli.ts +127 -0
  144. package/src/cli/ttsr-cli.ts +995 -0
  145. package/src/cli/update-cli.ts +671 -0
  146. package/src/cli/usage-cli.ts +774 -0
  147. package/src/cli/web-search-cli.ts +132 -0
  148. package/src/cli/worktree-cli.ts +291 -0
  149. package/src/cli-commands.ts +85 -0
  150. package/src/cli.ts +326 -0
  151. package/src/collab/crypto.ts +63 -0
  152. package/src/collab/guest.ts +450 -0
  153. package/src/collab/host.ts +577 -0
  154. package/src/collab/protocol.ts +274 -0
  155. package/src/collab/relay-client.ts +216 -0
  156. package/src/commands/acp.ts +24 -0
  157. package/src/commands/agents.ts +57 -0
  158. package/src/commands/auth-broker.ts +99 -0
  159. package/src/commands/auth-gateway.ts +69 -0
  160. package/src/commands/bench.ts +42 -0
  161. package/src/commands/commit.ts +46 -0
  162. package/src/commands/complete.ts +66 -0
  163. package/src/commands/completions.ts +60 -0
  164. package/src/commands/config.ts +51 -0
  165. package/src/commands/dry-balance.ts +43 -0
  166. package/src/commands/gallery.ts +52 -0
  167. package/src/commands/grep.ts +48 -0
  168. package/src/commands/grievances.ts +51 -0
  169. package/src/commands/install.ts +107 -0
  170. package/src/commands/join.ts +39 -0
  171. package/src/commands/launch.ts +182 -0
  172. package/src/commands/models.ts +61 -0
  173. package/src/commands/plugin.ts +78 -0
  174. package/src/commands/read.ts +38 -0
  175. package/src/commands/say.ts +102 -0
  176. package/src/commands/setup.ts +67 -0
  177. package/src/commands/shell.ts +29 -0
  178. package/src/commands/ssh.ts +60 -0
  179. package/src/commands/stats.ts +29 -0
  180. package/src/commands/tiny-models.ts +36 -0
  181. package/src/commands/token.ts +108 -0
  182. package/src/commands/ttsr.ts +125 -0
  183. package/src/commands/update.ts +21 -0
  184. package/src/commands/usage.ts +43 -0
  185. package/src/commands/web-search.ts +42 -0
  186. package/src/commands/worktree.ts +56 -0
  187. package/src/commit/agentic/agent.ts +318 -0
  188. package/src/commit/agentic/fallback.ts +96 -0
  189. package/src/commit/agentic/index.ts +355 -0
  190. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  191. package/src/commit/agentic/prompts/session-user.md +25 -0
  192. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  193. package/src/commit/agentic/prompts/system.md +38 -0
  194. package/src/commit/agentic/state.ts +60 -0
  195. package/src/commit/agentic/tools/analyze-file.ts +149 -0
  196. package/src/commit/agentic/tools/git-file-diff.ts +191 -0
  197. package/src/commit/agentic/tools/git-hunk.ts +52 -0
  198. package/src/commit/agentic/tools/git-overview.ts +81 -0
  199. package/src/commit/agentic/tools/index.ts +54 -0
  200. package/src/commit/agentic/tools/propose-changelog.ts +147 -0
  201. package/src/commit/agentic/tools/propose-commit.ts +109 -0
  202. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  203. package/src/commit/agentic/tools/schemas.ts +11 -0
  204. package/src/commit/agentic/tools/split-commit.ts +241 -0
  205. package/src/commit/agentic/topo-sort.ts +44 -0
  206. package/src/commit/agentic/trivial.ts +51 -0
  207. package/src/commit/agentic/validation.ts +183 -0
  208. package/src/commit/analysis/conventional.ts +64 -0
  209. package/src/commit/analysis/index.ts +4 -0
  210. package/src/commit/analysis/scope.ts +242 -0
  211. package/src/commit/analysis/summary.ts +107 -0
  212. package/src/commit/analysis/validation.ts +66 -0
  213. package/src/commit/changelog/detect.ts +40 -0
  214. package/src/commit/changelog/generate.ts +101 -0
  215. package/src/commit/changelog/index.ts +234 -0
  216. package/src/commit/changelog/parse.ts +44 -0
  217. package/src/commit/cli.ts +85 -0
  218. package/src/commit/git/diff.ts +148 -0
  219. package/src/commit/index.ts +5 -0
  220. package/src/commit/map-reduce/index.ts +69 -0
  221. package/src/commit/map-reduce/map-phase.ts +193 -0
  222. package/src/commit/map-reduce/reduce-phase.ts +49 -0
  223. package/src/commit/map-reduce/utils.ts +9 -0
  224. package/src/commit/message.ts +11 -0
  225. package/src/commit/model-selection.ts +89 -0
  226. package/src/commit/pipeline.ts +243 -0
  227. package/src/commit/prompts/analysis-system.md +148 -0
  228. package/src/commit/prompts/analysis-user.md +38 -0
  229. package/src/commit/prompts/changelog-system.md +50 -0
  230. package/src/commit/prompts/changelog-user.md +18 -0
  231. package/src/commit/prompts/file-observer-system.md +24 -0
  232. package/src/commit/prompts/file-observer-user.md +8 -0
  233. package/src/commit/prompts/reduce-system.md +50 -0
  234. package/src/commit/prompts/reduce-user.md +17 -0
  235. package/src/commit/prompts/summary-retry.md +3 -0
  236. package/src/commit/prompts/summary-system.md +38 -0
  237. package/src/commit/prompts/summary-user.md +13 -0
  238. package/src/commit/prompts/types-description.md +2 -0
  239. package/src/commit/shared-llm.ts +70 -0
  240. package/src/commit/types.ts +118 -0
  241. package/src/commit/utils/exclusions.ts +42 -0
  242. package/src/commit/utils.ts +58 -0
  243. package/src/config/api-key-resolver.ts +67 -0
  244. package/src/config/append-only-context-mode.ts +76 -0
  245. package/src/config/config-file.ts +315 -0
  246. package/src/config/file-lock.ts +164 -0
  247. package/src/config/keybindings.ts +634 -0
  248. package/src/config/mcp-schema.json +238 -0
  249. package/src/config/model-discovery.ts +589 -0
  250. package/src/config/model-registry.ts +2260 -0
  251. package/src/config/model-resolver.ts +1819 -0
  252. package/src/config/model-roles.ts +99 -0
  253. package/src/config/models-config-schema.ts +266 -0
  254. package/src/config/models-config.ts +131 -0
  255. package/src/config/prompt-templates.ts +185 -0
  256. package/src/config/resolve-config-value.ts +94 -0
  257. package/src/config/settings-schema.ts +4740 -0
  258. package/src/config/settings.ts +1243 -0
  259. package/src/config.ts +242 -0
  260. package/src/cursor.ts +340 -0
  261. package/src/dap/client.ts +760 -0
  262. package/src/dap/config.ts +189 -0
  263. package/src/dap/defaults.json +212 -0
  264. package/src/dap/index.ts +4 -0
  265. package/src/dap/session.ts +1441 -0
  266. package/src/dap/types.ts +610 -0
  267. package/src/debug/index.ts +559 -0
  268. package/src/debug/log-formatting.ts +58 -0
  269. package/src/debug/log-viewer.ts +908 -0
  270. package/src/debug/profiler.ts +162 -0
  271. package/src/debug/protocol-probe.ts +267 -0
  272. package/src/debug/raw-sse-buffer.ts +294 -0
  273. package/src/debug/raw-sse.ts +292 -0
  274. package/src/debug/remote-debugger.ts +151 -0
  275. package/src/debug/report-bundle.ts +375 -0
  276. package/src/debug/system-info.ts +111 -0
  277. package/src/debug/terminal-info.ts +124 -0
  278. package/src/discovery/agents-md.ts +67 -0
  279. package/src/discovery/agents.ts +230 -0
  280. package/src/discovery/at-imports.ts +273 -0
  281. package/src/discovery/builtin-defaults.ts +39 -0
  282. package/src/discovery/builtin-rules/index.ts +63 -0
  283. package/src/discovery/builtin-rules/low-end/no-hallucinated-apis.md +14 -0
  284. package/src/discovery/builtin-rules/low-end/no-hallucinated-paths.md +14 -0
  285. package/src/discovery/builtin-rules/low-end/no-premature-completion.md +14 -0
  286. package/src/discovery/builtin-rules/rs-box-leak.md +48 -0
  287. package/src/discovery/builtin-rules/rs-future-prelude.md +23 -0
  288. package/src/discovery/builtin-rules/rs-lazylock.md +51 -0
  289. package/src/discovery/builtin-rules/rs-match-ergonomics.md +67 -0
  290. package/src/discovery/builtin-rules/rs-parking-lot.md +44 -0
  291. package/src/discovery/builtin-rules/rs-result-type.md +19 -0
  292. package/src/discovery/builtin-rules/ts-bare-catch.md +38 -0
  293. package/src/discovery/builtin-rules/ts-import-type.md +42 -0
  294. package/src/discovery/builtin-rules/ts-no-any.md +65 -0
  295. package/src/discovery/builtin-rules/ts-no-deprecated-leftovers.md +44 -0
  296. package/src/discovery/builtin-rules/ts-no-dynamic-import.md +39 -0
  297. package/src/discovery/builtin-rules/ts-no-inline-cast-access.md +55 -0
  298. package/src/discovery/builtin-rules/ts-no-return-type.md +44 -0
  299. package/src/discovery/builtin-rules/ts-no-test-timers.md +55 -0
  300. package/src/discovery/builtin-rules/ts-no-tiny-functions.md +51 -0
  301. package/src/discovery/builtin-rules/ts-promise-with-resolvers.md +65 -0
  302. package/src/discovery/builtin-rules/ts-redundant-clear-guard.md +75 -0
  303. package/src/discovery/builtin-rules/ts-set-map.md +28 -0
  304. package/src/discovery/builtin.ts +934 -0
  305. package/src/discovery/claude-plugins.ts +386 -0
  306. package/src/discovery/claude.ts +584 -0
  307. package/src/discovery/cline.ts +83 -0
  308. package/src/discovery/codex.ts +522 -0
  309. package/src/discovery/cursor.ts +220 -0
  310. package/src/discovery/gemini.ts +383 -0
  311. package/src/discovery/github.ts +337 -0
  312. package/src/discovery/helpers.ts +1092 -0
  313. package/src/discovery/index.ts +81 -0
  314. package/src/discovery/mcp-json.ts +172 -0
  315. package/src/discovery/omp-extension-roots.ts +190 -0
  316. package/src/discovery/omp-plugins.ts +383 -0
  317. package/src/discovery/opencode.ts +398 -0
  318. package/src/discovery/plugin-dir-roots.ts +28 -0
  319. package/src/discovery/ssh.ts +153 -0
  320. package/src/discovery/substitute-plugin-root.ts +29 -0
  321. package/src/discovery/vscode.ts +105 -0
  322. package/src/discovery/windsurf.ts +147 -0
  323. package/src/edit/apply-patch/index.ts +87 -0
  324. package/src/edit/apply-patch/parser.ts +174 -0
  325. package/src/edit/diff.ts +999 -0
  326. package/src/edit/file-snapshot-store.ts +143 -0
  327. package/src/edit/hashline/block-resolver.ts +33 -0
  328. package/src/edit/hashline/diff.ts +290 -0
  329. package/src/edit/hashline/execute.ts +237 -0
  330. package/src/edit/hashline/filesystem.ts +130 -0
  331. package/src/edit/hashline/index.ts +5 -0
  332. package/src/edit/hashline/noop-loop-guard.ts +99 -0
  333. package/src/edit/hashline/params.ts +19 -0
  334. package/src/edit/index.ts +620 -0
  335. package/src/edit/modes/apply-patch.lark +19 -0
  336. package/src/edit/modes/apply-patch.ts +53 -0
  337. package/src/edit/modes/patch.ts +1888 -0
  338. package/src/edit/modes/replace.ts +1133 -0
  339. package/src/edit/normalize.ts +345 -0
  340. package/src/edit/notebook.ts +242 -0
  341. package/src/edit/read-file.ts +25 -0
  342. package/src/edit/renderer.ts +823 -0
  343. package/src/edit/streaming.ts +517 -0
  344. package/src/eval/__tests__/agent-bridge.test.ts +769 -0
  345. package/src/eval/__tests__/bridge-timeout.test.ts +64 -0
  346. package/src/eval/__tests__/budget-bridge.test.ts +69 -0
  347. package/src/eval/__tests__/completion-bridge.test.ts +412 -0
  348. package/src/eval/__tests__/helpers-local-roots.test.ts +58 -0
  349. package/src/eval/__tests__/idle-timeout.test.ts +80 -0
  350. package/src/eval/__tests__/js-context-manager.test.ts +291 -0
  351. package/src/eval/__tests__/kernel-spawn.test.ts +103 -0
  352. package/src/eval/__tests__/prelude-agent.test.ts +73 -0
  353. package/src/eval/agent-bridge.ts +319 -0
  354. package/src/eval/backend.ts +71 -0
  355. package/src/eval/bridge-timeout.ts +44 -0
  356. package/src/eval/budget-bridge.ts +48 -0
  357. package/src/eval/completion-bridge.ts +211 -0
  358. package/src/eval/concurrency-bridge.ts +34 -0
  359. package/src/eval/idle-timeout.ts +91 -0
  360. package/src/eval/index.ts +4 -0
  361. package/src/eval/js/context-manager.ts +621 -0
  362. package/src/eval/js/executor.ts +173 -0
  363. package/src/eval/js/index.ts +51 -0
  364. package/src/eval/js/shared/helpers.ts +283 -0
  365. package/src/eval/js/shared/indirect-eval.ts +30 -0
  366. package/src/eval/js/shared/local-module-loader.ts +342 -0
  367. package/src/eval/js/shared/prelude.ts +2 -0
  368. package/src/eval/js/shared/prelude.txt +307 -0
  369. package/src/eval/js/shared/rewrite-imports.ts +532 -0
  370. package/src/eval/js/shared/runtime.ts +580 -0
  371. package/src/eval/js/shared/types.ts +18 -0
  372. package/src/eval/js/tool-bridge.ts +163 -0
  373. package/src/eval/js/worker-core.ts +151 -0
  374. package/src/eval/js/worker-entry.ts +37 -0
  375. package/src/eval/js/worker-protocol.ts +47 -0
  376. package/src/eval/py/__tests__/prelude.test.ts +19 -0
  377. package/src/eval/py/display.ts +71 -0
  378. package/src/eval/py/executor.ts +742 -0
  379. package/src/eval/py/index.ts +68 -0
  380. package/src/eval/py/kernel.ts +748 -0
  381. package/src/eval/py/prelude.py +683 -0
  382. package/src/eval/py/prelude.ts +3 -0
  383. package/src/eval/py/runner.py +1177 -0
  384. package/src/eval/py/runtime.ts +276 -0
  385. package/src/eval/py/spawn-options.ts +126 -0
  386. package/src/eval/py/tool-bridge.ts +182 -0
  387. package/src/eval/session-id.ts +8 -0
  388. package/src/eval/types.ts +48 -0
  389. package/src/exa/index.ts +2 -0
  390. package/src/exa/mcp-client.ts +370 -0
  391. package/src/exa/types.ts +69 -0
  392. package/src/exec/bash-executor.ts +434 -0
  393. package/src/exec/exec.ts +53 -0
  394. package/src/exec/non-interactive-env.ts +119 -0
  395. package/src/export/custom-share.ts +65 -0
  396. package/src/export/html/index.ts +266 -0
  397. package/src/export/html/share-loader.js +102 -0
  398. package/src/export/html/template.css +1337 -0
  399. package/src/export/html/template.html +49 -0
  400. package/src/export/html/template.js +1626 -0
  401. package/src/export/html/tool-views.generated.js +37 -0
  402. package/src/export/html/vendor/highlight.min.js +1213 -0
  403. package/src/export/html/vendor/marked.min.js +6 -0
  404. package/src/export/share.ts +268 -0
  405. package/src/export/ttsr.ts +583 -0
  406. package/src/extensibility/custom-commands/bundled/ci-green/index.ts +54 -0
  407. package/src/extensibility/custom-commands/bundled/review/index.ts +698 -0
  408. package/src/extensibility/custom-commands/index.ts +2 -0
  409. package/src/extensibility/custom-commands/loader.ts +242 -0
  410. package/src/extensibility/custom-commands/types.ts +119 -0
  411. package/src/extensibility/custom-tools/index.ts +7 -0
  412. package/src/extensibility/custom-tools/loader.ts +268 -0
  413. package/src/extensibility/custom-tools/types.ts +277 -0
  414. package/src/extensibility/custom-tools/wrapper.ts +47 -0
  415. package/src/extensibility/extensions/compact-handler.ts +40 -0
  416. package/src/extensibility/extensions/get-commands-handler.ts +78 -0
  417. package/src/extensibility/extensions/index.ts +16 -0
  418. package/src/extensibility/extensions/loader.ts +587 -0
  419. package/src/extensibility/extensions/model-api.ts +41 -0
  420. package/src/extensibility/extensions/runner.ts +989 -0
  421. package/src/extensibility/extensions/types.ts +1394 -0
  422. package/src/extensibility/extensions/wrapper.ts +259 -0
  423. package/src/extensibility/hooks/index.ts +6 -0
  424. package/src/extensibility/hooks/loader.ts +262 -0
  425. package/src/extensibility/hooks/runner.ts +425 -0
  426. package/src/extensibility/hooks/tool-wrapper.ts +107 -0
  427. package/src/extensibility/hooks/types.ts +613 -0
  428. package/src/extensibility/legacy-pi-ai-shim.ts +61 -0
  429. package/src/extensibility/legacy-pi-coding-agent-shim.ts +128 -0
  430. package/src/extensibility/plugins/doctor.ts +65 -0
  431. package/src/extensibility/plugins/git-url.ts +367 -0
  432. package/src/extensibility/plugins/index.ts +9 -0
  433. package/src/extensibility/plugins/installer.ts +192 -0
  434. package/src/extensibility/plugins/legacy-pi-compat.ts +712 -0
  435. package/src/extensibility/plugins/loader.ts +458 -0
  436. package/src/extensibility/plugins/manager.ts +1026 -0
  437. package/src/extensibility/plugins/marketplace/cache.ts +136 -0
  438. package/src/extensibility/plugins/marketplace/fetcher.ts +315 -0
  439. package/src/extensibility/plugins/marketplace/index.ts +6 -0
  440. package/src/extensibility/plugins/marketplace/manager.ts +770 -0
  441. package/src/extensibility/plugins/marketplace/registry.ts +196 -0
  442. package/src/extensibility/plugins/marketplace/source-resolver.ts +147 -0
  443. package/src/extensibility/plugins/marketplace/types.ts +191 -0
  444. package/src/extensibility/plugins/marketplace-auto-update.ts +49 -0
  445. package/src/extensibility/plugins/parser.ts +105 -0
  446. package/src/extensibility/plugins/runtime-config.ts +9 -0
  447. package/src/extensibility/plugins/types.ts +194 -0
  448. package/src/extensibility/shared-events.ts +367 -0
  449. package/src/extensibility/skills.ts +408 -0
  450. package/src/extensibility/slash-commands.ts +131 -0
  451. package/src/extensibility/tool-proxy.ts +28 -0
  452. package/src/extensibility/typebox.ts +945 -0
  453. package/src/extensibility/utils.ts +44 -0
  454. package/src/goals/guided-setup.ts +142 -0
  455. package/src/goals/index.ts +3 -0
  456. package/src/goals/runtime.ts +521 -0
  457. package/src/goals/state.ts +37 -0
  458. package/src/goals/tools/goal-tool.ts +251 -0
  459. package/src/hindsight/backend.ts +354 -0
  460. package/src/hindsight/bank.ts +156 -0
  461. package/src/hindsight/client.ts +623 -0
  462. package/src/hindsight/config.ts +175 -0
  463. package/src/hindsight/content.ts +210 -0
  464. package/src/hindsight/index.ts +8 -0
  465. package/src/hindsight/mental-models.ts +429 -0
  466. package/src/hindsight/seeds.json +32 -0
  467. package/src/hindsight/state.ts +492 -0
  468. package/src/hindsight/transcript.ts +71 -0
  469. package/src/index.ts +66 -0
  470. package/src/internal-urls/agent-protocol.ts +146 -0
  471. package/src/internal-urls/artifact-protocol.ts +107 -0
  472. package/src/internal-urls/docs-index.generated.txt +2 -0
  473. package/src/internal-urls/docs-index.ts +102 -0
  474. package/src/internal-urls/history-protocol.ts +118 -0
  475. package/src/internal-urls/index.ts +25 -0
  476. package/src/internal-urls/issue-pr-protocol.ts +594 -0
  477. package/src/internal-urls/json-query.ts +126 -0
  478. package/src/internal-urls/local-protocol.ts +309 -0
  479. package/src/internal-urls/mcp-protocol.ts +151 -0
  480. package/src/internal-urls/memory-protocol.ts +169 -0
  481. package/src/internal-urls/omp-protocol.ts +94 -0
  482. package/src/internal-urls/parse.ts +72 -0
  483. package/src/internal-urls/registry-helpers.ts +25 -0
  484. package/src/internal-urls/router.ts +105 -0
  485. package/src/internal-urls/rule-protocol.ts +45 -0
  486. package/src/internal-urls/skill-protocol.ts +96 -0
  487. package/src/internal-urls/types.ts +152 -0
  488. package/src/internal-urls/vault-protocol.ts +936 -0
  489. package/src/irc/bus.ts +311 -0
  490. package/src/lib/xai-http.ts +124 -0
  491. package/src/lsp/client.ts +1217 -0
  492. package/src/lsp/clients/biome-client.ts +264 -0
  493. package/src/lsp/clients/index.ts +50 -0
  494. package/src/lsp/clients/lsp-linter-client.ts +85 -0
  495. package/src/lsp/clients/swiftlint-client.ts +120 -0
  496. package/src/lsp/config.ts +502 -0
  497. package/src/lsp/defaults.json +499 -0
  498. package/src/lsp/diagnostics-ledger.ts +51 -0
  499. package/src/lsp/edits.ts +267 -0
  500. package/src/lsp/format-options.ts +119 -0
  501. package/src/lsp/index.ts +2480 -0
  502. package/src/lsp/lspmux.ts +233 -0
  503. package/src/lsp/render.ts +668 -0
  504. package/src/lsp/startup-events.ts +13 -0
  505. package/src/lsp/types.ts +444 -0
  506. package/src/lsp/utils.ts +718 -0
  507. package/src/main.ts +1421 -0
  508. package/src/markit/NOTICE +32 -0
  509. package/src/markit/converters/docx.ts +56 -0
  510. package/src/markit/converters/epub.ts +136 -0
  511. package/src/markit/converters/mammoth.d.ts +24 -0
  512. package/src/markit/converters/pdf/columns.ts +103 -0
  513. package/src/markit/converters/pdf/extract.ts +574 -0
  514. package/src/markit/converters/pdf/grid.ts +780 -0
  515. package/src/markit/converters/pdf/headers.ts +106 -0
  516. package/src/markit/converters/pdf/index.ts +146 -0
  517. package/src/markit/converters/pdf/render.ts +501 -0
  518. package/src/markit/converters/pdf/types.ts +84 -0
  519. package/src/markit/converters/pptx.ts +325 -0
  520. package/src/markit/converters/xlsx.ts +173 -0
  521. package/src/markit/index.ts +2 -0
  522. package/src/markit/registry.ts +59 -0
  523. package/src/markit/types.ts +35 -0
  524. package/src/mcp/client.ts +509 -0
  525. package/src/mcp/config-writer.ts +229 -0
  526. package/src/mcp/config.ts +365 -0
  527. package/src/mcp/index.ts +29 -0
  528. package/src/mcp/json-rpc.ts +122 -0
  529. package/src/mcp/loader.ts +124 -0
  530. package/src/mcp/manager.ts +1326 -0
  531. package/src/mcp/oauth-credentials.ts +104 -0
  532. package/src/mcp/oauth-discovery.ts +467 -0
  533. package/src/mcp/oauth-flow.ts +555 -0
  534. package/src/mcp/render.ts +155 -0
  535. package/src/mcp/smithery-auth.ts +104 -0
  536. package/src/mcp/smithery-connect.ts +145 -0
  537. package/src/mcp/smithery-registry.ts +477 -0
  538. package/src/mcp/startup-events.ts +21 -0
  539. package/src/mcp/timeout.ts +59 -0
  540. package/src/mcp/tool-bridge.ts +429 -0
  541. package/src/mcp/tool-cache.ts +117 -0
  542. package/src/mcp/transports/http.ts +519 -0
  543. package/src/mcp/transports/index.ts +6 -0
  544. package/src/mcp/transports/stdio.ts +606 -0
  545. package/src/mcp/types.ts +427 -0
  546. package/src/memories/index.ts +1281 -0
  547. package/src/memories/storage.ts +578 -0
  548. package/src/memory-backend/index.ts +18 -0
  549. package/src/memory-backend/local-backend.ts +45 -0
  550. package/src/memory-backend/off-backend.ts +25 -0
  551. package/src/memory-backend/resolve.ts +25 -0
  552. package/src/memory-backend/runtime.ts +66 -0
  553. package/src/memory-backend/types.ts +166 -0
  554. package/src/mnemopi/backend.ts +612 -0
  555. package/src/mnemopi/config.ts +265 -0
  556. package/src/mnemopi/embed-client.ts +401 -0
  557. package/src/mnemopi/embed-protocol.ts +35 -0
  558. package/src/mnemopi/embed-worker.ts +113 -0
  559. package/src/mnemopi/index.ts +3 -0
  560. package/src/mnemopi/state.ts +657 -0
  561. package/src/modes/acp/acp-agent.ts +2362 -0
  562. package/src/modes/acp/acp-client-bridge.ts +154 -0
  563. package/src/modes/acp/acp-event-mapper.ts +933 -0
  564. package/src/modes/acp/acp-mode.ts +23 -0
  565. package/src/modes/acp/index.ts +2 -0
  566. package/src/modes/acp/terminal-auth.ts +37 -0
  567. package/src/modes/components/__tests__/skill-message.test.ts +92 -0
  568. package/src/modes/components/advisor-message.ts +99 -0
  569. package/src/modes/components/agent-dashboard.ts +1206 -0
  570. package/src/modes/components/agent-hub.ts +566 -0
  571. package/src/modes/components/agent-transcript-viewer.ts +461 -0
  572. package/src/modes/components/assistant-message.ts +612 -0
  573. package/src/modes/components/background-tan-message.ts +36 -0
  574. package/src/modes/components/bash-execution.ts +220 -0
  575. package/src/modes/components/bordered-loader.ts +41 -0
  576. package/src/modes/components/btw-panel.ts +112 -0
  577. package/src/modes/components/cache-invalidation-marker.ts +110 -0
  578. package/src/modes/components/chat-block.ts +111 -0
  579. package/src/modes/components/chat-transcript-builder.ts +476 -0
  580. package/src/modes/components/collab-prompt-message.ts +32 -0
  581. package/src/modes/components/compaction-summary-message.ts +215 -0
  582. package/src/modes/components/copy-selector.ts +206 -0
  583. package/src/modes/components/countdown-timer.ts +75 -0
  584. package/src/modes/components/custom-editor.test.ts +142 -0
  585. package/src/modes/components/custom-editor.ts +620 -0
  586. package/src/modes/components/custom-message.ts +67 -0
  587. package/src/modes/components/diff.ts +254 -0
  588. package/src/modes/components/dynamic-border.ts +34 -0
  589. package/src/modes/components/error-banner.ts +33 -0
  590. package/src/modes/components/eval-execution.ts +158 -0
  591. package/src/modes/components/execution-shared.ts +101 -0
  592. package/src/modes/components/extensions/extension-dashboard.ts +399 -0
  593. package/src/modes/components/extensions/extension-list.ts +502 -0
  594. package/src/modes/components/extensions/index.ts +9 -0
  595. package/src/modes/components/extensions/inspector-panel.ts +321 -0
  596. package/src/modes/components/extensions/state-manager.ts +627 -0
  597. package/src/modes/components/extensions/types.ts +186 -0
  598. package/src/modes/components/footer.ts +275 -0
  599. package/src/modes/components/history-search.ts +280 -0
  600. package/src/modes/components/hook-editor.ts +167 -0
  601. package/src/modes/components/hook-input.ts +87 -0
  602. package/src/modes/components/hook-message.ts +67 -0
  603. package/src/modes/components/hook-selector.ts +659 -0
  604. package/src/modes/components/index.ts +38 -0
  605. package/src/modes/components/keybinding-hints.ts +65 -0
  606. package/src/modes/components/late-diagnostics-message.ts +60 -0
  607. package/src/modes/components/login-dialog.ts +164 -0
  608. package/src/modes/components/logout-account-selector.ts +130 -0
  609. package/src/modes/components/mcp-add-wizard.ts +1360 -0
  610. package/src/modes/components/message-frame.ts +92 -0
  611. package/src/modes/components/model-selector.ts +1315 -0
  612. package/src/modes/components/oauth-selector.ts +457 -0
  613. package/src/modes/components/omfg-panel.ts +141 -0
  614. package/src/modes/components/overlay-box.ts +109 -0
  615. package/src/modes/components/plan-review-overlay.ts +847 -0
  616. package/src/modes/components/plan-toc.ts +138 -0
  617. package/src/modes/components/plugin-selector.ts +95 -0
  618. package/src/modes/components/plugin-settings.ts +739 -0
  619. package/src/modes/components/queue-mode-selector.ts +56 -0
  620. package/src/modes/components/read-tool-group.ts +676 -0
  621. package/src/modes/components/reset-usage-selector.ts +161 -0
  622. package/src/modes/components/segment-track.ts +89 -0
  623. package/src/modes/components/session-selector.ts +631 -0
  624. package/src/modes/components/settings-defs.ts +225 -0
  625. package/src/modes/components/settings-selector.ts +1095 -0
  626. package/src/modes/components/show-images-selector.ts +45 -0
  627. package/src/modes/components/skill-message.ts +110 -0
  628. package/src/modes/components/snapcompact-shape-preview-doc.md +18 -0
  629. package/src/modes/components/snapcompact-shape-preview.ts +192 -0
  630. package/src/modes/components/status-line/component.ts +1001 -0
  631. package/src/modes/components/status-line/context-thresholds.ts +78 -0
  632. package/src/modes/components/status-line/git-utils.ts +42 -0
  633. package/src/modes/components/status-line/index.ts +5 -0
  634. package/src/modes/components/status-line/presets.ts +106 -0
  635. package/src/modes/components/status-line/segments.ts +616 -0
  636. package/src/modes/components/status-line/separators.ts +55 -0
  637. package/src/modes/components/status-line/token-rate.ts +66 -0
  638. package/src/modes/components/status-line/types.ts +124 -0
  639. package/src/modes/components/theme-selector.ts +63 -0
  640. package/src/modes/components/thinking-selector.ts +52 -0
  641. package/src/modes/components/tiny-title-download-progress.ts +90 -0
  642. package/src/modes/components/tips.txt +24 -0
  643. package/src/modes/components/todo-reminder.ts +39 -0
  644. package/src/modes/components/tool-execution.ts +1165 -0
  645. package/src/modes/components/transcript-container.ts +806 -0
  646. package/src/modes/components/tree-selector.ts +994 -0
  647. package/src/modes/components/ttsr-notification.ts +123 -0
  648. package/src/modes/components/usage-row.ts +18 -0
  649. package/src/modes/components/user-message-selector.ts +227 -0
  650. package/src/modes/components/user-message.ts +68 -0
  651. package/src/modes/components/visual-truncate.ts +63 -0
  652. package/src/modes/components/welcome.ts +581 -0
  653. package/src/modes/controllers/btw-controller.ts +173 -0
  654. package/src/modes/controllers/command-controller-shared.ts +109 -0
  655. package/src/modes/controllers/command-controller.ts +1653 -0
  656. package/src/modes/controllers/event-controller.ts +1153 -0
  657. package/src/modes/controllers/extension-ui-controller.ts +893 -0
  658. package/src/modes/controllers/input-controller.ts +1627 -0
  659. package/src/modes/controllers/mcp-command-controller.ts +2162 -0
  660. package/src/modes/controllers/omfg-controller.ts +283 -0
  661. package/src/modes/controllers/omfg-rule.ts +647 -0
  662. package/src/modes/controllers/selector-controller.ts +1285 -0
  663. package/src/modes/controllers/session-focus-controller.ts +112 -0
  664. package/src/modes/controllers/ssh-command-controller.ts +384 -0
  665. package/src/modes/controllers/streaming-reveal.ts +295 -0
  666. package/src/modes/controllers/tan-command-controller.ts +190 -0
  667. package/src/modes/controllers/todo-command-controller.ts +485 -0
  668. package/src/modes/controllers/tool-args-reveal.ts +174 -0
  669. package/src/modes/data/emojis.json +1 -0
  670. package/src/modes/emoji-autocomplete.ts +285 -0
  671. package/src/modes/gradient-highlight.ts +99 -0
  672. package/src/modes/image-references.ts +137 -0
  673. package/src/modes/index.ts +17 -0
  674. package/src/modes/interactive-mode.ts +3940 -0
  675. package/src/modes/internal-url-autocomplete.ts +143 -0
  676. package/src/modes/loop-limit.ts +192 -0
  677. package/src/modes/magic-keywords.ts +42 -0
  678. package/src/modes/markdown-prose.ts +247 -0
  679. package/src/modes/oauth-manual-input.ts +69 -0
  680. package/src/modes/orchestrate.ts +42 -0
  681. package/src/modes/print-mode.ts +130 -0
  682. package/src/modes/prompt-action-autocomplete.ts +260 -0
  683. package/src/modes/rpc/host-tools.ts +186 -0
  684. package/src/modes/rpc/host-uris.ts +235 -0
  685. package/src/modes/rpc/rpc-client.ts +995 -0
  686. package/src/modes/rpc/rpc-mode.ts +1156 -0
  687. package/src/modes/rpc/rpc-subagents.ts +265 -0
  688. package/src/modes/rpc/rpc-types.ts +487 -0
  689. package/src/modes/runtime-init.ts +142 -0
  690. package/src/modes/session-observer-registry.ts +215 -0
  691. package/src/modes/setup-version.ts +11 -0
  692. package/src/modes/setup-wizard/index.ts +101 -0
  693. package/src/modes/setup-wizard/lazy.ts +16 -0
  694. package/src/modes/setup-wizard/scenes/glyph.ts +114 -0
  695. package/src/modes/setup-wizard/scenes/outro.ts +35 -0
  696. package/src/modes/setup-wizard/scenes/providers.ts +103 -0
  697. package/src/modes/setup-wizard/scenes/sign-in.ts +286 -0
  698. package/src/modes/setup-wizard/scenes/splash.ts +201 -0
  699. package/src/modes/setup-wizard/scenes/theme.ts +326 -0
  700. package/src/modes/setup-wizard/scenes/types.ts +57 -0
  701. package/src/modes/setup-wizard/scenes/web-search.ts +145 -0
  702. package/src/modes/setup-wizard/startup-splash.ts +107 -0
  703. package/src/modes/setup-wizard/wizard-overlay.ts +334 -0
  704. package/src/modes/shared.ts +49 -0
  705. package/src/modes/theme/dark.json +95 -0
  706. package/src/modes/theme/defaults/alabaster.json +93 -0
  707. package/src/modes/theme/defaults/amethyst.json +96 -0
  708. package/src/modes/theme/defaults/anthracite.json +93 -0
  709. package/src/modes/theme/defaults/basalt.json +91 -0
  710. package/src/modes/theme/defaults/birch.json +95 -0
  711. package/src/modes/theme/defaults/dark-abyss.json +91 -0
  712. package/src/modes/theme/defaults/dark-arctic.json +104 -0
  713. package/src/modes/theme/defaults/dark-aurora.json +95 -0
  714. package/src/modes/theme/defaults/dark-catppuccin.json +107 -0
  715. package/src/modes/theme/defaults/dark-cavern.json +91 -0
  716. package/src/modes/theme/defaults/dark-copper.json +95 -0
  717. package/src/modes/theme/defaults/dark-cosmos.json +90 -0
  718. package/src/modes/theme/defaults/dark-cyberpunk.json +102 -0
  719. package/src/modes/theme/defaults/dark-dracula.json +98 -0
  720. package/src/modes/theme/defaults/dark-eclipse.json +91 -0
  721. package/src/modes/theme/defaults/dark-ember.json +95 -0
  722. package/src/modes/theme/defaults/dark-equinox.json +90 -0
  723. package/src/modes/theme/defaults/dark-forest.json +96 -0
  724. package/src/modes/theme/defaults/dark-github.json +105 -0
  725. package/src/modes/theme/defaults/dark-gruvbox.json +112 -0
  726. package/src/modes/theme/defaults/dark-lavender.json +95 -0
  727. package/src/modes/theme/defaults/dark-lunar.json +89 -0
  728. package/src/modes/theme/defaults/dark-midnight.json +95 -0
  729. package/src/modes/theme/defaults/dark-monochrome.json +94 -0
  730. package/src/modes/theme/defaults/dark-monokai.json +98 -0
  731. package/src/modes/theme/defaults/dark-nebula.json +90 -0
  732. package/src/modes/theme/defaults/dark-nord.json +97 -0
  733. package/src/modes/theme/defaults/dark-ocean.json +101 -0
  734. package/src/modes/theme/defaults/dark-one.json +100 -0
  735. package/src/modes/theme/defaults/dark-poimandres.json +142 -0
  736. package/src/modes/theme/defaults/dark-rainforest.json +91 -0
  737. package/src/modes/theme/defaults/dark-reef.json +91 -0
  738. package/src/modes/theme/defaults/dark-retro.json +92 -0
  739. package/src/modes/theme/defaults/dark-rose-pine.json +96 -0
  740. package/src/modes/theme/defaults/dark-sakura.json +95 -0
  741. package/src/modes/theme/defaults/dark-slate.json +95 -0
  742. package/src/modes/theme/defaults/dark-solarized.json +97 -0
  743. package/src/modes/theme/defaults/dark-solstice.json +90 -0
  744. package/src/modes/theme/defaults/dark-starfall.json +91 -0
  745. package/src/modes/theme/defaults/dark-sunset.json +99 -0
  746. package/src/modes/theme/defaults/dark-swamp.json +90 -0
  747. package/src/modes/theme/defaults/dark-synthwave.json +103 -0
  748. package/src/modes/theme/defaults/dark-taiga.json +91 -0
  749. package/src/modes/theme/defaults/dark-terminal.json +95 -0
  750. package/src/modes/theme/defaults/dark-tokyo-night.json +101 -0
  751. package/src/modes/theme/defaults/dark-tundra.json +91 -0
  752. package/src/modes/theme/defaults/dark-twilight.json +91 -0
  753. package/src/modes/theme/defaults/dark-volcanic.json +91 -0
  754. package/src/modes/theme/defaults/graphite.json +92 -0
  755. package/src/modes/theme/defaults/index.ts +199 -0
  756. package/src/modes/theme/defaults/light-arctic.json +107 -0
  757. package/src/modes/theme/defaults/light-aurora-day.json +91 -0
  758. package/src/modes/theme/defaults/light-canyon.json +91 -0
  759. package/src/modes/theme/defaults/light-catppuccin.json +106 -0
  760. package/src/modes/theme/defaults/light-cirrus.json +90 -0
  761. package/src/modes/theme/defaults/light-coral.json +95 -0
  762. package/src/modes/theme/defaults/light-cyberpunk.json +96 -0
  763. package/src/modes/theme/defaults/light-dawn.json +90 -0
  764. package/src/modes/theme/defaults/light-dunes.json +91 -0
  765. package/src/modes/theme/defaults/light-eucalyptus.json +95 -0
  766. package/src/modes/theme/defaults/light-forest.json +100 -0
  767. package/src/modes/theme/defaults/light-frost.json +95 -0
  768. package/src/modes/theme/defaults/light-github.json +115 -0
  769. package/src/modes/theme/defaults/light-glacier.json +91 -0
  770. package/src/modes/theme/defaults/light-gruvbox.json +108 -0
  771. package/src/modes/theme/defaults/light-haze.json +90 -0
  772. package/src/modes/theme/defaults/light-honeycomb.json +95 -0
  773. package/src/modes/theme/defaults/light-lagoon.json +91 -0
  774. package/src/modes/theme/defaults/light-lavender.json +95 -0
  775. package/src/modes/theme/defaults/light-meadow.json +91 -0
  776. package/src/modes/theme/defaults/light-mint.json +95 -0
  777. package/src/modes/theme/defaults/light-monochrome.json +101 -0
  778. package/src/modes/theme/defaults/light-ocean.json +99 -0
  779. package/src/modes/theme/defaults/light-one.json +99 -0
  780. package/src/modes/theme/defaults/light-opal.json +91 -0
  781. package/src/modes/theme/defaults/light-orchard.json +91 -0
  782. package/src/modes/theme/defaults/light-paper.json +95 -0
  783. package/src/modes/theme/defaults/light-poimandres.json +142 -0
  784. package/src/modes/theme/defaults/light-prism.json +90 -0
  785. package/src/modes/theme/defaults/light-retro.json +98 -0
  786. package/src/modes/theme/defaults/light-sand.json +95 -0
  787. package/src/modes/theme/defaults/light-savanna.json +91 -0
  788. package/src/modes/theme/defaults/light-solarized.json +102 -0
  789. package/src/modes/theme/defaults/light-soleil.json +90 -0
  790. package/src/modes/theme/defaults/light-sunset.json +99 -0
  791. package/src/modes/theme/defaults/light-synthwave.json +98 -0
  792. package/src/modes/theme/defaults/light-tokyo-night.json +111 -0
  793. package/src/modes/theme/defaults/light-wetland.json +91 -0
  794. package/src/modes/theme/defaults/light-zenith.json +89 -0
  795. package/src/modes/theme/defaults/limestone.json +94 -0
  796. package/src/modes/theme/defaults/mahogany.json +97 -0
  797. package/src/modes/theme/defaults/marble.json +93 -0
  798. package/src/modes/theme/defaults/obsidian.json +91 -0
  799. package/src/modes/theme/defaults/onyx.json +91 -0
  800. package/src/modes/theme/defaults/pearl.json +93 -0
  801. package/src/modes/theme/defaults/porcelain.json +91 -0
  802. package/src/modes/theme/defaults/quartz.json +96 -0
  803. package/src/modes/theme/defaults/sandstone.json +95 -0
  804. package/src/modes/theme/defaults/titanium.json +90 -0
  805. package/src/modes/theme/light.json +93 -0
  806. package/src/modes/theme/mermaid-cache.ts +92 -0
  807. package/src/modes/theme/shimmer.ts +235 -0
  808. package/src/modes/theme/theme-schema.json +459 -0
  809. package/src/modes/theme/theme.ts +2915 -0
  810. package/src/modes/turn-budget.ts +31 -0
  811. package/src/modes/types.ts +406 -0
  812. package/src/modes/ultrathink.ts +41 -0
  813. package/src/modes/utils/context-usage.ts +432 -0
  814. package/src/modes/utils/copy-targets.ts +360 -0
  815. package/src/modes/utils/hotkeys-markdown.ts +62 -0
  816. package/src/modes/utils/keybinding-matchers.ts +51 -0
  817. package/src/modes/utils/tools-markdown.ts +27 -0
  818. package/src/modes/utils/ui-helpers.ts +886 -0
  819. package/src/modes/workflow.ts +42 -0
  820. package/src/plan-mode/approved-plan.ts +186 -0
  821. package/src/plan-mode/plan-handoff.ts +37 -0
  822. package/src/plan-mode/plan-protection.ts +31 -0
  823. package/src/plan-mode/state.ts +6 -0
  824. package/src/priority.json +45 -0
  825. package/src/prompts/advisor/advise-tool.md +3 -0
  826. package/src/prompts/advisor/system.md +113 -0
  827. package/src/prompts/agents/designer.md +74 -0
  828. package/src/prompts/agents/explore.md +58 -0
  829. package/src/prompts/agents/frontmatter.md +11 -0
  830. package/src/prompts/agents/init.md +33 -0
  831. package/src/prompts/agents/librarian.md +119 -0
  832. package/src/prompts/agents/oracle.md +54 -0
  833. package/src/prompts/agents/plan.md +48 -0
  834. package/src/prompts/agents/reviewer.md +139 -0
  835. package/src/prompts/agents/task.md +17 -0
  836. package/src/prompts/bench.md +12 -0
  837. package/src/prompts/ci-green-request.md +36 -0
  838. package/src/prompts/dry-balance-bench.md +8 -0
  839. package/src/prompts/goals/goal-budget-limit.md +16 -0
  840. package/src/prompts/goals/goal-continuation.md +28 -0
  841. package/src/prompts/goals/goal-mode-active.md +23 -0
  842. package/src/prompts/goals/guided-goal-interview.md +8 -0
  843. package/src/prompts/goals/guided-goal-system.md +12 -0
  844. package/src/prompts/low-end/system.md +47 -0
  845. package/src/prompts/memories/consolidation.md +30 -0
  846. package/src/prompts/memories/consolidation_system.md +4 -0
  847. package/src/prompts/memories/read-path.md +17 -0
  848. package/src/prompts/memories/stage_one_input.md +6 -0
  849. package/src/prompts/memories/stage_one_system.md +21 -0
  850. package/src/prompts/review-custom-request.md +22 -0
  851. package/src/prompts/review-headless-request.md +16 -0
  852. package/src/prompts/review-request.md +69 -0
  853. package/src/prompts/steering/user-interjection.md +9 -0
  854. package/src/prompts/system/agent-creation-architect.md +50 -0
  855. package/src/prompts/system/agent-creation-user.md +6 -0
  856. package/src/prompts/system/auto-continue.md +1 -0
  857. package/src/prompts/system/auto-thinking-difficulty-local.md +14 -0
  858. package/src/prompts/system/auto-thinking-difficulty.md +12 -0
  859. package/src/prompts/system/autolearn-guidance-learn.md +1 -0
  860. package/src/prompts/system/autolearn-guidance.md +7 -0
  861. package/src/prompts/system/autolearn-nudge.md +3 -0
  862. package/src/prompts/system/background-tan-dispatch.md +8 -0
  863. package/src/prompts/system/btw-user.md +8 -0
  864. package/src/prompts/system/commit-message-system.md +14 -0
  865. package/src/prompts/system/custom-system-prompt.md +64 -0
  866. package/src/prompts/system/eager-task.md +7 -0
  867. package/src/prompts/system/eager-todo.md +18 -0
  868. package/src/prompts/system/empty-stop-retry.md +4 -0
  869. package/src/prompts/system/irc-autoreply.md +6 -0
  870. package/src/prompts/system/irc-incoming.md +7 -0
  871. package/src/prompts/system/manual-continue.md +7 -0
  872. package/src/prompts/system/memory-consolidation-system.md +8 -0
  873. package/src/prompts/system/memory-extraction-system.md +26 -0
  874. package/src/prompts/system/omfg-user.md +50 -0
  875. package/src/prompts/system/orchestrate-notice.md +40 -0
  876. package/src/prompts/system/personalities/default.md +18 -0
  877. package/src/prompts/system/personalities/friendly.md +17 -0
  878. package/src/prompts/system/personalities/pragmatic.md +15 -0
  879. package/src/prompts/system/plan-mode-active.md +109 -0
  880. package/src/prompts/system/plan-mode-approved.md +25 -0
  881. package/src/prompts/system/plan-mode-compact-instructions.md +16 -0
  882. package/src/prompts/system/plan-mode-reference.md +11 -0
  883. package/src/prompts/system/plan-mode-subagent.md +33 -0
  884. package/src/prompts/system/plan-mode-tool-decision-reminder.md +9 -0
  885. package/src/prompts/system/project-prompt.md +52 -0
  886. package/src/prompts/system/snapcompact-context-frames-note.md +1 -0
  887. package/src/prompts/system/snapcompact-context-stub.md +1 -0
  888. package/src/prompts/system/snapcompact-system-frames-note.md +1 -0
  889. package/src/prompts/system/snapcompact-system-stub.md +1 -0
  890. package/src/prompts/system/snapcompact-toolresult-note.md +1 -0
  891. package/src/prompts/system/subagent-system-prompt.md +71 -0
  892. package/src/prompts/system/subagent-user-prompt.md +3 -0
  893. package/src/prompts/system/subagent-yield-reminder.md +12 -0
  894. package/src/prompts/system/system-prompt.md +251 -0
  895. package/src/prompts/system/tiny-title-system.md +8 -0
  896. package/src/prompts/system/title-marker-instruction.md +1 -0
  897. package/src/prompts/system/title-system-marker.md +16 -0
  898. package/src/prompts/system/title-system.md +16 -0
  899. package/src/prompts/system/ttsr-interrupt.md +7 -0
  900. package/src/prompts/system/ttsr-tool-reminder.md +5 -0
  901. package/src/prompts/system/ultrathink-notice.md +3 -0
  902. package/src/prompts/system/unexpected-stop-classifier.md +17 -0
  903. package/src/prompts/system/unexpected-stop-retry.md +4 -0
  904. package/src/prompts/system/web-search.md +25 -0
  905. package/src/prompts/system/workflow-notice.md +70 -0
  906. package/src/prompts/tools/apply-patch.md +65 -0
  907. package/src/prompts/tools/ask.md +22 -0
  908. package/src/prompts/tools/ast-edit.md +22 -0
  909. package/src/prompts/tools/ast-grep.md +25 -0
  910. package/src/prompts/tools/async-result.md +8 -0
  911. package/src/prompts/tools/bash.md +45 -0
  912. package/src/prompts/tools/browser.md +42 -0
  913. package/src/prompts/tools/checkpoint.md +15 -0
  914. package/src/prompts/tools/debug.md +17 -0
  915. package/src/prompts/tools/eval.md +70 -0
  916. package/src/prompts/tools/find.md +19 -0
  917. package/src/prompts/tools/github.md +17 -0
  918. package/src/prompts/tools/goal.md +11 -0
  919. package/src/prompts/tools/image-attachment-describe-system.md +8 -0
  920. package/src/prompts/tools/image-attachment-describe.md +10 -0
  921. package/src/prompts/tools/image-gen.md +7 -0
  922. package/src/prompts/tools/inspect-image-system.md +20 -0
  923. package/src/prompts/tools/inspect-image.md +22 -0
  924. package/src/prompts/tools/irc.md +33 -0
  925. package/src/prompts/tools/job.md +17 -0
  926. package/src/prompts/tools/learn.md +7 -0
  927. package/src/prompts/tools/lsp-late-diagnostic.md +8 -0
  928. package/src/prompts/tools/lsp.md +39 -0
  929. package/src/prompts/tools/manage-skill.md +9 -0
  930. package/src/prompts/tools/memory-edit.md +8 -0
  931. package/src/prompts/tools/patch.md +57 -0
  932. package/src/prompts/tools/read.md +76 -0
  933. package/src/prompts/tools/recall.md +5 -0
  934. package/src/prompts/tools/reflect.md +5 -0
  935. package/src/prompts/tools/replace.md +29 -0
  936. package/src/prompts/tools/resolve.md +4 -0
  937. package/src/prompts/tools/retain.md +6 -0
  938. package/src/prompts/tools/rewind.md +13 -0
  939. package/src/prompts/tools/search-tool-bm25.md +32 -0
  940. package/src/prompts/tools/search.md +22 -0
  941. package/src/prompts/tools/ssh.md +22 -0
  942. package/src/prompts/tools/task-summary.md +17 -0
  943. package/src/prompts/tools/task.md +91 -0
  944. package/src/prompts/tools/todo.md +39 -0
  945. package/src/prompts/tools/web-search.md +6 -0
  946. package/src/prompts/tools/write.md +14 -0
  947. package/src/registry/agent-lifecycle.ts +270 -0
  948. package/src/registry/agent-registry.ts +190 -0
  949. package/src/sdk.ts +2919 -0
  950. package/src/secrets/index.ts +123 -0
  951. package/src/secrets/obfuscator.ts +298 -0
  952. package/src/secrets/regex.ts +21 -0
  953. package/src/session/agent-session.ts +12539 -0
  954. package/src/session/agent-storage.ts +478 -0
  955. package/src/session/artifacts.ts +153 -0
  956. package/src/session/auth-broker-config.ts +92 -0
  957. package/src/session/auth-storage.ts +24 -0
  958. package/src/session/blob-store.ts +255 -0
  959. package/src/session/client-bridge.ts +85 -0
  960. package/src/session/codex-auto-reset.ts +202 -0
  961. package/src/session/compact-modes.ts +105 -0
  962. package/src/session/history-storage.ts +361 -0
  963. package/src/session/indexed-session-storage.ts +427 -0
  964. package/src/session/messages.ts +546 -0
  965. package/src/session/redis-session-storage.ts +170 -0
  966. package/src/session/session-context.ts +399 -0
  967. package/src/session/session-dump-format.ts +216 -0
  968. package/src/session/session-entries.ts +198 -0
  969. package/src/session/session-history-format.ts +308 -0
  970. package/src/session/session-listing.ts +588 -0
  971. package/src/session/session-loader.ts +93 -0
  972. package/src/session/session-manager.ts +1748 -0
  973. package/src/session/session-migrations.ts +78 -0
  974. package/src/session/session-paths.ts +193 -0
  975. package/src/session/session-persistence.ts +147 -0
  976. package/src/session/session-storage.ts +590 -0
  977. package/src/session/shake-types.ts +43 -0
  978. package/src/session/snapcompact-inline.ts +542 -0
  979. package/src/session/snapcompact-savings-journal.ts +113 -0
  980. package/src/session/sql-session-storage.ts +314 -0
  981. package/src/session/streaming-output.ts +1330 -0
  982. package/src/session/tool-choice-queue.ts +290 -0
  983. package/src/session/unexpected-stop-classifier.ts +129 -0
  984. package/src/session/yield-queue.ts +183 -0
  985. package/src/slash-commands/acp-builtins.ts +70 -0
  986. package/src/slash-commands/available-commands.ts +105 -0
  987. package/src/slash-commands/builtin-registry.ts +2332 -0
  988. package/src/slash-commands/helpers/active-oauth-account.ts +44 -0
  989. package/src/slash-commands/helpers/collab-qrcode.ts +28 -0
  990. package/src/slash-commands/helpers/context-report.ts +66 -0
  991. package/src/slash-commands/helpers/format.ts +46 -0
  992. package/src/slash-commands/helpers/logout.ts +88 -0
  993. package/src/slash-commands/helpers/marketplace-manager.ts +25 -0
  994. package/src/slash-commands/helpers/mcp.ts +532 -0
  995. package/src/slash-commands/helpers/parse.ts +85 -0
  996. package/src/slash-commands/helpers/reset-usage.ts +66 -0
  997. package/src/slash-commands/helpers/ssh.ts +195 -0
  998. package/src/slash-commands/helpers/stats-dashboard.ts +85 -0
  999. package/src/slash-commands/helpers/todo.ts +279 -0
  1000. package/src/slash-commands/helpers/usage-report.ts +128 -0
  1001. package/src/slash-commands/marketplace-install-parser.ts +99 -0
  1002. package/src/slash-commands/types.ts +135 -0
  1003. package/src/ssh/config-writer.ts +183 -0
  1004. package/src/ssh/connection-manager.ts +510 -0
  1005. package/src/ssh/ssh-executor.ts +189 -0
  1006. package/src/ssh/sshfs-mount.ts +140 -0
  1007. package/src/ssh/utils.ts +8 -0
  1008. package/src/startup-splash.ts +19 -0
  1009. package/src/stt/asr-client.ts +521 -0
  1010. package/src/stt/asr-protocol.ts +65 -0
  1011. package/src/stt/asr-worker.ts +790 -0
  1012. package/src/stt/downloader.ts +138 -0
  1013. package/src/stt/endpointer.ts +259 -0
  1014. package/src/stt/index.ts +7 -0
  1015. package/src/stt/models.ts +150 -0
  1016. package/src/stt/recorder.ts +538 -0
  1017. package/src/stt/stt-controller.ts +380 -0
  1018. package/src/stt/transcriber.ts +60 -0
  1019. package/src/stt/wav.ts +173 -0
  1020. package/src/system-prompt.ts +709 -0
  1021. package/src/task/agents.ts +166 -0
  1022. package/src/task/commands.ts +132 -0
  1023. package/src/task/discovery.ts +122 -0
  1024. package/src/task/executor.ts +2356 -0
  1025. package/src/task/index.ts +1580 -0
  1026. package/src/task/name-generator.ts +1577 -0
  1027. package/src/task/omp-command.ts +26 -0
  1028. package/src/task/output-manager.ts +93 -0
  1029. package/src/task/parallel.ts +116 -0
  1030. package/src/task/persisted-revive.ts +128 -0
  1031. package/src/task/render.ts +1558 -0
  1032. package/src/task/repair-args.ts +129 -0
  1033. package/src/task/subprocess-tool-registry.ts +88 -0
  1034. package/src/task/types.ts +401 -0
  1035. package/src/task/worktree.ts +514 -0
  1036. package/src/telemetry-export.ts +144 -0
  1037. package/src/thinking.ts +187 -0
  1038. package/src/tiny/device.ts +111 -0
  1039. package/src/tiny/dtype.ts +101 -0
  1040. package/src/tiny/models.ts +252 -0
  1041. package/src/tiny/text.ts +169 -0
  1042. package/src/tiny/title-client.ts +538 -0
  1043. package/src/tiny/title-protocol.ts +56 -0
  1044. package/src/tiny/worker.ts +491 -0
  1045. package/src/tool-discovery/mode.ts +24 -0
  1046. package/src/tool-discovery/tool-index.ts +271 -0
  1047. package/src/tools/__tests__/json-tree.test.ts +35 -0
  1048. package/src/tools/approval.ts +189 -0
  1049. package/src/tools/ask.ts +977 -0
  1050. package/src/tools/ast-edit.ts +700 -0
  1051. package/src/tools/ast-grep.ts +483 -0
  1052. package/src/tools/auto-generated-guard.ts +322 -0
  1053. package/src/tools/bash-command-fixup.ts +37 -0
  1054. package/src/tools/bash-interactive.ts +408 -0
  1055. package/src/tools/bash-interceptor.ts +67 -0
  1056. package/src/tools/bash-pty-selection.ts +14 -0
  1057. package/src/tools/bash-skill-urls.ts +248 -0
  1058. package/src/tools/bash.ts +1405 -0
  1059. package/src/tools/browser/attach.ts +194 -0
  1060. package/src/tools/browser/cmux/cmux-tab.ts +1264 -0
  1061. package/src/tools/browser/cmux/rpc.ts +156 -0
  1062. package/src/tools/browser/cmux/socket-client.ts +309 -0
  1063. package/src/tools/browser/launch.ts +673 -0
  1064. package/src/tools/browser/readable.ts +112 -0
  1065. package/src/tools/browser/registry.ts +241 -0
  1066. package/src/tools/browser/render.ts +221 -0
  1067. package/src/tools/browser/tab-protocol.ts +107 -0
  1068. package/src/tools/browser/tab-supervisor.ts +799 -0
  1069. package/src/tools/browser/tab-worker-entry.ts +29 -0
  1070. package/src/tools/browser/tab-worker.ts +1226 -0
  1071. package/src/tools/browser.ts +403 -0
  1072. package/src/tools/builtin-names.ts +34 -0
  1073. package/src/tools/checkpoint.ts +136 -0
  1074. package/src/tools/conflict-detect.ts +718 -0
  1075. package/src/tools/context.ts +39 -0
  1076. package/src/tools/debug.ts +1087 -0
  1077. package/src/tools/eval-backends.ts +27 -0
  1078. package/src/tools/eval-render.ts +762 -0
  1079. package/src/tools/eval.ts +600 -0
  1080. package/src/tools/fetch.ts +1902 -0
  1081. package/src/tools/file-recorder.ts +35 -0
  1082. package/src/tools/find.ts +629 -0
  1083. package/src/tools/fs-cache-invalidation.ts +28 -0
  1084. package/src/tools/gh-cache-invalidation.ts +255 -0
  1085. package/src/tools/gh-format.ts +12 -0
  1086. package/src/tools/gh-renderer.ts +481 -0
  1087. package/src/tools/gh.ts +3752 -0
  1088. package/src/tools/github-cache.ts +663 -0
  1089. package/src/tools/grouped-file-output.ts +210 -0
  1090. package/src/tools/image-gen.ts +1586 -0
  1091. package/src/tools/index.ts +649 -0
  1092. package/src/tools/inspect-image-renderer.ts +132 -0
  1093. package/src/tools/inspect-image.ts +260 -0
  1094. package/src/tools/irc.ts +788 -0
  1095. package/src/tools/job.ts +612 -0
  1096. package/src/tools/json-tree.ts +260 -0
  1097. package/src/tools/jtd-to-json-schema.ts +219 -0
  1098. package/src/tools/jtd-to-typescript.ts +136 -0
  1099. package/src/tools/jtd-utils.ts +102 -0
  1100. package/src/tools/learn.ts +141 -0
  1101. package/src/tools/list-limit.ts +40 -0
  1102. package/src/tools/manage-skill.ts +100 -0
  1103. package/src/tools/match-line-format.ts +20 -0
  1104. package/src/tools/memory-edit.ts +59 -0
  1105. package/src/tools/memory-recall.ts +102 -0
  1106. package/src/tools/memory-reflect.ts +88 -0
  1107. package/src/tools/memory-render.ts +202 -0
  1108. package/src/tools/memory-retain.ts +89 -0
  1109. package/src/tools/output-meta.ts +768 -0
  1110. package/src/tools/output-schema-validator.ts +132 -0
  1111. package/src/tools/path-utils.ts +1116 -0
  1112. package/src/tools/plan-mode-guard.ts +142 -0
  1113. package/src/tools/puppeteer/00_stealth_tampering.txt +63 -0
  1114. package/src/tools/puppeteer/01_stealth_activity.txt +20 -0
  1115. package/src/tools/puppeteer/02_stealth_hairline.txt +11 -0
  1116. package/src/tools/puppeteer/03_stealth_botd.txt +384 -0
  1117. package/src/tools/puppeteer/04_stealth_iframe.txt +81 -0
  1118. package/src/tools/puppeteer/05_stealth_webgl.txt +75 -0
  1119. package/src/tools/puppeteer/06_stealth_screen.txt +72 -0
  1120. package/src/tools/puppeteer/07_stealth_fonts.txt +97 -0
  1121. package/src/tools/puppeteer/08_stealth_audio.txt +51 -0
  1122. package/src/tools/puppeteer/09_stealth_locale.txt +46 -0
  1123. package/src/tools/puppeteer/10_stealth_plugins.txt +208 -0
  1124. package/src/tools/puppeteer/11_stealth_hardware.txt +8 -0
  1125. package/src/tools/puppeteer/12_stealth_codecs.txt +40 -0
  1126. package/src/tools/puppeteer/13_stealth_worker.txt +74 -0
  1127. package/src/tools/read.ts +3124 -0
  1128. package/src/tools/render-utils.ts +895 -0
  1129. package/src/tools/renderers.ts +86 -0
  1130. package/src/tools/report-tool-issue.ts +530 -0
  1131. package/src/tools/resolve.ts +302 -0
  1132. package/src/tools/review.ts +251 -0
  1133. package/src/tools/search-tool-bm25.ts +351 -0
  1134. package/src/tools/search.ts +1583 -0
  1135. package/src/tools/sqlite-reader.ts +828 -0
  1136. package/src/tools/ssh.ts +369 -0
  1137. package/src/tools/todo.ts +938 -0
  1138. package/src/tools/tool-errors.ts +62 -0
  1139. package/src/tools/tool-result.ts +102 -0
  1140. package/src/tools/tool-timeouts.ts +30 -0
  1141. package/src/tools/tts.ts +265 -0
  1142. package/src/tools/write.ts +1182 -0
  1143. package/src/tools/yield.ts +269 -0
  1144. package/src/tts/downloader.ts +64 -0
  1145. package/src/tts/index.ts +8 -0
  1146. package/src/tts/models.ts +137 -0
  1147. package/src/tts/player.ts +137 -0
  1148. package/src/tts/runtime.ts +21 -0
  1149. package/src/tts/streaming-player.ts +266 -0
  1150. package/src/tts/tts-client.ts +642 -0
  1151. package/src/tts/tts-protocol.ts +60 -0
  1152. package/src/tts/tts-worker.ts +505 -0
  1153. package/src/tts/vocalizer.ts +162 -0
  1154. package/src/tts/wav.ts +58 -0
  1155. package/src/tui/code-cell.ts +257 -0
  1156. package/src/tui/file-list.ts +55 -0
  1157. package/src/tui/hyperlink.ts +178 -0
  1158. package/src/tui/index.ts +13 -0
  1159. package/src/tui/output-block.ts +240 -0
  1160. package/src/tui/status-line.ts +54 -0
  1161. package/src/tui/tree-list.ts +133 -0
  1162. package/src/tui/types.ts +15 -0
  1163. package/src/tui/utils.ts +103 -0
  1164. package/src/tui/width-aware-text.ts +58 -0
  1165. package/src/utils/block-context.ts +312 -0
  1166. package/src/utils/changelog.ts +132 -0
  1167. package/src/utils/clipboard.ts +262 -0
  1168. package/src/utils/command-args.ts +76 -0
  1169. package/src/utils/commit-message-generator.ts +147 -0
  1170. package/src/utils/edit-mode.ts +41 -0
  1171. package/src/utils/enhanced-paste.ts +230 -0
  1172. package/src/utils/event-bus.ts +33 -0
  1173. package/src/utils/external-editor.ts +78 -0
  1174. package/src/utils/file-display-mode.ts +45 -0
  1175. package/src/utils/file-mentions.ts +284 -0
  1176. package/src/utils/git.ts +1838 -0
  1177. package/src/utils/image-loading.ts +231 -0
  1178. package/src/utils/image-resize.ts +309 -0
  1179. package/src/utils/image-vision-fallback.ts +197 -0
  1180. package/src/utils/ipc.ts +38 -0
  1181. package/src/utils/jj.ts +248 -0
  1182. package/src/utils/lang-from-path.ts +244 -0
  1183. package/src/utils/markit.ts +143 -0
  1184. package/src/utils/mupdf-wasm-embed.ts +12 -0
  1185. package/src/utils/open.ts +55 -0
  1186. package/src/utils/qrcode.ts +535 -0
  1187. package/src/utils/session-color.ts +142 -0
  1188. package/src/utils/shell-snapshot.ts +187 -0
  1189. package/src/utils/sixel.ts +69 -0
  1190. package/src/utils/thinking-display.ts +11 -0
  1191. package/src/utils/title-generator.ts +416 -0
  1192. package/src/utils/tool-choice.ts +49 -0
  1193. package/src/utils/tools-manager.ts +372 -0
  1194. package/src/utils/turndown.ts +83 -0
  1195. package/src/utils/zip.ts +1091 -0
  1196. package/src/web/kagi.ts +304 -0
  1197. package/src/web/parallel.ts +353 -0
  1198. package/src/web/scrapers/artifacthub.ts +207 -0
  1199. package/src/web/scrapers/arxiv.ts +83 -0
  1200. package/src/web/scrapers/aur.ts +162 -0
  1201. package/src/web/scrapers/biorxiv.ts +133 -0
  1202. package/src/web/scrapers/bluesky.ts +262 -0
  1203. package/src/web/scrapers/brew.ts +172 -0
  1204. package/src/web/scrapers/cheatsh.ts +68 -0
  1205. package/src/web/scrapers/chocolatey.ts +196 -0
  1206. package/src/web/scrapers/choosealicense.ts +95 -0
  1207. package/src/web/scrapers/cisa-kev.ts +87 -0
  1208. package/src/web/scrapers/clojars.ts +154 -0
  1209. package/src/web/scrapers/coingecko.ts +177 -0
  1210. package/src/web/scrapers/crates-io.ts +97 -0
  1211. package/src/web/scrapers/crossref.ts +136 -0
  1212. package/src/web/scrapers/devto.ts +147 -0
  1213. package/src/web/scrapers/discogs.ts +306 -0
  1214. package/src/web/scrapers/discourse.ts +197 -0
  1215. package/src/web/scrapers/dockerhub.ts +138 -0
  1216. package/src/web/scrapers/docs-rs.ts +652 -0
  1217. package/src/web/scrapers/fdroid.ts +134 -0
  1218. package/src/web/scrapers/firefox-addons.ts +191 -0
  1219. package/src/web/scrapers/flathub.ts +223 -0
  1220. package/src/web/scrapers/github-gist.ts +58 -0
  1221. package/src/web/scrapers/github.ts +800 -0
  1222. package/src/web/scrapers/gitlab.ts +401 -0
  1223. package/src/web/scrapers/go-pkg.ts +266 -0
  1224. package/src/web/scrapers/hackage.ts +140 -0
  1225. package/src/web/scrapers/hackernews.ts +189 -0
  1226. package/src/web/scrapers/hex.ts +105 -0
  1227. package/src/web/scrapers/huggingface.ts +321 -0
  1228. package/src/web/scrapers/iacr.ts +89 -0
  1229. package/src/web/scrapers/index.ts +252 -0
  1230. package/src/web/scrapers/jetbrains-marketplace.ts +159 -0
  1231. package/src/web/scrapers/lemmy.ts +203 -0
  1232. package/src/web/scrapers/lobsters.ts +175 -0
  1233. package/src/web/scrapers/mastodon.ts +292 -0
  1234. package/src/web/scrapers/maven.ts +138 -0
  1235. package/src/web/scrapers/mdn.ts +173 -0
  1236. package/src/web/scrapers/metacpan.ts +222 -0
  1237. package/src/web/scrapers/musicbrainz.ts +250 -0
  1238. package/src/web/scrapers/npm.ts +98 -0
  1239. package/src/web/scrapers/nuget.ts +183 -0
  1240. package/src/web/scrapers/nvd.ts +222 -0
  1241. package/src/web/scrapers/ollama.ts +239 -0
  1242. package/src/web/scrapers/open-vsx.ts +106 -0
  1243. package/src/web/scrapers/opencorporates.ts +292 -0
  1244. package/src/web/scrapers/openlibrary.ts +336 -0
  1245. package/src/web/scrapers/orcid.ts +286 -0
  1246. package/src/web/scrapers/osv.ts +176 -0
  1247. package/src/web/scrapers/packagist.ts +160 -0
  1248. package/src/web/scrapers/pub-dev.ts +143 -0
  1249. package/src/web/scrapers/pubmed.ts +211 -0
  1250. package/src/web/scrapers/pypi.ts +112 -0
  1251. package/src/web/scrapers/rawg.ts +110 -0
  1252. package/src/web/scrapers/readthedocs.ts +120 -0
  1253. package/src/web/scrapers/reddit.ts +95 -0
  1254. package/src/web/scrapers/repology.ts +251 -0
  1255. package/src/web/scrapers/rfc.ts +201 -0
  1256. package/src/web/scrapers/rubygems.ts +103 -0
  1257. package/src/web/scrapers/searchcode.ts +189 -0
  1258. package/src/web/scrapers/sec-edgar.ts +261 -0
  1259. package/src/web/scrapers/semantic-scholar.ts +171 -0
  1260. package/src/web/scrapers/snapcraft.ts +187 -0
  1261. package/src/web/scrapers/sourcegraph.ts +336 -0
  1262. package/src/web/scrapers/spdx.ts +108 -0
  1263. package/src/web/scrapers/spotify.ts +198 -0
  1264. package/src/web/scrapers/stackoverflow.ts +120 -0
  1265. package/src/web/scrapers/terraform.ts +277 -0
  1266. package/src/web/scrapers/tldr.ts +47 -0
  1267. package/src/web/scrapers/twitter.ts +94 -0
  1268. package/src/web/scrapers/types.ts +354 -0
  1269. package/src/web/scrapers/utils.ts +109 -0
  1270. package/src/web/scrapers/vimeo.ts +133 -0
  1271. package/src/web/scrapers/vscode-marketplace.ts +187 -0
  1272. package/src/web/scrapers/w3c.ts +156 -0
  1273. package/src/web/scrapers/wikidata.ts +344 -0
  1274. package/src/web/scrapers/wikipedia.ts +84 -0
  1275. package/src/web/scrapers/youtube.ts +325 -0
  1276. package/src/web/search/index.ts +317 -0
  1277. package/src/web/search/provider.ts +169 -0
  1278. package/src/web/search/providers/anthropic.ts +343 -0
  1279. package/src/web/search/providers/base.ts +90 -0
  1280. package/src/web/search/providers/brave.ts +152 -0
  1281. package/src/web/search/providers/codex.ts +593 -0
  1282. package/src/web/search/providers/exa.ts +400 -0
  1283. package/src/web/search/providers/gemini.ts +518 -0
  1284. package/src/web/search/providers/jina.ts +111 -0
  1285. package/src/web/search/providers/kagi.ts +86 -0
  1286. package/src/web/search/providers/kimi.ts +196 -0
  1287. package/src/web/search/providers/parallel.ts +225 -0
  1288. package/src/web/search/providers/perplexity-auth.ts +133 -0
  1289. package/src/web/search/providers/perplexity.ts +866 -0
  1290. package/src/web/search/providers/searxng.ts +325 -0
  1291. package/src/web/search/providers/synthetic.ts +114 -0
  1292. package/src/web/search/providers/tavily.ts +176 -0
  1293. package/src/web/search/providers/utils.ts +128 -0
  1294. package/src/web/search/providers/zai.ts +333 -0
  1295. package/src/web/search/render.ts +262 -0
  1296. package/src/web/search/types.ts +462 -0
  1297. package/src/web/search/utils.ts +17 -0
  1298. package/src/workspace-tree.ts +326 -0
@@ -0,0 +1,1653 @@
1
+ import * as fs from "node:fs/promises";
2
+ import * as os from "node:os";
3
+ import * as path from "node:path";
4
+ import { CompactionCancelledError, type CompactionOutcome } from "@oh-my-pi/pi-agent-core/compaction";
5
+ import {
6
+ getEnvApiKey,
7
+ getProviderDetails,
8
+ type ProviderDetails,
9
+ type UsageLimit,
10
+ type UsageReport,
11
+ } from "@oh-my-pi/pi-ai";
12
+ import { Loader, Markdown, padding, Spacer, Text, visibleWidth } from "@oh-my-pi/pi-tui";
13
+ import { formatDuration, Snowflake } from "@oh-my-pi/pi-utils";
14
+ import { shouldEnableAppendOnlyContext } from "../../config/append-only-context-mode";
15
+ import { type LoadedCustomShare, loadCustomShare } from "../../export/custom-share";
16
+ import { shareSession } from "../../export/share";
17
+ import type { CompactOptions } from "../../extensibility/extensions/types";
18
+ import {
19
+ diffMentalModelContent,
20
+ type HindsightApi,
21
+ type HindsightSessionState,
22
+ loadHindsightConfig,
23
+ reloadMentalModelsForSession,
24
+ resolveSeedsForScope,
25
+ seedAlreadyExists,
26
+ summarizeMentalModel,
27
+ } from "../../hindsight";
28
+ import { resolveMemoryBackend } from "../../memory-backend";
29
+ import { BashExecutionComponent } from "../../modes/components/bash-execution";
30
+ import { BorderedLoader } from "../../modes/components/bordered-loader";
31
+ import { DynamicBorder } from "../../modes/components/dynamic-border";
32
+ import { EvalExecutionComponent } from "../../modes/components/eval-execution";
33
+ import { TranscriptBlock } from "../../modes/components/transcript-container";
34
+ import { getMarkdownTheme, getSymbolTheme, theme } from "../../modes/theme/theme";
35
+ import type { InteractiveModeContext } from "../../modes/types";
36
+ import { computeContextBreakdown, renderContextUsage } from "../../modes/utils/context-usage";
37
+ import { buildHotkeysMarkdown } from "../../modes/utils/hotkeys-markdown";
38
+ import { buildToolsMarkdown } from "../../modes/utils/tools-markdown";
39
+ import type { AsyncJobSnapshotItem } from "../../session/agent-session";
40
+ import type { AuthStorage, OAuthAccountIdentity } from "../../session/auth-storage";
41
+ import type { CompactMode } from "../../session/compact-modes";
42
+ import type { NewSessionOptions } from "../../session/session-entries";
43
+ import { formatShakeSummary, type ShakeMode, type ShakeResult } from "../../session/shake-types";
44
+ import { limitMatchesActiveAccount } from "../../slash-commands/helpers/active-oauth-account";
45
+ import { outputMeta } from "../../tools/output-meta";
46
+ import { resolveToCwd, stripOuterDoubleQuotes } from "../../tools/path-utils";
47
+ import { replaceTabs } from "../../tools/render-utils";
48
+ import { getChangelogPath, parseChangelog } from "../../utils/changelog";
49
+ import { copyToClipboard } from "../../utils/clipboard";
50
+ import { openPath } from "../../utils/open";
51
+ import { setSessionTerminalTitle } from "../../utils/title-generator";
52
+
53
+ function showMarkdownPanel(ctx: InteractiveModeContext, title: string, markdown: string): void {
54
+ const block = new TranscriptBlock();
55
+ block.addChild(new DynamicBorder());
56
+ block.addChild(new Text(theme.bold(theme.fg("accent", title)), 1, 0));
57
+ block.addChild(new Spacer(1));
58
+ block.addChild(new Markdown(markdown.trim(), 1, 1, getMarkdownTheme()));
59
+ block.addChild(new DynamicBorder());
60
+ ctx.present(block);
61
+ }
62
+
63
+ export class CommandController {
64
+ constructor(private readonly ctx: InteractiveModeContext) {}
65
+
66
+ openInBrowser(urlOrPath: string): void {
67
+ openPath(urlOrPath);
68
+ }
69
+
70
+ async handleExportCommand(text: string): Promise<void> {
71
+ const parts = text.split(/\s+/);
72
+ const arg = parts.length > 1 ? parts[1] : undefined;
73
+
74
+ if (arg === "--copy" || arg === "clipboard" || arg === "copy") {
75
+ this.ctx.showWarning("Use /dump to copy the session to clipboard.");
76
+ return;
77
+ }
78
+
79
+ try {
80
+ const filePath = await this.ctx.session.exportToHtml(arg);
81
+ this.ctx.showStatus(`Session exported to: ${filePath}`);
82
+ this.openInBrowser(filePath);
83
+ } catch (error: unknown) {
84
+ this.ctx.showError(`Failed to export session: ${error instanceof Error ? error.message : "Unknown error"}`);
85
+ }
86
+ }
87
+
88
+ handleDumpCommand() {
89
+ try {
90
+ const formatted = this.ctx.session.formatSessionAsText();
91
+ if (!formatted) {
92
+ this.ctx.showError("No messages to dump yet.");
93
+ return;
94
+ }
95
+ copyToClipboard(formatted);
96
+ this.ctx.showStatus("Session copied to clipboard");
97
+ } catch (error: unknown) {
98
+ this.ctx.showError(`Failed to copy session: ${error instanceof Error ? error.message : "Unknown error"}`);
99
+ }
100
+ }
101
+
102
+ handleAdvisorDumpCommand(isRaw = false) {
103
+ try {
104
+ const advisorHistory = this.ctx.session.formatAdvisorHistoryAsText({ compact: !isRaw });
105
+ if (advisorHistory === null) {
106
+ this.ctx.showError("Advisor is not active for this session.");
107
+ return;
108
+ }
109
+ if (!advisorHistory) {
110
+ this.ctx.showError("Advisor has no history yet.");
111
+ return;
112
+ }
113
+ copyToClipboard(advisorHistory);
114
+ this.ctx.showStatus("Advisor history copied to clipboard");
115
+ } catch (error: unknown) {
116
+ this.ctx.showError(
117
+ `Failed to copy advisor history: ${error instanceof Error ? error.message : "Unknown error"}`,
118
+ );
119
+ }
120
+ }
121
+
122
+ async handleDebugTranscriptCommand(): Promise<void> {
123
+ try {
124
+ const width = Math.max(1, this.ctx.ui.terminal.columns);
125
+ const renderedLines = this.ctx.chatContainer.render(width).map(line => replaceTabs(Bun.stripANSI(line)));
126
+ const rendered = renderedLines.join("\n").trimEnd();
127
+ if (!rendered) {
128
+ this.ctx.showError("No messages to dump yet.");
129
+ return;
130
+ }
131
+ const tmpPath = path.join(os.tmpdir(), `${Snowflake.next()}-tmp.txt`);
132
+ await Bun.write(tmpPath, `${rendered}\n`);
133
+ this.ctx.showStatus(`Debug transcript written to:\n${tmpPath}`);
134
+ } catch (error: unknown) {
135
+ this.ctx.showError(
136
+ `Failed to write debug transcript: ${error instanceof Error ? error.message : "Unknown error"}`,
137
+ );
138
+ }
139
+ }
140
+
141
+ async handleShareCommand(): Promise<void> {
142
+ let customShare: LoadedCustomShare | null;
143
+ try {
144
+ customShare = await loadCustomShare();
145
+ } catch (err) {
146
+ this.ctx.showError(err instanceof Error ? err.message : String(err));
147
+ return;
148
+ }
149
+
150
+ const loader = new BorderedLoader(this.ctx.ui, theme, "Sharing session...");
151
+ this.ctx.editorContainer.clear();
152
+ this.ctx.editorContainer.addChild(loader);
153
+ this.ctx.ui.setFocus(loader);
154
+ this.ctx.ui.requestRender();
155
+
156
+ const restoreEditor = () => {
157
+ loader.dispose();
158
+ this.ctx.editorContainer.clear();
159
+ this.ctx.editorContainer.addChild(this.ctx.editor);
160
+ this.ctx.ui.setFocus(this.ctx.editor);
161
+ };
162
+ loader.onAbort = () => {
163
+ restoreEditor();
164
+ this.ctx.showStatus("Share cancelled");
165
+ };
166
+
167
+ // Custom share scripts keep their legacy contract: they receive a path
168
+ // to a standalone HTML export. No fallback to the default flow on error.
169
+ if (customShare) {
170
+ const tmpFile = path.join(os.tmpdir(), `${Snowflake.next()}.html`);
171
+ try {
172
+ await this.ctx.session.exportToHtml(tmpFile);
173
+ const result = await customShare.fn(tmpFile);
174
+ if (loader.signal.aborted) return;
175
+ restoreEditor();
176
+
177
+ if (typeof result === "string") {
178
+ this.ctx.showStatus(`Share URL: ${result}`);
179
+ this.openInBrowser(result);
180
+ } else if (result) {
181
+ const parts: string[] = [];
182
+ if (result.url) parts.push(`Share URL: ${result.url}`);
183
+ if (result.message) parts.push(result.message);
184
+ if (parts.length > 0) this.ctx.showStatus(parts.join("\n"));
185
+ if (result.url) this.openInBrowser(result.url);
186
+ } else {
187
+ this.ctx.showStatus("Session shared");
188
+ }
189
+ } catch (err) {
190
+ if (!loader.signal.aborted) {
191
+ restoreEditor();
192
+ this.ctx.showError(`Custom share failed: ${err instanceof Error ? err.message : String(err)}`);
193
+ }
194
+ } finally {
195
+ await fs.rm(tmpFile, { force: true }).catch(() => {});
196
+ }
197
+ return;
198
+ }
199
+
200
+ // Default: encrypted snapshot to a secret gist (preferred) or the share
201
+ // server; the key rides in the link fragment and never leaves the client.
202
+ try {
203
+ const result = await shareSession(this.ctx.session.sessionManager, {
204
+ serverUrl: this.ctx.settings.get("share.serverUrl"),
205
+ state: this.ctx.session.state,
206
+ obfuscator: this.ctx.settings.get("share.redactSecrets") ? this.ctx.session.obfuscator : undefined,
207
+ });
208
+ if (loader.signal.aborted) return;
209
+ restoreEditor();
210
+
211
+ const lines = [`Share URL: ${result.url}`];
212
+ if (result.gistUrl) lines.push(`Gist: ${result.gistUrl}`);
213
+ if (result.truncated) lines.push("Note: large content was trimmed to fit the share size limit.");
214
+ this.ctx.showStatus(lines.join("\n"));
215
+ this.openInBrowser(result.url);
216
+ } catch (error: unknown) {
217
+ if (!loader.signal.aborted) {
218
+ restoreEditor();
219
+ this.ctx.showError(`Failed to share session: ${error instanceof Error ? error.message : "Unknown error"}`);
220
+ }
221
+ }
222
+ }
223
+
224
+ async handleSessionCommand(): Promise<void> {
225
+ const stats = this.ctx.session.getSessionStats();
226
+ const premiumRequests =
227
+ "premiumRequests" in stats && typeof stats.premiumRequests === "number"
228
+ ? stats.premiumRequests
229
+ : this.ctx.session.sessionManager.getUsageStatistics().premiumRequests;
230
+ const normalizedPremiumRequests = Math.round((premiumRequests + Number.EPSILON) * 100) / 100;
231
+
232
+ let info = `${theme.bold("Session Info")}\n\n`;
233
+ info += `${theme.fg("dim", "File:")} ${stats.sessionFile ?? "In-memory"}\n`;
234
+ info += `${theme.fg("dim", "ID:")} ${stats.sessionId}\n\n`;
235
+ info += `\n${theme.bold("Provider")}\n`;
236
+ const model = this.ctx.session.model;
237
+ if (!model) {
238
+ info += `${theme.fg("dim", "No model selected")}\n`;
239
+ } else {
240
+ const authMode = resolveProviderAuthMode(this.ctx.session.modelRegistry.authStorage, model.provider);
241
+ const openaiWebsocketSetting = this.ctx.settings.get("providers.openaiWebsockets") ?? "auto";
242
+ const preferOpenAICodexWebsockets =
243
+ openaiWebsocketSetting === "on" ? true : openaiWebsocketSetting === "off" ? false : undefined;
244
+ const credentialSource = this.ctx.session.modelRegistry.authStorage.describeCredentialSource(
245
+ model.provider,
246
+ stats.sessionId,
247
+ );
248
+ const providerDetails = getProviderDetails({
249
+ model,
250
+ sessionId: stats.sessionId,
251
+ authMode,
252
+ credentialSource,
253
+ preferWebsockets: preferOpenAICodexWebsockets,
254
+ providerSessionState: this.ctx.session.providerSessionState,
255
+ });
256
+ info += renderProviderSection(providerDetails, theme);
257
+ }
258
+ info += `\n`;
259
+ info += `${theme.bold("Messages")}\n`;
260
+ info += `${theme.fg("dim", "User:")} ${stats.userMessages}\n`;
261
+ info += `${theme.fg("dim", "Assistant:")} ${stats.assistantMessages}\n`;
262
+ info += `${theme.fg("dim", "Tool Calls:")} ${stats.toolCalls}\n`;
263
+ info += `${theme.fg("dim", "Tool Results:")} ${stats.toolResults}\n`;
264
+ info += `${theme.fg("dim", "Total:")} ${stats.totalMessages}\n\n`;
265
+ // Append-only context
266
+ {
267
+ const setting = this.ctx.settings.get("provider.appendOnlyContext") ?? "auto";
268
+ const model = this.ctx.session.model;
269
+ const mode = shouldEnableAppendOnlyContext(setting, model);
270
+ const activeLabel = mode ? theme.fg("success", "active") : theme.fg("dim", "inactive");
271
+ const settingLabel = setting === "auto" ? `${setting} (${model?.provider ?? "?"})` : setting;
272
+ info += `${theme.fg("dim", "Append-Only:")} ${activeLabel} (setting: ${settingLabel})\n`;
273
+ }
274
+ info += `${theme.bold("Tokens")}\n`;
275
+ info += `${theme.fg("dim", "Input:")} ${stats.tokens.input.toLocaleString()}\n`;
276
+ info += `${theme.fg("dim", "Output:")} ${stats.tokens.output.toLocaleString()}\n`;
277
+ if (stats.tokens.cacheRead > 0) {
278
+ info += `${theme.fg("dim", "Cache Read:")} ${stats.tokens.cacheRead.toLocaleString()}\n`;
279
+ }
280
+ if (stats.tokens.cacheWrite > 0) {
281
+ info += `${theme.fg("dim", "Cache Write:")} ${stats.tokens.cacheWrite.toLocaleString()}\n`;
282
+ }
283
+ info += `${theme.fg("dim", "Total:")} ${stats.tokens.total.toLocaleString()}\n`;
284
+
285
+ if (stats.cost > 0 || normalizedPremiumRequests > 0) {
286
+ info += `\n${theme.bold("Cost")}\n`;
287
+ if (stats.cost > 0) {
288
+ info += `${theme.fg("dim", "Total:")} ${stats.cost.toFixed(4)}\n`;
289
+ }
290
+ if (normalizedPremiumRequests > 0) {
291
+ info += `${theme.fg("dim", "Premium Requests:")} ${normalizedPremiumRequests.toLocaleString()}\n`;
292
+ }
293
+ }
294
+
295
+ if (this.ctx.lspServers && this.ctx.lspServers.length > 0) {
296
+ info += `\n${theme.bold("LSP Servers")}\n`;
297
+ for (const server of this.ctx.lspServers) {
298
+ const statusColor =
299
+ server.status === "ready"
300
+ ? "success"
301
+ : server.status === "available"
302
+ ? "dim"
303
+ : server.status === "connecting"
304
+ ? "warning"
305
+ : "error";
306
+ const statusText =
307
+ server.status === "error" && server.error ? `${server.status}: ${server.error}` : server.status;
308
+ info += `${theme.fg("dim", `${server.name}:`)} ${theme.fg(statusColor, statusText)} ${theme.fg("dim", `(${server.fileTypes.join(", ")})`)}\n`;
309
+ }
310
+ }
311
+
312
+ if (this.ctx.mcpManager) {
313
+ const mcpServers = this.ctx.mcpManager.getConnectedServers();
314
+ info += `\n${theme.bold("MCP Servers")}\n`;
315
+ if (mcpServers.length === 0) {
316
+ info += `${theme.fg("dim", "None connected")}\n`;
317
+ } else {
318
+ for (const name of mcpServers) {
319
+ const conn = this.ctx.mcpManager.getConnection(name);
320
+ const toolCount = conn?.tools?.length ?? 0;
321
+ info += `${theme.fg("dim", `${name}:`)} ${theme.fg("success", "connected")} ${theme.fg("dim", `(${toolCount} tools)`)}\n`;
322
+ }
323
+ }
324
+ }
325
+
326
+ this.ctx.present([new Spacer(1), new Text(info, 1, 0)]);
327
+ }
328
+
329
+ async handleAdvisorStatusCommand(): Promise<void> {
330
+ const stats = this.ctx.session.getAdvisorStats();
331
+ if (!stats.active) {
332
+ this.ctx.present([
333
+ new Spacer(1),
334
+ new Text(
335
+ stats.configured
336
+ ? "Advisor setting is enabled, but no model is assigned to the 'advisor' role."
337
+ : "Advisor is disabled.",
338
+ 1,
339
+ 0,
340
+ ),
341
+ ]);
342
+ return;
343
+ }
344
+ const model = stats.model!;
345
+ let info = `${theme.bold("Advisor Status")}\n\n`;
346
+ info += `${theme.bold("Provider")}\n`;
347
+ info += `${theme.fg("dim", "Model:")} ${model.provider}/${model.id}\n`;
348
+ info += `\n${theme.bold("Messages")}\n`;
349
+ info += `${theme.fg("dim", "User:")} ${stats.messages.user.toLocaleString()}\n`;
350
+ info += `${theme.fg("dim", "Assistant:")} ${stats.messages.assistant.toLocaleString()}\n`;
351
+ info += `${theme.fg("dim", "Total:")} ${stats.messages.total.toLocaleString()}\n`;
352
+ info += `\n${theme.bold("Context")}\n`;
353
+ if (stats.contextWindow > 0) {
354
+ const percent = Math.round((stats.contextTokens / stats.contextWindow) * 100);
355
+ info += `${theme.fg("dim", "Tokens:")} ${stats.contextTokens.toLocaleString()} / ${stats.contextWindow.toLocaleString()} (${percent}%)\n`;
356
+ } else {
357
+ info += `${theme.fg("dim", "Tokens:")} ${stats.contextTokens.toLocaleString()}\n`;
358
+ }
359
+ info += `\n${theme.bold("Spend")}\n`;
360
+ info += `${theme.fg("dim", "Input:")} ${stats.tokens.input.toLocaleString()}\n`;
361
+ info += `${theme.fg("dim", "Output:")} ${stats.tokens.output.toLocaleString()}\n`;
362
+ if (stats.tokens.cacheRead > 0) {
363
+ info += `${theme.fg("dim", "Cache Read:")} ${stats.tokens.cacheRead.toLocaleString()}\n`;
364
+ }
365
+ if (stats.tokens.cacheWrite > 0) {
366
+ info += `${theme.fg("dim", "Cache Write:")} ${stats.tokens.cacheWrite.toLocaleString()}\n`;
367
+ }
368
+ info += `${theme.fg("dim", "Total:")} ${stats.tokens.total.toLocaleString()}\n`;
369
+ if (stats.cost > 0) {
370
+ info += `\n${theme.bold("Cost")}\n`;
371
+ info += `${theme.fg("dim", "Total:")} $${stats.cost.toFixed(4)}\n`;
372
+ }
373
+ this.ctx.present([new Spacer(1), new Text(info, 1, 0)]);
374
+ }
375
+
376
+ async handleJobsCommand(): Promise<void> {
377
+ const snapshot = this.ctx.session.getAsyncJobSnapshot({ recentLimit: 5 });
378
+ if (!snapshot) {
379
+ this.ctx.showWarning("Async background jobs are unavailable in this session.");
380
+ return;
381
+ }
382
+
383
+ const now = Date.now();
384
+ const lineWidth = Math.max(24, (this.ctx.ui.terminal.columns ?? 100) - 24);
385
+ let info = `${theme.bold("Background Jobs")}\n\n`;
386
+ info += `${theme.fg("dim", "Running:")} ${snapshot.running.length}\n`;
387
+
388
+ if (snapshot.running.length === 0 && snapshot.recent.length === 0) {
389
+ info += `\n${theme.fg("dim", "No async jobs yet.")}\n`;
390
+ this.ctx.present([new Spacer(1), new Text(info, 1, 0)]);
391
+ return;
392
+ }
393
+
394
+ if (snapshot.running.length > 0) {
395
+ info += `\n${theme.bold("Running Jobs")}\n`;
396
+ for (const job of snapshot.running) {
397
+ info += `${renderJobLine(job, now)}\n`;
398
+ info += ` ${theme.fg("dim", truncateJobLabel(job.label, lineWidth))}\n`;
399
+ }
400
+ }
401
+
402
+ if (snapshot.recent.length > 0) {
403
+ info += `\n${theme.bold("Recent Jobs")}\n`;
404
+ for (const job of snapshot.recent) {
405
+ info += `${renderJobLine(job, now)}\n`;
406
+ info += ` ${theme.fg("dim", truncateJobLabel(job.label, lineWidth))}\n`;
407
+ }
408
+ }
409
+
410
+ this.ctx.present([new Spacer(1), new Text(info.trimEnd(), 1, 0)]);
411
+ }
412
+
413
+ async handleUsageCommand(reports?: UsageReport[] | null): Promise<void> {
414
+ let usageReports = reports ?? null;
415
+ if (!usageReports) {
416
+ const provider = this.ctx.session as { fetchUsageReports?: () => Promise<UsageReport[] | null> };
417
+ if (!provider.fetchUsageReports) {
418
+ this.ctx.showWarning("Usage reporting is not configured for this session.");
419
+ return;
420
+ }
421
+ try {
422
+ usageReports = await provider.fetchUsageReports();
423
+ } catch (error) {
424
+ this.ctx.showError(`Failed to fetch usage data: ${error instanceof Error ? error.message : String(error)}`);
425
+ return;
426
+ }
427
+ }
428
+
429
+ if (!usageReports || usageReports.length === 0) {
430
+ this.ctx.showWarning("No usage data available.");
431
+ return;
432
+ }
433
+
434
+ const availableWidth = Math.max(40, (this.ctx.ui.terminal.columns ?? 100) - 2);
435
+ const currentProvider = this.ctx.session.model?.provider;
436
+ const activeAccount = currentProvider
437
+ ? this.ctx.session.modelRegistry.authStorage.getOAuthAccountIdentity(
438
+ currentProvider,
439
+ this.ctx.session.sessionId,
440
+ )
441
+ : undefined;
442
+ const output = renderUsageReports(usageReports, theme, Date.now(), availableWidth, provider =>
443
+ provider === currentProvider ? activeAccount : undefined,
444
+ );
445
+ this.ctx.present([new Spacer(1), new Text(output, 1, 0)]);
446
+ }
447
+
448
+ async handleChangelogCommand(showFull = false): Promise<void> {
449
+ const changelogPath = getChangelogPath();
450
+ const allEntries = await parseChangelog(changelogPath);
451
+ // Default to showing only the latest 3 versions unless --full is specified
452
+ // allEntries comes from parseChangelog with newest first, reverse to show oldest->newest
453
+ const entriesToShow = showFull ? allEntries : allEntries.slice(0, 3);
454
+ const changelogMarkdown =
455
+ entriesToShow.length > 0
456
+ ? [...entriesToShow]
457
+ .reverse()
458
+ .map(e => e.content)
459
+ .join("\n\n")
460
+ : "No changelog entries found.";
461
+ const title = showFull ? "Full Changelog" : "Recent Changes";
462
+ const hint = showFull
463
+ ? ""
464
+ : `\n\n${theme.fg("dim", "Use")} ${theme.bold("/changelog full")} ${theme.fg("dim", "to view the complete changelog.")}`;
465
+
466
+ const block = new TranscriptBlock();
467
+ block.addChild(new DynamicBorder());
468
+ block.addChild(new Text(theme.bold(theme.fg("accent", title)), 1, 0));
469
+ block.addChild(new Spacer(1));
470
+ block.addChild(new Markdown(changelogMarkdown + hint, 1, 1, getMarkdownTheme()));
471
+ block.addChild(new DynamicBorder());
472
+ this.ctx.present(block);
473
+ }
474
+
475
+ handleHotkeysCommand(): void {
476
+ const hotkeys = buildHotkeysMarkdown({ keybindings: this.ctx.keybindings });
477
+ showMarkdownPanel(this.ctx, "Keyboard Shortcuts", hotkeys);
478
+ }
479
+
480
+ handleToolsCommand(): void {
481
+ const tools = buildToolsMarkdown({ tools: this.ctx.session.agent.state.tools });
482
+ showMarkdownPanel(this.ctx, "Available Tools", tools);
483
+ }
484
+
485
+ handleContextCommand(): void {
486
+ const breakdown = computeContextBreakdown(this.ctx.session, { snapcompactSavings: true });
487
+ if (breakdown.contextWindow <= 0) {
488
+ this.ctx.showWarning("Context usage is unavailable: no model is selected for this session.");
489
+ return;
490
+ }
491
+ const output = renderContextUsage(breakdown, theme);
492
+ const block = new TranscriptBlock();
493
+ block.addChild(new DynamicBorder());
494
+ block.addChild(new Text(theme.bold(theme.fg("accent", "Context Usage")), 1, 0));
495
+ block.addChild(new Spacer(1));
496
+ block.addChild(new Text(output, 1, 0));
497
+ block.addChild(new DynamicBorder());
498
+ this.ctx.present(block);
499
+ }
500
+
501
+ async handleMemoryCommand(text: string): Promise<void> {
502
+ const argumentText = text.slice(7).trim();
503
+ const action = argumentText.split(/\s+/, 1)[0]?.toLowerCase() || "view";
504
+ const agentDir = this.ctx.settings.getAgentDir();
505
+ const backend = await resolveMemoryBackend(this.ctx.settings);
506
+
507
+ if (action === "view") {
508
+ const payload = await backend.buildDeveloperInstructions(agentDir, this.ctx.settings, this.ctx.session);
509
+ if (!payload) {
510
+ this.ctx.showWarning("Memory payload is empty (memory backend off, disabled, or no memory available).");
511
+ return;
512
+ }
513
+ const block = new TranscriptBlock();
514
+ block.addChild(new DynamicBorder());
515
+ block.addChild(new Text(theme.bold(theme.fg("accent", "Memory Injection Payload")), 1, 0));
516
+ block.addChild(new Spacer(1));
517
+ block.addChild(new Markdown(payload, 1, 1, getMarkdownTheme()));
518
+ block.addChild(new DynamicBorder());
519
+ this.ctx.present(block);
520
+ return;
521
+ }
522
+
523
+ if (action === "reset" || action === "clear") {
524
+ try {
525
+ await backend.clear(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
526
+ await this.ctx.session.refreshBaseSystemPrompt();
527
+ this.ctx.showStatus("Memory data cleared and system prompt refreshed.");
528
+ } catch (error) {
529
+ this.ctx.showError(`Memory clear failed: ${error instanceof Error ? error.message : String(error)}`);
530
+ }
531
+ return;
532
+ }
533
+
534
+ if (action === "enqueue" || action === "rebuild") {
535
+ try {
536
+ await backend.enqueue(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
537
+ this.ctx.showStatus("Memory consolidation enqueued.");
538
+ } catch (error) {
539
+ this.ctx.showError(`Memory enqueue failed: ${error instanceof Error ? error.message : String(error)}`);
540
+ }
541
+ return;
542
+ }
543
+
544
+ if (action === "stats" || action === "diagnose") {
545
+ const hook = action === "stats" ? backend.stats : backend.diagnose;
546
+ try {
547
+ const payload = await hook?.(agentDir, this.ctx.sessionManager.getCwd(), this.ctx.session);
548
+ if (!payload) {
549
+ this.ctx.showWarning(`Memory ${action} is not available for the ${backend.id} backend.`);
550
+ return;
551
+ }
552
+ showMarkdownPanel(this.ctx, `Memory ${action === "stats" ? "Stats" : "Diagnostics"}`, payload);
553
+ } catch (error) {
554
+ this.ctx.showError(`Memory ${action} failed: ${error instanceof Error ? error.message : String(error)}`);
555
+ }
556
+ return;
557
+ }
558
+
559
+ if (action === "mm") {
560
+ await this.#handleMentalModelsSubcommand(argumentText);
561
+ return;
562
+ }
563
+
564
+ this.ctx.showError("Usage: /memory <view|stats|diagnose|clear|reset|enqueue|rebuild|mm ...>");
565
+ }
566
+
567
+ async #handleMentalModelsSubcommand(argumentText: string): Promise<void> {
568
+ // Parse: "mm <verb> [arg]"
569
+ const parts = argumentText.split(/\s+/).slice(1);
570
+ const verb = parts[0]?.toLowerCase() ?? "list";
571
+ const arg = parts[1];
572
+
573
+ const state = this.ctx.session.getHindsightSessionState();
574
+ const primary = state && !state.aliasOf ? state : undefined;
575
+ if (!primary) {
576
+ this.ctx.showError("Hindsight backend is not active for this session.");
577
+ return;
578
+ }
579
+ if (!primary.config.mentalModelsEnabled) {
580
+ this.ctx.showError("Mental models are disabled (hindsight.mentalModelsEnabled = false).");
581
+ return;
582
+ }
583
+
584
+ switch (verb) {
585
+ case "list":
586
+ await this.#mmList(primary);
587
+ return;
588
+ case "show":
589
+ if (!arg) return this.ctx.showError("Usage: /memory mm show <id>");
590
+ await this.#mmShow(primary, arg);
591
+ return;
592
+ case "refresh":
593
+ await this.#mmRefresh(primary, arg);
594
+ return;
595
+ case "history":
596
+ if (!arg) return this.ctx.showError("Usage: /memory mm history <id>");
597
+ await this.#mmHistory(primary, arg);
598
+ return;
599
+ case "seed":
600
+ await this.#mmSeed(primary);
601
+ return;
602
+ case "reload":
603
+ await this.#mmReload(primary);
604
+ return;
605
+ case "delete":
606
+ case "remove":
607
+ if (!arg) return this.ctx.showError("Usage: /memory mm delete <id>");
608
+ await this.#mmDelete(primary, arg);
609
+ return;
610
+ default:
611
+ this.ctx.showError("Usage: /memory mm <list|show|refresh|history|seed|reload|delete>");
612
+ }
613
+ }
614
+
615
+ async #mmList(state: HindsightSessionState): Promise<void> {
616
+ const client: HindsightApi = state.client;
617
+ try {
618
+ const response = await client.listMentalModels(state.bankId, { detail: "metadata" });
619
+ const items = response.items ?? [];
620
+ if (items.length === 0) {
621
+ this.ctx.showStatus(`No mental models on bank ${state.bankId}.`);
622
+ return;
623
+ }
624
+ const lines = items
625
+ .slice()
626
+ .sort((a, b) => a.id.localeCompare(b.id))
627
+ .map(summarizeMentalModel);
628
+ showMarkdownPanel(this.ctx, `Mental Models — ${state.bankId}`, lines.join("\n"));
629
+ } catch (error) {
630
+ this.ctx.showError(`mm list failed: ${error instanceof Error ? error.message : String(error)}`);
631
+ }
632
+ }
633
+
634
+ async #mmShow(state: HindsightSessionState, id: string): Promise<void> {
635
+ try {
636
+ const model = await state.client.getMentalModel(state.bankId, id, { detail: "content" });
637
+ if (!model) {
638
+ this.ctx.showError(`Mental model not found: ${id}`);
639
+ return;
640
+ }
641
+ const tags = model.tags && model.tags.length > 0 ? `\n_tags: ${model.tags.join(", ")}_` : "";
642
+ const refreshed = model.last_refreshed_at ? `\n_last refreshed: ${model.last_refreshed_at}_` : "";
643
+ const sourceQuery = model.source_query ? `\n\n**Source query:** ${model.source_query}` : "";
644
+ const content = (model.content ?? "_(empty — background reflect may still be running)_").trim();
645
+ showMarkdownPanel(
646
+ this.ctx,
647
+ model.name,
648
+ `**id:** \`${model.id}\`${tags}${refreshed}${sourceQuery}\n\n${content}`,
649
+ );
650
+ } catch (error) {
651
+ this.ctx.showError(`mm show failed: ${error instanceof Error ? error.message : String(error)}`);
652
+ }
653
+ }
654
+
655
+ async #mmRefresh(state: HindsightSessionState, id: string | undefined): Promise<void> {
656
+ try {
657
+ if (id) {
658
+ // Single-model refresh is explicit operator intent: bypass the
659
+ // auto-refresh filter so curated/manual models can still be
660
+ // refreshed on demand.
661
+ await state.client.refreshMentalModel(state.bankId, id);
662
+ this.ctx.showStatus(`Refresh queued for mental model ${id}.`);
663
+ } else {
664
+ // Bulk refresh: only touch models that opted into automatic
665
+ // refresh via `trigger.refresh_after_consolidation`. Curated
666
+ // models are reviewed before publishing and must not be
667
+ // silently regenerated by a bank-wide refresh sweep. Reading
668
+ // `detail: "content"` here is required because the trigger
669
+ // field is excluded from `detail: "metadata"`.
670
+ const list = await state.client.listMentalModels(state.bankId, { detail: "content" });
671
+ const items = list.items ?? [];
672
+ if (items.length === 0) {
673
+ this.ctx.showStatus(`No mental models on bank ${state.bankId}.`);
674
+ return;
675
+ }
676
+ const targets = items.filter(m => m.trigger?.refresh_after_consolidation === true);
677
+ const skipped = items.length - targets.length;
678
+ if (targets.length === 0) {
679
+ this.ctx.showStatus(
680
+ `No mental models opted into auto-refresh; ${skipped} curated model(s) left untouched. Pass an explicit id to refresh one of them.`,
681
+ );
682
+ return;
683
+ }
684
+ let queued = 0;
685
+ for (const item of targets) {
686
+ try {
687
+ await state.client.refreshMentalModel(state.bankId, item.id);
688
+ queued++;
689
+ } catch (error) {
690
+ this.ctx.showWarning(
691
+ `Refresh failed for ${item.id}: ${error instanceof Error ? error.message : String(error)}`,
692
+ );
693
+ }
694
+ }
695
+ const skippedSuffix = skipped > 0 ? `; skipped ${skipped} curated model(s)` : "";
696
+ this.ctx.showStatus(
697
+ `Refresh queued for ${queued}/${targets.length} auto-refresh model(s)${skippedSuffix}.`,
698
+ );
699
+ }
700
+ // Reload the cache after a brief grace so the new content (if the refresh
701
+ // completes synchronously on the server) flows into the system prompt.
702
+ await Bun.sleep(500);
703
+ await reloadMentalModelsForSession(state.session);
704
+ } catch (error) {
705
+ this.ctx.showError(`mm refresh failed: ${error instanceof Error ? error.message : String(error)}`);
706
+ }
707
+ }
708
+
709
+ async #mmHistory(state: HindsightSessionState, id: string): Promise<void> {
710
+ try {
711
+ const [model, history] = await Promise.all([
712
+ state.client.getMentalModel(state.bankId, id, { detail: "content" }),
713
+ state.client.getMentalModelHistory(state.bankId, id),
714
+ ]);
715
+ if (!model) {
716
+ this.ctx.showError(`Mental model not found: ${id}`);
717
+ return;
718
+ }
719
+ if (history.length === 0) {
720
+ this.ctx.showStatus(`No history recorded for ${id}.`);
721
+ return;
722
+ }
723
+ // History is most-recent first. Each entry stores the content BEFORE that
724
+ // change. To diff "what changed at entry N", compare entry N's
725
+ // previous_content (= state before that change) with entry N-1's
726
+ // previous_content (= state after that change, which was state before
727
+ // the next change). For the most recent change, compare against the
728
+ // model's CURRENT content.
729
+ const sections: string[] = [];
730
+ for (let i = 0; i < history.length; i++) {
731
+ const before = history[i].previous_content ?? "";
732
+ const after = i === 0 ? (model.content ?? "") : (history[i - 1].previous_content ?? "");
733
+ const diff = diffMentalModelContent(before, after);
734
+ sections.push(`### ${history[i].changed_at}\n\n\`\`\`diff\n${diff}\n\`\`\``);
735
+ }
736
+ showMarkdownPanel(this.ctx, `History — ${model.name}`, sections.join("\n\n"));
737
+ } catch (error) {
738
+ this.ctx.showError(`mm history failed: ${error instanceof Error ? error.message : String(error)}`);
739
+ }
740
+ }
741
+
742
+ async #mmSeed(state: HindsightSessionState): Promise<void> {
743
+ try {
744
+ const config = loadHindsightConfig(this.ctx.settings);
745
+ const seeds = resolveSeedsForScope(
746
+ {
747
+ bankId: state.bankId,
748
+ retainTags: state.retainTags,
749
+ recallTags: state.recallTags,
750
+ recallTagsMatch: state.recallTagsMatch,
751
+ },
752
+ config.scoping,
753
+ );
754
+ if (seeds.length === 0) {
755
+ this.ctx.showStatus(`No built-in seeds apply to scoping=${config.scoping}.`);
756
+ return;
757
+ }
758
+ const list = await state.client.listMentalModels(state.bankId, { detail: "metadata" });
759
+ const existing = list.items ?? [];
760
+ let created = 0;
761
+ let skipped = 0;
762
+ for (const seed of seeds) {
763
+ if (seedAlreadyExists(seed, existing)) {
764
+ skipped++;
765
+ continue;
766
+ }
767
+ try {
768
+ await state.client.createMentalModel(state.bankId, seed.name, seed.sourceQuery, {
769
+ id: seed.id,
770
+ tags: seed.tags.length > 0 ? seed.tags : undefined,
771
+ maxTokens: seed.maxTokens,
772
+ trigger: seed.trigger,
773
+ });
774
+ created++;
775
+ } catch (error) {
776
+ this.ctx.showWarning(
777
+ `Seed failed for ${seed.id}: ${error instanceof Error ? error.message : String(error)}`,
778
+ );
779
+ }
780
+ }
781
+ this.ctx.showStatus(`Seeded ${created} new mental model(s); ${skipped} already present.`);
782
+ } catch (error) {
783
+ this.ctx.showError(`mm seed failed: ${error instanceof Error ? error.message : String(error)}`);
784
+ }
785
+ }
786
+
787
+ async #mmReload(state: HindsightSessionState): Promise<void> {
788
+ const ok = await reloadMentalModelsForSession(state.session);
789
+ if (ok) {
790
+ this.ctx.showStatus("Mental-model cache reloaded.");
791
+ } else {
792
+ this.ctx.showError("Reload failed (Hindsight backend not active or mental models disabled).");
793
+ }
794
+ }
795
+
796
+ async #mmDelete(state: HindsightSessionState, id: string): Promise<void> {
797
+ try {
798
+ const removed = await state.client.deleteMentalModel(state.bankId, id);
799
+ if (!removed) {
800
+ this.ctx.showError(`Mental model not found: ${id}`);
801
+ return;
802
+ }
803
+ // Drop the cached snippet so the closing tag does not silently keep
804
+ // stale content in the system prompt until the next agent_end TTL.
805
+ await reloadMentalModelsForSession(state.session);
806
+ this.ctx.showStatus(`Deleted mental model ${id} from bank ${state.bankId}.`);
807
+ } catch (error) {
808
+ this.ctx.showError(`mm delete failed: ${error instanceof Error ? error.message : String(error)}`);
809
+ }
810
+ }
811
+
812
+ async #runNewSessionFlow(options?: NewSessionOptions, label: string = "New session started"): Promise<void> {
813
+ if (this.ctx.loadingAnimation) {
814
+ this.ctx.loadingAnimation.stop();
815
+ this.ctx.loadingAnimation = undefined;
816
+ }
817
+ this.ctx.statusContainer.clear();
818
+
819
+ if (this.ctx.session.isCompacting) {
820
+ this.ctx.session.abortCompaction();
821
+ while (this.ctx.session.isCompacting) {
822
+ await Bun.sleep(10);
823
+ }
824
+ }
825
+ if (!(await this.ctx.session.newSession(options))) return;
826
+ this.ctx.resetObserverRegistry();
827
+ setSessionTerminalTitle(this.ctx.sessionManager.getSessionName(), this.ctx.sessionManager.getCwd());
828
+
829
+ this.ctx.statusLine.invalidate();
830
+ this.ctx.statusLine.setSessionStartTime(Date.now());
831
+ this.ctx.updateEditorTopBorder();
832
+ this.ctx.updateEditorBorderColor();
833
+ this.ctx.chatContainer.clear();
834
+ this.ctx.pendingMessagesContainer.clear();
835
+ this.ctx.compactionQueuedMessages = [];
836
+ this.ctx.streamingComponent = undefined;
837
+ this.ctx.streamingMessage = undefined;
838
+ this.ctx.pendingTools.clear();
839
+
840
+ this.ctx.present([new Spacer(1), new Text(`${theme.fg("accent", `${theme.status.success} ${label}`)}`, 1, 1)]);
841
+ await this.ctx.reloadTodos();
842
+ this.ctx.ui.requestRender(true, { clearScrollback: true });
843
+ }
844
+
845
+ async handleClearCommand(): Promise<void> {
846
+ await this.#runNewSessionFlow();
847
+ }
848
+
849
+ async handleFreshCommand(): Promise<void> {
850
+ const result = this.ctx.session.freshSession();
851
+ if (!result) {
852
+ this.ctx.showWarning("Wait for the current response to finish or abort it before refreshing provider state.");
853
+ return;
854
+ }
855
+ const stateLabel = result.closedProviderSessions === 1 ? "provider state" : "provider states";
856
+ this.ctx.statusLine.invalidate();
857
+ this.ctx.updateEditorTopBorder();
858
+ this.ctx.showStatus(`Fresh provider session started (${result.closedProviderSessions} ${stateLabel} pruned).`);
859
+ }
860
+
861
+ async handleDropCommand(): Promise<void> {
862
+ if (!this.ctx.sessionManager.getSessionFile()) {
863
+ this.ctx.showError("Nothing to drop (in-memory session)");
864
+ return;
865
+ }
866
+ await this.#runNewSessionFlow({ drop: true }, "Session dropped");
867
+ }
868
+
869
+ async handleForkCommand(): Promise<void> {
870
+ if (this.ctx.session.isStreaming) {
871
+ this.ctx.showWarning("Wait for the current response to finish or abort it before forking.");
872
+ return;
873
+ }
874
+ if (this.ctx.loadingAnimation) {
875
+ this.ctx.loadingAnimation.stop();
876
+ this.ctx.loadingAnimation = undefined;
877
+ }
878
+ this.ctx.statusContainer.clear();
879
+
880
+ const success = await this.ctx.session.fork();
881
+ if (!success) {
882
+ this.ctx.showError("Fork failed (session not persisted or cancelled)");
883
+ return;
884
+ }
885
+
886
+ this.ctx.statusLine.invalidate();
887
+ this.ctx.updateEditorTopBorder();
888
+
889
+ const sessionFile = this.ctx.session.sessionFile;
890
+ const shortPath = sessionFile ? sessionFile.split("/").pop() : "new session";
891
+ this.ctx.present([
892
+ new Spacer(1),
893
+ new Text(`${theme.fg("accent", `${theme.status.success} Session forked to ${shortPath}`)}`, 1, 1),
894
+ ]);
895
+ }
896
+
897
+ async handleMoveCommand(targetPath: string): Promise<void> {
898
+ if (this.ctx.session.isStreaming) {
899
+ this.ctx.showWarning("Wait for the current response to finish or abort it before moving.");
900
+ return;
901
+ }
902
+
903
+ const unquoted = stripOuterDoubleQuotes(targetPath);
904
+ if (!unquoted) {
905
+ this.ctx.showError("Usage: /move <path>");
906
+ return;
907
+ }
908
+
909
+ const cwd = this.ctx.sessionManager.getCwd();
910
+ const resolvedPath = resolveToCwd(unquoted, cwd);
911
+
912
+ try {
913
+ const stat = await fs.stat(resolvedPath);
914
+ if (!stat.isDirectory()) {
915
+ this.ctx.showError(`Not a directory: ${resolvedPath}`);
916
+ return;
917
+ }
918
+ } catch {
919
+ this.ctx.showError(`Directory does not exist: ${resolvedPath}`);
920
+ return;
921
+ }
922
+
923
+ try {
924
+ await this.ctx.sessionManager.flush();
925
+ await this.ctx.sessionManager.moveTo(resolvedPath);
926
+ await this.ctx.applyCwdChange(resolvedPath);
927
+
928
+ this.ctx.present([
929
+ new Spacer(1),
930
+ new Text(`${theme.fg("accent", `${theme.status.success} Session moved to ${resolvedPath}`)}`, 1, 1),
931
+ ]);
932
+ } catch (err) {
933
+ this.ctx.showError(`Move failed: ${err instanceof Error ? err.message : String(err)}`);
934
+ }
935
+ }
936
+
937
+ async handleRenameCommand(title: string): Promise<void> {
938
+ try {
939
+ const stored = await this.ctx.sessionManager.setSessionName(title, "user");
940
+ if (!stored) {
941
+ this.ctx.showError("Session name cannot be empty.");
942
+ return;
943
+ }
944
+ const name = this.ctx.sessionManager.getSessionName()!;
945
+ setSessionTerminalTitle(name, this.ctx.sessionManager.getCwd());
946
+ this.ctx.statusLine.invalidate();
947
+ this.ctx.updateEditorBorderColor();
948
+ this.ctx.showStatus(`Session renamed to "${name}".`);
949
+ } catch (err) {
950
+ this.ctx.showError(`Rename failed: ${err instanceof Error ? err.message : String(err)}`);
951
+ }
952
+ }
953
+
954
+ async handleBashCommand(command: string, excludeFromContext = false): Promise<void> {
955
+ const isDeferred = this.ctx.session.isStreaming;
956
+ this.ctx.bashComponent = new BashExecutionComponent(command, this.ctx.ui, excludeFromContext);
957
+
958
+ if (isDeferred) {
959
+ this.ctx.pendingMessagesContainer.addChild(this.ctx.bashComponent);
960
+ this.ctx.pendingBashComponents.push(this.ctx.bashComponent);
961
+ } else {
962
+ this.ctx.present(this.ctx.bashComponent);
963
+ }
964
+ this.ctx.ui.requestRender();
965
+
966
+ try {
967
+ const result = await this.ctx.session.executeBash(
968
+ command,
969
+ chunk => {
970
+ if (this.ctx.bashComponent) {
971
+ this.ctx.bashComponent.appendOutput(chunk);
972
+ }
973
+ },
974
+ { excludeFromContext, useUserShell: true },
975
+ );
976
+
977
+ if (this.ctx.bashComponent) {
978
+ const meta = outputMeta().truncationFromSummary(result, { direction: "tail" }).get();
979
+ this.ctx.bashComponent.setComplete(result.exitCode, result.cancelled, {
980
+ output: result.output,
981
+ truncation: meta?.truncation,
982
+ });
983
+ }
984
+ } catch (error) {
985
+ if (this.ctx.bashComponent) {
986
+ this.ctx.bashComponent.setComplete(undefined, false);
987
+ }
988
+ this.ctx.showError(`Bash command failed: ${error instanceof Error ? error.message : "Unknown error"}`);
989
+ }
990
+
991
+ this.ctx.bashComponent = undefined;
992
+ this.ctx.ui.requestRender();
993
+ }
994
+
995
+ async handlePythonCommand(code: string, excludeFromContext = false): Promise<void> {
996
+ const isDeferred = this.ctx.session.isStreaming;
997
+ this.ctx.pythonComponent = new EvalExecutionComponent(code, this.ctx.ui, excludeFromContext);
998
+
999
+ if (isDeferred) {
1000
+ this.ctx.pendingMessagesContainer.addChild(this.ctx.pythonComponent);
1001
+ this.ctx.pendingPythonComponents.push(this.ctx.pythonComponent);
1002
+ } else {
1003
+ this.ctx.present(this.ctx.pythonComponent);
1004
+ }
1005
+ this.ctx.ui.requestRender();
1006
+
1007
+ try {
1008
+ const result = await this.ctx.session.executePython(
1009
+ code,
1010
+ chunk => {
1011
+ if (this.ctx.pythonComponent) {
1012
+ this.ctx.pythonComponent.appendOutput(chunk);
1013
+ }
1014
+ },
1015
+ { excludeFromContext },
1016
+ );
1017
+
1018
+ if (this.ctx.pythonComponent) {
1019
+ const meta = outputMeta().truncationFromSummary(result, { direction: "tail" }).get();
1020
+ this.ctx.pythonComponent.setComplete(result.exitCode, result.cancelled, {
1021
+ output: result.output,
1022
+ truncation: meta?.truncation,
1023
+ });
1024
+ }
1025
+ } catch (error) {
1026
+ if (this.ctx.pythonComponent) {
1027
+ this.ctx.pythonComponent.setComplete(undefined, false);
1028
+ }
1029
+ this.ctx.showError(`Python execution failed: ${error instanceof Error ? error.message : "Unknown error"}`);
1030
+ }
1031
+
1032
+ this.ctx.pythonComponent = undefined;
1033
+ this.ctx.ui.requestRender();
1034
+ }
1035
+
1036
+ async handleCompactCommand(
1037
+ customInstructions?: string,
1038
+ mode?: CompactMode,
1039
+ beforeFlush?: (outcome: CompactionOutcome) => void | Promise<void>,
1040
+ ): Promise<CompactionOutcome> {
1041
+ const entries = this.ctx.sessionManager.getEntries();
1042
+ const messageCount = entries.filter(e => e.type === "message").length;
1043
+
1044
+ if (messageCount < 2) {
1045
+ this.ctx.showWarning("Nothing to compact (no messages yet)");
1046
+ return "ok";
1047
+ }
1048
+
1049
+ return this.executeCompaction(customInstructions, false, beforeFlush, mode);
1050
+ }
1051
+
1052
+ /**
1053
+ * TUI handler for `/shake`. `elide` drops heavy structural content and
1054
+ * `images` strips image blocks. Rebuilds the chat and reports counts.
1055
+ */
1056
+ async handleShakeCommand(mode: ShakeMode): Promise<void> {
1057
+ let result: ShakeResult;
1058
+ try {
1059
+ result = await this.ctx.session.shake(mode);
1060
+ } catch (error) {
1061
+ this.ctx.showError(`Shake failed: ${error instanceof Error ? error.message : String(error)}`);
1062
+ return;
1063
+ }
1064
+
1065
+ const dropped = result.toolResultsDropped + result.blocksDropped + (result.imagesDropped ?? 0);
1066
+ if (dropped === 0) {
1067
+ this.ctx.showStatus("Nothing to shake.");
1068
+ return;
1069
+ }
1070
+ this.ctx.rebuildChatFromMessages();
1071
+ this.ctx.statusLine.invalidate();
1072
+ this.ctx.updateEditorTopBorder();
1073
+ this.ctx.showStatus(formatShakeSummary(result));
1074
+ }
1075
+
1076
+ async handleSkillCommand(skillPath: string, args: string): Promise<void> {
1077
+ try {
1078
+ const content = await Bun.file(skillPath).text();
1079
+ const body = content.replace(/^---\n[\s\S]*?\n---\n/, "").trim();
1080
+ const metaLines = [`Skill: ${skillPath}`];
1081
+ if (args) {
1082
+ metaLines.push(`User: ${args}`);
1083
+ }
1084
+ const message = `${body}\n\n---\n\n${metaLines.join("\n")}`;
1085
+ await this.ctx.session.prompt(message);
1086
+ } catch (err) {
1087
+ this.ctx.showError(`Failed to load skill: ${err instanceof Error ? err.message : String(err)}`);
1088
+ }
1089
+ }
1090
+
1091
+ async executeCompaction(
1092
+ customInstructionsOrOptions?: string | CompactOptions,
1093
+ isAuto = false,
1094
+ beforeFlush?: (outcome: CompactionOutcome) => void | Promise<void>,
1095
+ mode?: CompactMode,
1096
+ ): Promise<CompactionOutcome> {
1097
+ if (this.ctx.loadingAnimation) {
1098
+ this.ctx.loadingAnimation.stop();
1099
+ this.ctx.loadingAnimation = undefined;
1100
+ }
1101
+ this.ctx.statusContainer.clear();
1102
+
1103
+ const label = isAuto ? "Auto-compacting context... (esc to cancel)" : "Compacting context... (esc to cancel)";
1104
+ const compactingLoader = new Loader(
1105
+ this.ctx.ui,
1106
+ spinner => theme.fg("accent", spinner),
1107
+ text => theme.fg("muted", text),
1108
+ label,
1109
+ getSymbolTheme().spinnerFrames,
1110
+ );
1111
+ this.ctx.statusContainer.addChild(compactingLoader);
1112
+ this.ctx.ui.requestRender();
1113
+
1114
+ let outcome: CompactionOutcome = "ok";
1115
+ try {
1116
+ const instructions = typeof customInstructionsOrOptions === "string" ? customInstructionsOrOptions : undefined;
1117
+ const baseOptions =
1118
+ customInstructionsOrOptions && typeof customInstructionsOrOptions === "object"
1119
+ ? customInstructionsOrOptions
1120
+ : undefined;
1121
+ // The slash path passes `mode` positionally; the extension path carries
1122
+ // it inside the options object. Either source wins over no mode.
1123
+ const effectiveMode = mode ?? baseOptions?.mode;
1124
+ const options =
1125
+ baseOptions || effectiveMode
1126
+ ? { ...baseOptions, ...(effectiveMode ? { mode: effectiveMode } : {}) }
1127
+ : undefined;
1128
+ await this.ctx.session.compact(instructions, options);
1129
+
1130
+ compactingLoader.stop();
1131
+ this.ctx.statusContainer.clear();
1132
+ this.ctx.rebuildChatFromMessages();
1133
+
1134
+ this.ctx.statusLine.invalidate();
1135
+ this.ctx.updateEditorTopBorder();
1136
+ } catch (error) {
1137
+ if (error instanceof CompactionCancelledError) {
1138
+ outcome = "cancelled";
1139
+ this.ctx.showError("Compaction cancelled");
1140
+ } else {
1141
+ outcome = "failed";
1142
+ const message = error instanceof Error ? error.message : String(error);
1143
+ this.ctx.showError(`Compaction failed: ${message}`);
1144
+ }
1145
+ } finally {
1146
+ compactingLoader.stop();
1147
+ this.ctx.statusContainer.clear();
1148
+ }
1149
+ // Run the caller's pre-flush hook (e.g. the plan-approval model transition)
1150
+ // before queued user input is dispatched, so any turn queued during
1151
+ // compaction executes on the post-compaction model rather than the model
1152
+ // compaction itself ran on.
1153
+ if (beforeFlush) await beforeFlush(outcome);
1154
+ await this.ctx.flushCompactionQueue({ willRetry: false });
1155
+ return outcome;
1156
+ }
1157
+
1158
+ async handleHandoffCommand(customInstructions?: string): Promise<void> {
1159
+ const entries = this.ctx.sessionManager.getEntries();
1160
+ const messageCount = entries.filter(e => e.type === "message").length;
1161
+
1162
+ if (messageCount < 2) {
1163
+ this.ctx.showWarning("Nothing to hand off (no messages yet)");
1164
+ return;
1165
+ }
1166
+
1167
+ if (this.ctx.loadingAnimation) {
1168
+ this.ctx.loadingAnimation.stop();
1169
+ this.ctx.loadingAnimation = undefined;
1170
+ }
1171
+ this.ctx.statusContainer.clear();
1172
+
1173
+ const handoffLoader = new Loader(
1174
+ this.ctx.ui,
1175
+ spinner => theme.fg("accent", spinner),
1176
+ text => theme.fg("muted", text),
1177
+ "Generating handoff… (esc to cancel)",
1178
+ getSymbolTheme().spinnerFrames,
1179
+ );
1180
+ this.ctx.statusContainer.addChild(handoffLoader);
1181
+ this.ctx.ui.requestRender();
1182
+
1183
+ try {
1184
+ // Handoff generation runs as a oneshot request; the new session is shown after it completes.
1185
+ const result = await this.ctx.session.handoff(customInstructions);
1186
+
1187
+ if (!result) {
1188
+ this.ctx.showError("Handoff cancelled");
1189
+ return;
1190
+ }
1191
+
1192
+ // Rebuild chat from the new session (which now contains the handoff document)
1193
+ this.ctx.rebuildChatFromMessages();
1194
+
1195
+ this.ctx.statusLine.invalidate();
1196
+ this.ctx.updateEditorTopBorder();
1197
+ this.ctx.updateEditorBorderColor();
1198
+ await this.ctx.reloadTodos();
1199
+
1200
+ this.ctx.present([
1201
+ new Spacer(1),
1202
+ new Text(`${theme.fg("accent", `${theme.status.success} New session started with handoff context`)}`, 1, 1),
1203
+ ]);
1204
+ if (result.savedPath) {
1205
+ this.ctx.showStatus(`Handoff document saved to: ${result.savedPath}`);
1206
+ }
1207
+ } catch (error) {
1208
+ const message = error instanceof Error ? error.message : String(error);
1209
+ if (message === "Handoff cancelled" || (error instanceof Error && error.name === "AbortError")) {
1210
+ this.ctx.showError("Handoff cancelled");
1211
+ } else {
1212
+ this.ctx.showError(`Handoff failed: ${message}`);
1213
+ }
1214
+ } finally {
1215
+ handoffLoader.stop();
1216
+ this.ctx.statusContainer.clear();
1217
+ }
1218
+ this.ctx.ui.requestRender();
1219
+ }
1220
+ }
1221
+
1222
+ const BAR_WIDTH_MAX = 24;
1223
+ const BAR_WIDTH_MIN = 4;
1224
+
1225
+ function renderJobLine(job: AsyncJobSnapshotItem, now: number): string {
1226
+ const duration = formatDuration(Math.max(0, now - job.startTime));
1227
+ const status = formatJobStatus(job.status);
1228
+ return `${theme.fg("dim", job.id)} ${theme.fg("dim", `[${job.type}]`)} ${status} ${theme.fg("dim", `(${duration})`)}`;
1229
+ }
1230
+
1231
+ function formatJobStatus(status: AsyncJobSnapshotItem["status"]): string {
1232
+ if (status === "running") return theme.fg("warning", "running");
1233
+ if (status === "completed") return theme.fg("success", "completed");
1234
+ if (status === "cancelled") return theme.fg("dim", "cancelled");
1235
+ return theme.fg("error", "failed");
1236
+ }
1237
+
1238
+ function truncateJobLabel(label: string, maxWidth: number): string {
1239
+ if (visibleWidth(label) <= maxWidth) return label;
1240
+ if (maxWidth <= 1) return "…";
1241
+
1242
+ let out = "";
1243
+ for (const char of label) {
1244
+ const next = `${out}${char}`;
1245
+ if (visibleWidth(`${next}…`) > maxWidth) break;
1246
+ out = next;
1247
+ }
1248
+
1249
+ return `${out}…`;
1250
+ }
1251
+
1252
+ function formatProviderName(provider: string): string {
1253
+ return provider
1254
+ .split(/[-_]/g)
1255
+ .map(part => (part ? part[0].toUpperCase() + part.slice(1) : ""))
1256
+ .join(" ");
1257
+ }
1258
+
1259
+ function formatNumber(value: number, maxFractionDigits = 1): string {
1260
+ return new Intl.NumberFormat("en-US", { maximumFractionDigits: maxFractionDigits }).format(value);
1261
+ }
1262
+
1263
+ function resolveProviderAuthMode(authStorage: AuthStorage, provider: string): string {
1264
+ if (authStorage.hasOAuth(provider)) {
1265
+ return "oauth";
1266
+ }
1267
+ if (authStorage.has(provider)) {
1268
+ return "api key";
1269
+ }
1270
+ if (getEnvApiKey(provider)) {
1271
+ return "env api key";
1272
+ }
1273
+ if (authStorage.hasAuth(provider)) {
1274
+ return "runtime/fallback";
1275
+ }
1276
+ return "unknown";
1277
+ }
1278
+
1279
+ export function renderProviderSection(details: ProviderDetails, uiTheme: Pick<typeof theme, "fg">): string {
1280
+ const lines: string[] = [];
1281
+ lines.push(`${uiTheme.fg("dim", "Name:")} ${details.provider}`);
1282
+ for (const field of details.fields) {
1283
+ lines.push(`${uiTheme.fg("dim", `${field.label}:`)} ${field.value}`);
1284
+ }
1285
+ return `${lines.join("\n")}\n`;
1286
+ }
1287
+
1288
+ function resolveFraction(limit: UsageLimit): number | undefined {
1289
+ const amount = limit.amount;
1290
+ if (amount.usedFraction !== undefined) return amount.usedFraction;
1291
+ if (amount.used !== undefined && amount.limit !== undefined && amount.limit > 0) {
1292
+ return amount.used / amount.limit;
1293
+ }
1294
+ if (amount.unit === "percent" && amount.used !== undefined) {
1295
+ return amount.used / 100;
1296
+ }
1297
+ return undefined;
1298
+ }
1299
+
1300
+ function resolveProviderUsageTotal(reports: UsageReport[]): number {
1301
+ return reports
1302
+ .flatMap(report => report.limits)
1303
+ .map(limit => resolveFraction(limit) ?? 0)
1304
+ .reduce((sum, value) => sum + value, 0);
1305
+ }
1306
+
1307
+ function formatLimitTitle(limit: UsageLimit): string {
1308
+ const tier = limit.scope.tier;
1309
+ if (tier && !limit.label.toLowerCase().includes(tier.toLowerCase())) {
1310
+ return `${limit.label} (${tier})`;
1311
+ }
1312
+ return limit.label;
1313
+ }
1314
+
1315
+ function formatWindowSuffix(label: string, windowLabel: string, uiTheme: typeof theme): string {
1316
+ const normalizedLabel = label.toLowerCase();
1317
+ const normalizedWindow = windowLabel.toLowerCase();
1318
+ if (normalizedWindow === "quota window") return "";
1319
+ if (normalizedLabel.includes(normalizedWindow)) return "";
1320
+ return uiTheme.fg("dim", `(${windowLabel})`);
1321
+ }
1322
+
1323
+ function formatAccountLabel(limit: UsageLimit, report: UsageReport, index: number): string {
1324
+ const email = (report.metadata?.email as string | undefined) ?? limit.scope.accountId;
1325
+ if (email) return email;
1326
+ const accountId = (report.metadata?.accountId as string | undefined) ?? limit.scope.accountId;
1327
+ if (accountId) return accountId;
1328
+ const projectId = (report.metadata?.projectId as string | undefined) ?? limit.scope.projectId;
1329
+ if (projectId) return projectId;
1330
+ return `account ${index + 1}`;
1331
+ }
1332
+
1333
+ function formatUnlimitedReportLabel(report: UsageReport, index: number): string {
1334
+ const email = report.metadata?.email as string | undefined;
1335
+ if (email) return email;
1336
+ const accountId = report.metadata?.accountId as string | undefined;
1337
+ if (accountId) return accountId;
1338
+ const projectId = report.metadata?.projectId as string | undefined;
1339
+ if (projectId) return projectId;
1340
+ return `account ${index + 1}`;
1341
+ }
1342
+
1343
+ function formatResetShort(limit: UsageLimit, nowMs: number): string | undefined {
1344
+ const resetsAt = limit.window?.resetsAt;
1345
+ if (resetsAt === undefined) return undefined;
1346
+ // Codex returns the prior window's reset_at until a new request opens a fresh window —
1347
+ // rendering a negative delta is meaningless, so drop the suffix in that case.
1348
+ if (resetsAt <= nowMs) return undefined;
1349
+ return formatDuration(resetsAt - nowMs);
1350
+ }
1351
+
1352
+ function formatAccountHeaderRow(
1353
+ limits: UsageLimit[],
1354
+ reports: UsageReport[],
1355
+ nowMs: number,
1356
+ columnWidth: number,
1357
+ uiTheme: typeof theme,
1358
+ activeAccount?: OAuthAccountIdentity,
1359
+ ): string[] {
1360
+ const parts = limits.map((limit, index) => {
1361
+ const reset = formatResetShort(limit, nowMs);
1362
+ const report = reports[index];
1363
+ const active = report !== undefined && limitMatchesActiveAccount(report, limit, activeAccount);
1364
+ const label = formatAccountLabel(limit, report, index);
1365
+ return {
1366
+ label: active ? `● ${label}` : label,
1367
+ suffix: reset ? `(${reset})` : "",
1368
+ active,
1369
+ };
1370
+ });
1371
+ const maxSuffixWidth = parts.reduce((max, p) => Math.max(max, visibleWidth(p.suffix)), 0);
1372
+ const gap = maxSuffixWidth > 0 ? 1 : 0;
1373
+ const prefixBudget = columnWidth - maxSuffixWidth - gap;
1374
+
1375
+ // If suffix can't share the cell with at least `x…`, fall back to whole-label truncation.
1376
+ if (prefixBudget < 2) {
1377
+ return parts.map(p => {
1378
+ const full = p.suffix ? `${p.label} ${p.suffix}` : p.label;
1379
+ const cell = padColumn(truncateJobLabel(full, columnWidth), columnWidth);
1380
+ return p.active ? uiTheme.fg("accent", cell) : cell;
1381
+ });
1382
+ }
1383
+
1384
+ return parts.map(p => {
1385
+ const prefix = truncateJobLabel(p.label, prefixBudget);
1386
+ const prefixCell = prefix + " ".repeat(prefixBudget - visibleWidth(prefix));
1387
+ const styledPrefix = p.active ? uiTheme.fg("accent", prefixCell) : prefixCell;
1388
+ if (!p.suffix) return styledPrefix + " ".repeat(maxSuffixWidth + gap);
1389
+ const suffixPad = " ".repeat(maxSuffixWidth - visibleWidth(p.suffix));
1390
+ return `${styledPrefix} ${suffixPad}${uiTheme.fg("dim", p.suffix)}`;
1391
+ });
1392
+ }
1393
+
1394
+ function padColumn(text: string, width: number): string {
1395
+ const visible = visibleWidth(text);
1396
+ if (visible >= width) return text;
1397
+ return `${text}${padding(width - visible)}`;
1398
+ }
1399
+
1400
+ function resolveAggregateStatus(limits: UsageLimit[]): UsageLimit["status"] {
1401
+ const hasOk = limits.some(limit => limit.status === "ok");
1402
+ const hasWarning = limits.some(limit => limit.status === "warning");
1403
+ const hasExhausted = limits.some(limit => limit.status === "exhausted");
1404
+ if (!hasOk && !hasWarning && !hasExhausted) return "unknown";
1405
+ if (hasOk) {
1406
+ return hasWarning || hasExhausted ? "warning" : "ok";
1407
+ }
1408
+ if (hasWarning) return "warning";
1409
+ return "exhausted";
1410
+ }
1411
+
1412
+ function formatAggregateAmount(limits: UsageLimit[]): string {
1413
+ const fractions = limits
1414
+ .map(limit => resolveFraction(limit))
1415
+ .filter((value): value is number => value !== undefined);
1416
+ if (fractions.length === limits.length && fractions.length > 0) {
1417
+ const sum = fractions.reduce((total, value) => total + value, 0);
1418
+ const avgRemaining = Math.max(0, ((limits.length - sum) / limits.length) * 100);
1419
+ return `${formatNumber(avgRemaining)}% free`;
1420
+ }
1421
+
1422
+ const amounts = limits
1423
+ .map(limit => limit.amount)
1424
+ .filter(amount => amount.used !== undefined && amount.limit !== undefined && amount.limit > 0);
1425
+ if (amounts.length === limits.length && amounts.length > 0) {
1426
+ const totalUsed = amounts.reduce((sum, amount) => sum + (amount.used ?? 0), 0);
1427
+ const totalLimit = amounts.reduce((sum, amount) => sum + (amount.limit ?? 0), 0);
1428
+ const remainingPct = totalLimit > 0 ? Math.max(0, 100 - (totalUsed / totalLimit) * 100) : 0;
1429
+ return `${formatNumber(remainingPct)}% free`;
1430
+ }
1431
+
1432
+ // Count unique accounts from limit scopes — not limits.length.
1433
+ const uniqueAccountIds = new Set(
1434
+ limits.map(limit => limit.scope.accountId).filter((id): id is string => typeof id === "string" && id.length > 0),
1435
+ );
1436
+ if (uniqueAccountIds.size > 0) return `${uniqueAccountIds.size} ${uniqueAccountIds.size === 1 ? "acct" : "accts"}`;
1437
+ // No account IDs available — keep the pre-existing fallback so providers
1438
+ // that don't populate scope.accountId still show a summary.
1439
+ return `${limits.length} accts`;
1440
+ }
1441
+
1442
+ function resolveResetRange(limits: UsageLimit[], nowMs: number): string | null {
1443
+ const absolute = limits
1444
+ .map(limit => limit.window?.resetsAt)
1445
+ .filter((value): value is number => value !== undefined && Number.isFinite(value) && value > nowMs);
1446
+ if (absolute.length === 0) return null;
1447
+ const offsets = absolute.map(value => value - nowMs);
1448
+ const minReset = Math.min(...offsets);
1449
+ const maxReset = Math.max(...offsets);
1450
+ if (maxReset - minReset > 60_000) {
1451
+ return `resets in ${formatDuration(minReset)}–${formatDuration(maxReset)}`;
1452
+ }
1453
+ return `resets in ${formatDuration(minReset)}`;
1454
+ }
1455
+
1456
+ function resolveStatusIcon(status: UsageLimit["status"], uiTheme: typeof theme): string {
1457
+ if (status === "exhausted") return uiTheme.fg("error", uiTheme.status.error);
1458
+ if (status === "warning") return uiTheme.fg("warning", uiTheme.status.warning);
1459
+ if (status === "ok") return uiTheme.fg("success", uiTheme.status.success);
1460
+ return uiTheme.fg("dim", uiTheme.status.pending);
1461
+ }
1462
+
1463
+ function resolveStatusColor(status: UsageLimit["status"]): "success" | "warning" | "error" | "dim" {
1464
+ if (status === "exhausted") return "error";
1465
+ if (status === "warning") return "warning";
1466
+ if (status === "ok") return "success";
1467
+ return "dim";
1468
+ }
1469
+
1470
+ function renderUsageBar(limit: UsageLimit, uiTheme: typeof theme, barWidth: number): string {
1471
+ const fraction = resolveFraction(limit);
1472
+ if (fraction === undefined) {
1473
+ return uiTheme.fg("dim", "·".repeat(barWidth));
1474
+ }
1475
+ const clamped = Math.min(Math.max(fraction, 0), 1);
1476
+ const exact = clamped * barWidth;
1477
+ const fullCells = Math.floor(exact);
1478
+ const remainder = exact - fullCells;
1479
+ let partial = "";
1480
+ if (remainder >= 2 / 3) partial = "▓";
1481
+ else if (remainder >= 1 / 3) partial = "▒";
1482
+ const leading = "█".repeat(fullCells) + partial;
1483
+ const empty = "░".repeat(Math.max(0, barWidth - fullCells - (partial ? 1 : 0)));
1484
+ const color = resolveStatusColor(limit.status);
1485
+ return `${uiTheme.fg(color, leading)}${uiTheme.fg("dim", empty)}`;
1486
+ }
1487
+
1488
+ /**
1489
+ * Pick a per-column width so n bars + a trailing amount string fit in `available` columns.
1490
+ * Falls back to the minimum when the terminal is too narrow rather than wrapping.
1491
+ */
1492
+ function resolveColumnWidth(count: number, available: number, trailing: number): number {
1493
+ if (count <= 0) return BAR_WIDTH_MAX;
1494
+ const indent = 2;
1495
+ const gaps = count - 1;
1496
+ const spaceForBars = available - indent - gaps - (trailing > 0 ? trailing + 1 : 0);
1497
+ const ideal = Math.floor(spaceForBars / count);
1498
+ const min = BAR_WIDTH_MIN;
1499
+ const max = BAR_WIDTH_MAX;
1500
+ if (ideal < min) return min;
1501
+ if (ideal > max) return max;
1502
+ return ideal;
1503
+ }
1504
+
1505
+ function renderUsageReports(
1506
+ reports: UsageReport[],
1507
+ uiTheme: typeof theme,
1508
+ nowMs: number,
1509
+ availableWidth: number,
1510
+ resolveActiveAccount?: (provider: string) => OAuthAccountIdentity | undefined,
1511
+ ): string {
1512
+ const lines: string[] = [];
1513
+ const latestFetchedAt = Math.max(...reports.map(report => report.fetchedAt ?? 0));
1514
+ const headerSuffix = latestFetchedAt ? ` (${formatDuration(nowMs - latestFetchedAt)} ago)` : "";
1515
+ lines.push(uiTheme.bold(uiTheme.fg("accent", `Usage${headerSuffix}`)));
1516
+ const grouped = new Map<string, UsageReport[]>();
1517
+ for (const report of reports) {
1518
+ const list = grouped.get(report.provider) ?? [];
1519
+ list.push(report);
1520
+ grouped.set(report.provider, list);
1521
+ }
1522
+ const providerEntries = Array.from(grouped.entries())
1523
+ .map(([provider, providerReports]) => ({
1524
+ provider,
1525
+ providerReports,
1526
+ totalUsage: resolveProviderUsageTotal(providerReports),
1527
+ }))
1528
+ .sort((a, b) => {
1529
+ if (a.totalUsage !== b.totalUsage) return a.totalUsage - b.totalUsage;
1530
+ return a.provider.localeCompare(b.provider);
1531
+ });
1532
+
1533
+ for (const { provider, providerReports } of providerEntries) {
1534
+ lines.push("");
1535
+ const providerName = formatProviderName(provider);
1536
+ const activeAccount = resolveActiveAccount?.(provider);
1537
+
1538
+ const limitGroups = new Map<
1539
+ string,
1540
+ { label: string; windowLabel: string; limits: UsageLimit[]; reports: UsageReport[] }
1541
+ >();
1542
+ for (const report of providerReports) {
1543
+ for (const limit of report.limits) {
1544
+ const windowId = limit.window?.id ?? limit.scope.windowId ?? "default";
1545
+ const key = `${formatLimitTitle(limit)}|${windowId}`;
1546
+ const windowLabel = limit.window?.label ?? windowId;
1547
+ const entry = limitGroups.get(key) ?? {
1548
+ label: formatLimitTitle(limit),
1549
+ windowLabel,
1550
+ limits: [],
1551
+ reports: [],
1552
+ };
1553
+ entry.limits.push(limit);
1554
+ entry.reports.push(report);
1555
+ limitGroups.set(key, entry);
1556
+ }
1557
+ }
1558
+
1559
+ lines.push(uiTheme.bold(uiTheme.fg("accent", providerName)));
1560
+ const activeAccountLabel = activeAccount?.email ?? activeAccount?.accountId ?? activeAccount?.projectId;
1561
+ if (activeAccountLabel) {
1562
+ lines.push(` ${uiTheme.fg("accent", "in use by this session:")} ${activeAccountLabel}`);
1563
+ }
1564
+
1565
+ const resetAccountLines: string[] = [];
1566
+ for (const report of providerReports) {
1567
+ const count = report.resetCredits?.availableCount ?? 0;
1568
+ if (count <= 0) continue;
1569
+ const label =
1570
+ (report.metadata?.email as string | undefined) ??
1571
+ (report.metadata?.accountId as string | undefined) ??
1572
+ "account";
1573
+ const isActive =
1574
+ !!activeAccount &&
1575
+ ((!!activeAccount.accountId && activeAccount.accountId === report.metadata?.accountId) ||
1576
+ (!!activeAccount.email && activeAccount.email === report.metadata?.email));
1577
+ resetAccountLines.push(
1578
+ ` • ${label}: ${count} saved reset${count === 1 ? "" : "s"}${isActive ? " (active)" : ""}`,
1579
+ );
1580
+ }
1581
+ if (resetAccountLines.length > 0) {
1582
+ lines.push(
1583
+ ` ${uiTheme.fg("accent", "Saved rate-limit resets")} ${uiTheme.fg("dim", "(/usage reset to spend)")}`,
1584
+ );
1585
+ for (const line of resetAccountLines) lines.push(uiTheme.fg("dim", line));
1586
+ }
1587
+
1588
+ const renderableGroups = Array.from(limitGroups.values()).map(group => {
1589
+ const entries = group.limits.map((limit, index) => ({
1590
+ limit,
1591
+ report: group.reports[index],
1592
+ fraction: resolveFraction(limit),
1593
+ index,
1594
+ }));
1595
+ entries.sort((a, b) => {
1596
+ const aFraction = a.fraction ?? -1;
1597
+ const bFraction = b.fraction ?? -1;
1598
+ if (aFraction !== bFraction) return bFraction - aFraction;
1599
+ return a.index - b.index;
1600
+ });
1601
+ const sortedLimits = entries.map(entry => entry.limit);
1602
+ const sortedReports = entries.map(entry => entry.report);
1603
+ return { group, sortedLimits, sortedReports, amountText: formatAggregateAmount(sortedLimits) };
1604
+ });
1605
+
1606
+ const sectionCount = renderableGroups.reduce((max, g) => Math.max(max, g.sortedLimits.length), 0);
1607
+ const sectionTrailing = renderableGroups.reduce((max, g) => Math.max(max, visibleWidth(g.amountText)), 0);
1608
+ const sectionColumnWidth = resolveColumnWidth(sectionCount, availableWidth, sectionTrailing);
1609
+
1610
+ for (const { group, sortedLimits, sortedReports, amountText } of renderableGroups) {
1611
+ const status = resolveAggregateStatus(sortedLimits);
1612
+ const statusIcon = resolveStatusIcon(status, uiTheme);
1613
+
1614
+ const windowSuffix = formatWindowSuffix(group.label, group.windowLabel, uiTheme);
1615
+ lines.push(`${statusIcon} ${uiTheme.bold(group.label)} ${windowSuffix}`.trim());
1616
+ const accountLabels = formatAccountHeaderRow(
1617
+ sortedLimits,
1618
+ sortedReports,
1619
+ nowMs,
1620
+ sectionColumnWidth,
1621
+ uiTheme,
1622
+ activeAccount,
1623
+ );
1624
+ lines.push(` ${accountLabels.join(" ")}`.trimEnd());
1625
+ const bars = sortedLimits.map(limit =>
1626
+ padColumn(renderUsageBar(limit, uiTheme, sectionColumnWidth), sectionColumnWidth),
1627
+ );
1628
+ lines.push(` ${bars.join(" ")} ${amountText}`.trimEnd());
1629
+ const resetText = sortedLimits.length <= 1 ? resolveResetRange(sortedLimits, nowMs) : null;
1630
+ if (resetText) {
1631
+ lines.push(` ${uiTheme.fg("dim", resetText)}`.trimEnd());
1632
+ }
1633
+ const notes = sortedLimits.flatMap(limit => limit.notes ?? []);
1634
+ if (notes.length > 0) {
1635
+ lines.push(` ${uiTheme.fg("dim", notes.join(" • "))}`.trimEnd());
1636
+ }
1637
+ }
1638
+
1639
+ // Render accounts with no rate limits (e.g. business/enterprise plans).
1640
+ const unlimitedReports = providerReports.filter(report => report.limits.length === 0);
1641
+ for (const report of unlimitedReports) {
1642
+ const label = formatUnlimitedReportLabel(report, 0);
1643
+ const tier = report.metadata?.planType as string | undefined;
1644
+ const tierSuffix = tier ? ` ${uiTheme.fg("dim", `(${tier})`)}` : "";
1645
+ lines.push(
1646
+ `${uiTheme.fg("success", uiTheme.status.success)} ${label}${tierSuffix} ${uiTheme.fg("dim", "-- no limits")}`,
1647
+ );
1648
+ }
1649
+ // No per-provider footer; global header shows last check.
1650
+ }
1651
+
1652
+ return lines.join("\n");
1653
+ }