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,1819 @@
1
+ /**
2
+ * Model resolution, scoping, and initial selection.
3
+ *
4
+ * Layering:
5
+ * - `matchModel` is the single matching engine. Order: exact `provider/id`
6
+ * reference (with variant-alias and OpenRouter routed/date fallbacks) →
7
+ * exact canonical id → exact bare id → retired variant alias →
8
+ * provider-scoped fuzzy → substring with alias-vs-dated pick.
9
+ * - `parseModelPatternWithContext`/`parseModelPattern` layer the selector
10
+ * grammar on top: trailing `:level` thinking suffixes (`splitThinkingSuffix`)
11
+ * and `@upstream` provider routing (`splitUpstreamRouting`).
12
+ * - Everything else (`resolveModelFromString`, `resolveModelOverride*`,
13
+ * `resolveRoleSelection`, `resolveModelScope`, `resolveCliModel`,
14
+ * `findSmolModel`/`findSlowModel`) adapts inputs — roles, settings patterns,
15
+ * CLI flags, scope globs — onto that pipeline.
16
+ */
17
+
18
+ import { ThinkingLevel } from "@oh-my-pi/pi-agent-core";
19
+ import type { Api, Effort, KnownProvider, Model, ModelSpec } from "@oh-my-pi/pi-ai";
20
+ import { buildModel } from "@oh-my-pi/pi-catalog/build";
21
+ import { modelMatchesHost } from "@oh-my-pi/pi-catalog/hosts";
22
+ import { buildModelProviderPriorityRank } from "@oh-my-pi/pi-catalog/identity";
23
+ import { stripThinkingVariantToken } from "@oh-my-pi/pi-catalog/identity/family";
24
+ import { clampThinkingLevelForModel } from "@oh-my-pi/pi-catalog/model-thinking";
25
+ import { modelsAreEqual } from "@oh-my-pi/pi-catalog/models";
26
+ import { DEFAULT_MODEL_PER_PROVIDER } from "@oh-my-pi/pi-catalog/provider-models";
27
+ import { resolveBareVariantAlias, resolveVariantAlias } from "@oh-my-pi/pi-catalog/variant-collapse";
28
+ import { fuzzyMatch } from "@oh-my-pi/pi-tui";
29
+ import { logger } from "@oh-my-pi/pi-utils";
30
+ import chalk from "chalk";
31
+ import MODEL_PRIO from "../priority.json" with { type: "json" };
32
+ import { parseThinkingLevel, resolveThinkingLevelForModel } from "../thinking";
33
+ import { isAuthenticated, kNoAuth, type ModelRegistry } from "./model-registry";
34
+ import { MODEL_ROLE_IDS, type ModelRole } from "./model-roles";
35
+ import type { Settings } from "./settings";
36
+
37
+ function isKnownProvider(provider: string): provider is KnownProvider {
38
+ return provider in DEFAULT_MODEL_PER_PROVIDER;
39
+ }
40
+
41
+ /**
42
+ * Pick the first provider-default model in availability order.
43
+ *
44
+ * If multiple providers expose that same default id, rank only that shared-id
45
+ * group by canonical provider priority so native/OAuth transports beat mirrors
46
+ * without changing unrelated provider fallback precedence.
47
+ */
48
+ export function pickDefaultAvailableModel(availableModels: Model<Api>[]): Model<Api> | undefined {
49
+ const firstDefault = availableModels.find(
50
+ model => isKnownProvider(model.provider) && DEFAULT_MODEL_PER_PROVIDER[model.provider] === model.id,
51
+ );
52
+ if (!firstDefault) return availableModels[0];
53
+
54
+ const providerPriority = buildModelProviderPriorityRank();
55
+ const sharedDefaultMatches = availableModels.filter(
56
+ model =>
57
+ model.id === firstDefault.id &&
58
+ isKnownProvider(model.provider) &&
59
+ DEFAULT_MODEL_PER_PROVIDER[model.provider] === model.id,
60
+ );
61
+ return [...sharedDefaultMatches].sort((a, b) => {
62
+ const aRank = providerPriority.get(a.provider.toLowerCase()) ?? Number.POSITIVE_INFINITY;
63
+ const bRank = providerPriority.get(b.provider.toLowerCase()) ?? Number.POSITIVE_INFINITY;
64
+ if (aRank !== bRank) return aRank - bRank;
65
+ return availableModels.indexOf(a) - availableModels.indexOf(b);
66
+ })[0];
67
+ }
68
+
69
+ export interface ScopedModel {
70
+ model: Model<Api>;
71
+ thinkingLevel?: ThinkingLevel;
72
+ explicitThinkingLevel: boolean;
73
+ }
74
+
75
+ interface ThinkingSuffixOptions {
76
+ allowMaxAlias?: boolean;
77
+ }
78
+
79
+ interface ModelStringParseOptions extends ThinkingSuffixOptions {
80
+ isLiteralModelId?: (provider: string, id: string) => boolean;
81
+ }
82
+ const MAX_THINKING_SUFFIX_OPTIONS: ThinkingSuffixOptions = { allowMaxAlias: true };
83
+
84
+ function parseThinkingSuffix(value: string, options?: ThinkingSuffixOptions): ThinkingLevel | undefined {
85
+ const level = parseThinkingLevel(value);
86
+ if (level !== undefined) return level;
87
+ return options?.allowMaxAlias === true && value === "max" ? ThinkingLevel.XHigh : undefined;
88
+ }
89
+
90
+ /**
91
+ * Split a trailing `:<level>` thinking selector off a model pattern.
92
+ *
93
+ * `level` is set only when the suffix parses as a valid thinking level, in
94
+ * which case `base` has the suffix stripped; otherwise `base` is the input.
95
+ * `minColonIndex` requires the colon to appear strictly after that index —
96
+ * role-alias callers pass `PREFIX_MODEL_ROLE.length` so the base is at least
97
+ * as long as the `pi/` prefix.
98
+ */
99
+ function splitThinkingSuffix(
100
+ pattern: string,
101
+ minColonIndex = -1,
102
+ options?: ThinkingSuffixOptions,
103
+ ): { base: string; level?: ThinkingLevel } {
104
+ const colonIdx = pattern.lastIndexOf(":");
105
+ if (colonIdx <= minColonIndex) return { base: pattern };
106
+ const level = parseThinkingSuffix(pattern.slice(colonIdx + 1), options);
107
+ return level ? { base: pattern.slice(0, colonIdx), level } : { base: pattern };
108
+ }
109
+
110
+ function hasExactModelPattern(pattern: string, availableModels: readonly Model<Api>[]): boolean {
111
+ const normalized = pattern.toLowerCase();
112
+ return availableModels.some(
113
+ model => model.id.toLowerCase() === normalized || `${model.provider}/${model.id}`.toLowerCase() === normalized,
114
+ );
115
+ }
116
+
117
+ function matchingGlobModels(pattern: string, availableModels: readonly Model<Api>[]): Model<Api>[] {
118
+ const glob = new Bun.Glob(pattern.toLowerCase());
119
+ return availableModels.filter(model => {
120
+ const fullId = `${model.provider}/${model.id}`;
121
+ return glob.match(fullId.toLowerCase()) || glob.match(model.id.toLowerCase());
122
+ });
123
+ }
124
+
125
+ function resolveGlobScopePattern(
126
+ pattern: string,
127
+ availableModels: readonly Model<Api>[],
128
+ ): { models: Model<Api>[]; thinkingLevel?: ThinkingLevel; explicitThinkingLevel: boolean } {
129
+ const strictSuffix = splitThinkingSuffix(pattern);
130
+ if (strictSuffix.level !== undefined) {
131
+ return {
132
+ models: matchingGlobModels(strictSuffix.base, availableModels),
133
+ thinkingLevel: strictSuffix.level,
134
+ explicitThinkingLevel: true,
135
+ };
136
+ }
137
+
138
+ const maxSuffix = splitThinkingSuffix(pattern, -1, MAX_THINKING_SUFFIX_OPTIONS);
139
+ if (maxSuffix.level !== undefined) {
140
+ const literalMatches = matchingGlobModels(pattern, availableModels);
141
+ if (literalMatches.length > 0) {
142
+ return { models: literalMatches, thinkingLevel: undefined, explicitThinkingLevel: false };
143
+ }
144
+ return {
145
+ models: matchingGlobModels(maxSuffix.base, availableModels),
146
+ thinkingLevel: maxSuffix.level,
147
+ explicitThinkingLevel: true,
148
+ };
149
+ }
150
+
151
+ return {
152
+ models: matchingGlobModels(pattern, availableModels),
153
+ thinkingLevel: undefined,
154
+ explicitThinkingLevel: false,
155
+ };
156
+ }
157
+
158
+ /**
159
+ * Parse a model string in "provider/modelId" format.
160
+ * Returns undefined if the format is invalid.
161
+ */
162
+ export function parseModelString(
163
+ modelStr: string,
164
+ options?: ModelStringParseOptions,
165
+ ): { provider: string; id: string; thinkingLevel?: ThinkingLevel } | undefined {
166
+ const slashIdx = modelStr.indexOf("/");
167
+ if (slashIdx <= 0) return undefined;
168
+ const id = modelStr.slice(slashIdx + 1);
169
+ const provider = modelStr.slice(0, slashIdx);
170
+ // Strip strict thinking level suffixes first (e.g. "claude-sonnet-4-6:high" -> id "claude-sonnet-4-6", thinkingLevel "high").
171
+ const strict = splitThinkingSuffix(id);
172
+ if (strict.level) return { provider, id: strict.base, thinkingLevel: strict.level };
173
+ // `max` is a provider-facing alias for xhigh, but real model IDs can end in
174
+ // `:max`. Context-aware callers pass a literal lookup so those models win.
175
+ const maxAlias = splitThinkingSuffix(id, -1, options);
176
+ if (maxAlias.level) {
177
+ return options?.isLiteralModelId?.(provider, id) === true
178
+ ? { provider, id }
179
+ : { provider, id: maxAlias.base, thinkingLevel: maxAlias.level };
180
+ }
181
+ return { provider, id };
182
+ }
183
+
184
+ /**
185
+ * Format a model as "provider/modelId" string.
186
+ */
187
+ export function formatModelString(model: Model<Api>): string {
188
+ return `${model.provider}/${model.id}`;
189
+ }
190
+
191
+ function getSingleRoutingOnly(routing: unknown): string | undefined {
192
+ if (!routing || typeof routing !== "object" || !("only" in routing) || !Array.isArray(routing.only)) {
193
+ return undefined;
194
+ }
195
+ if (routing.only.length !== 1) return undefined;
196
+ const upstream = routing.only[0];
197
+ return typeof upstream === "string" && upstream ? upstream : undefined;
198
+ }
199
+
200
+ function getSingleUpstreamRoute(model: Model<Api>): string | undefined {
201
+ const compat = model.compat;
202
+ if (!compat || typeof compat !== "object") return undefined;
203
+ if (modelMatchesHost(model, "vercelAIGateway") && "vercelGatewayRouting" in compat) {
204
+ return getSingleRoutingOnly(compat.vercelGatewayRouting);
205
+ }
206
+ if (modelMatchesHost(model, "openrouter") && "openRouterRouting" in compat) {
207
+ return getSingleRoutingOnly(compat.openRouterRouting);
208
+ }
209
+ return undefined;
210
+ }
211
+
212
+ export function formatModelStringWithRouting(model: Model<Api>): string {
213
+ const selector = formatModelString(model);
214
+ const upstream = getSingleUpstreamRoute(model);
215
+ return upstream ? `${selector}@${upstream}` : selector;
216
+ }
217
+
218
+ export function formatModelSelectorValue(selector: string, thinkingLevel: ThinkingLevel | undefined): string {
219
+ return thinkingLevel && thinkingLevel !== ThinkingLevel.Inherit ? `${selector}:${thinkingLevel}` : selector;
220
+ }
221
+
222
+ function getOpenRouterRouteSuffix(modelId: string): { baseId: string; suffix: string } | undefined {
223
+ const colonIdx = modelId.lastIndexOf(":");
224
+ if (colonIdx === -1) {
225
+ return undefined;
226
+ }
227
+
228
+ const suffix = modelId.slice(colonIdx + 1).trim();
229
+ // `max` is a thinking-level alias (xhigh), never an OpenRouter route suffix, so
230
+ // `openrouter/<id>:max` falls through to the max-aware selector split instead of
231
+ // being cloned into a literal `<id>:max` model id with the reasoning level lost.
232
+ if (!suffix || parseThinkingSuffix(suffix, MAX_THINKING_SUFFIX_OPTIONS)) {
233
+ return undefined;
234
+ }
235
+
236
+ return { baseId: modelId.slice(0, colonIdx), suffix };
237
+ }
238
+
239
+ function stripOpenRouterDateSuffix(modelId: string): string | undefined {
240
+ const stripped = modelId.replace(/-\d{8}(?=$|:)/i, "");
241
+ return stripped !== modelId ? stripped : undefined;
242
+ }
243
+
244
+ function getOpenRouterFallbackModelIds(modelId: string): string[] {
245
+ const orderedCandidates: string[] = [];
246
+ const queue = [modelId];
247
+ const seen = new Set<string>();
248
+
249
+ while (queue.length > 0) {
250
+ const candidate = queue.shift();
251
+ if (!candidate || seen.has(candidate)) {
252
+ continue;
253
+ }
254
+ seen.add(candidate);
255
+ orderedCandidates.push(candidate);
256
+
257
+ const routedSuffix = getOpenRouterRouteSuffix(candidate);
258
+ if (routedSuffix) {
259
+ queue.push(routedSuffix.baseId);
260
+ }
261
+
262
+ const strippedDate = stripOpenRouterDateSuffix(candidate);
263
+ if (strippedDate) {
264
+ queue.push(strippedDate);
265
+ }
266
+ }
267
+
268
+ return orderedCandidates;
269
+ }
270
+
271
+ function cloneModelWithRequestedId(model: Model<Api>, requestedId: string): Model<Api> {
272
+ return {
273
+ ...model,
274
+ id: requestedId,
275
+ ...(model.name === model.id ? { name: requestedId } : {}),
276
+ };
277
+ }
278
+
279
+ const AMAZON_BEDROCK_PROVIDER = "amazon-bedrock";
280
+ const BEDROCK_INFERENCE_PROFILE_ARN =
281
+ /^arn:aws(?:-[a-z]+)*:bedrock:[a-z0-9-]+:[0-9]*:(?:application-inference-profile|inference-profile)\/[a-z0-9][a-z0-9._:-]*$/i;
282
+
283
+ function hasBedrockInferenceProfileThinkingSuffix(modelId: string): boolean {
284
+ const { base, level } = splitThinkingSuffix(modelId);
285
+ return level !== undefined && BEDROCK_INFERENCE_PROFILE_ARN.test(base.trim());
286
+ }
287
+
288
+ function resolveBedrockInferenceProfileModelId(
289
+ modelId: string,
290
+ availableModels: readonly Model<Api>[],
291
+ ): Model<Api> | undefined {
292
+ const requestedId = modelId.trim();
293
+ if (hasBedrockInferenceProfileThinkingSuffix(requestedId) || !BEDROCK_INFERENCE_PROFILE_ARN.test(requestedId)) {
294
+ return undefined;
295
+ }
296
+
297
+ const template = availableModels.find(model => model.provider.toLowerCase() === AMAZON_BEDROCK_PROVIDER);
298
+ if (!template) return undefined;
299
+
300
+ return buildModel({
301
+ id: requestedId,
302
+ name: "Bedrock inference profile",
303
+ api: "bedrock-converse-stream",
304
+ provider: AMAZON_BEDROCK_PROVIDER,
305
+ baseUrl: template.baseUrl,
306
+ reasoning: false,
307
+ input: ["text"],
308
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
309
+ contextWindow: null,
310
+ maxTokens: null,
311
+ });
312
+ }
313
+
314
+ function resolveBedrockInferenceProfileReference(
315
+ provider: string,
316
+ modelId: string,
317
+ availableModels: readonly Model<Api>[],
318
+ ): Model<Api> | undefined {
319
+ if (provider.toLowerCase() !== AMAZON_BEDROCK_PROVIDER) return undefined;
320
+ return resolveBedrockInferenceProfileModelId(modelId, availableModels);
321
+ }
322
+
323
+ const UPSTREAM_ROUTING_SLUG = /^[a-z0-9](?:[a-z0-9-]*[a-z0-9])?$/i;
324
+
325
+ /**
326
+ * Split a trailing `@<upstream>` provider-routing selector off a model pattern.
327
+ *
328
+ * `openrouter/z-ai/glm-4.7@cerebras` -> base `openrouter/z-ai/glm-4.7`, upstream
329
+ * `cerebras`. A `:thinking` suffix after the slug is kept on the base
330
+ * (`...@cerebras:high` -> base `...:high`). Returns undefined when there is no
331
+ * `@` or the suffix is not a bare provider slug, so model ids that legitimately
332
+ * contain `@` (`claude-opus-4-8@default`, `workers-ai/@cf/...`) are never split.
333
+ */
334
+ export function splitUpstreamRouting(pattern: string): { base: string; upstream: string } | undefined {
335
+ const at = pattern.lastIndexOf("@");
336
+ if (at <= 0) return undefined;
337
+ const rest = pattern.slice(at + 1);
338
+ const colon = rest.indexOf(":");
339
+ const upstream = colon === -1 ? rest : rest.slice(0, colon);
340
+ if (!UPSTREAM_ROUTING_SLUG.test(upstream)) return undefined;
341
+ const trailing = colon === -1 ? "" : rest.slice(colon);
342
+ return { base: pattern.slice(0, at) + trailing, upstream };
343
+ }
344
+
345
+ /** OpenRouter and Vercel AI Gateway are the aggregators that honor per-request upstream routing. */
346
+ function supportsUpstreamRouting(model: Model<Api>): boolean {
347
+ return modelMatchesHost(model, "openrouter") || modelMatchesHost(model, "vercelAIGateway");
348
+ }
349
+
350
+ /** Pin a resolved aggregator model to a single upstream provider via its compat routing block. */
351
+ function applyUpstreamRouting(model: Model<Api>, upstream: string): Model<Api> {
352
+ const aggregatorModel = model as Model<"openai-completions">;
353
+ const routing = { only: [upstream] };
354
+ return buildModel({
355
+ ...model,
356
+ compat: modelMatchesHost(model, "vercelAIGateway")
357
+ ? { ...aggregatorModel.compatConfig, vercelGatewayRouting: routing }
358
+ : { ...aggregatorModel.compatConfig, openRouterRouting: routing },
359
+ } as ModelSpec<Api>);
360
+ }
361
+
362
+ const kProviderModelIndex = Symbol("model-resolver.providerIndex");
363
+ type ModelsWithProviderIndex = readonly Model<Api>[] & {
364
+ [kProviderModelIndex]?: Map<string, Model<Api> | null>;
365
+ };
366
+
367
+ function getProviderModelIndex(availableModels: readonly Model<Api>[]): Map<string, Model<Api> | null> {
368
+ const tagged = availableModels as ModelsWithProviderIndex;
369
+ const cached = tagged[kProviderModelIndex];
370
+ if (cached) return cached;
371
+ const index = new Map<string, Model<Api> | null>();
372
+ for (const m of availableModels) {
373
+ const key = `${m.provider.toLowerCase()}\u0000${m.id.toLowerCase()}`;
374
+ if (index.has(key)) {
375
+ index.set(key, null); // ambiguous sentinel; do not overwrite back
376
+ } else {
377
+ index.set(key, m);
378
+ }
379
+ }
380
+ tagged[kProviderModelIndex] = index;
381
+ return index;
382
+ }
383
+
384
+ export function resolveProviderModelReference(
385
+ provider: string,
386
+ modelId: string,
387
+ availableModels: readonly Model<Api>[],
388
+ ): Model<Api> | undefined {
389
+ const normalizedProvider = provider.trim().toLowerCase();
390
+ const normalizedModelId = modelId.trim().toLowerCase();
391
+ if (!normalizedProvider || !normalizedModelId) {
392
+ return undefined;
393
+ }
394
+
395
+ const index = getProviderModelIndex(availableModels);
396
+ const exact = index.get(`${normalizedProvider}\u0000${normalizedModelId}`);
397
+ if (exact === null) {
398
+ return undefined; // ambiguous
399
+ }
400
+ if (exact !== undefined) {
401
+ return exact;
402
+ }
403
+
404
+ // Retired effort-tier variant ids resolve to their collapsed logical
405
+ // model: hand-table aliases first, then the `X-thinking` → `X` grammar
406
+ // for auto-derived pairs. Exact lookup above always wins while raw is live.
407
+ const variantAliasId =
408
+ resolveVariantAlias(normalizedProvider, normalizedModelId) ?? stripThinkingVariantToken(normalizedModelId);
409
+ if (variantAliasId) {
410
+ const aliased = index.get(`${normalizedProvider}\u0000${variantAliasId.toLowerCase()}`);
411
+ if (aliased) {
412
+ return aliased;
413
+ }
414
+ }
415
+
416
+ const bedrockInferenceProfile = resolveBedrockInferenceProfileReference(provider, modelId, availableModels);
417
+ if (bedrockInferenceProfile) {
418
+ return bedrockInferenceProfile;
419
+ }
420
+
421
+ if (normalizedProvider !== "openrouter") {
422
+ return undefined;
423
+ }
424
+
425
+ for (const fallbackId of getOpenRouterFallbackModelIds(modelId).slice(1)) {
426
+ const fallback = index.get(`${normalizedProvider}\u0000${fallbackId.toLowerCase()}`);
427
+ if (fallback === null) {
428
+ return undefined;
429
+ }
430
+ if (fallback !== undefined) {
431
+ return cloneModelWithRequestedId(fallback, modelId);
432
+ }
433
+ }
434
+
435
+ return undefined;
436
+ }
437
+
438
+ export interface ModelMatchPreferences {
439
+ /** Most-recently-used model keys (provider/modelId) to prefer when ambiguous. */
440
+ usageOrder?: string[];
441
+ /** Provider precedence used for ambiguous unqualified model patterns. */
442
+ providerOrder?: readonly string[];
443
+ /** Providers to deprioritize when no recent usage or provider priority is available. */
444
+ deprioritizeProviders?: string[];
445
+ }
446
+
447
+ export type CanonicalModelRegistry = Partial<
448
+ Pick<ModelRegistry, "resolveCanonicalModel" | "getCanonicalVariants" | "getCanonicalId">
449
+ >;
450
+ export type ModelLookupRegistry = Pick<ModelRegistry, "getAvailable"> & Partial<CanonicalModelRegistry>;
451
+ type CliModelRegistry = Pick<ModelRegistry, "getAll"> & Partial<CanonicalModelRegistry>;
452
+ type InitialModelRegistry = Pick<ModelRegistry, "getAvailable" | "find">;
453
+ type RestorableModelRegistry = Pick<ModelRegistry, "getAvailable" | "find" | "getApiKey">;
454
+
455
+ interface ModelPreferenceContext {
456
+ modelUsageRank: Map<string, number>;
457
+ providerUsageRank: Map<string, number>;
458
+ providerPriorityRank: Map<string, number>;
459
+ deprioritizedProviders: Set<string>;
460
+ modelOrder: Map<string, number>;
461
+ }
462
+
463
+ function buildPreferenceContext(
464
+ availableModels: Model<Api>[],
465
+ preferences: ModelMatchPreferences | undefined,
466
+ ): ModelPreferenceContext {
467
+ const modelUsageRank = new Map<string, number>();
468
+ const providerUsageRank = new Map<string, number>();
469
+ const usageOrder = preferences?.usageOrder ?? [];
470
+ for (let i = 0; i < usageOrder.length; i += 1) {
471
+ const key = usageOrder[i];
472
+ if (!modelUsageRank.has(key)) {
473
+ modelUsageRank.set(key, i);
474
+ }
475
+ const parsed = parseModelString(key);
476
+ if (parsed && !providerUsageRank.has(parsed.provider)) {
477
+ providerUsageRank.set(parsed.provider, i);
478
+ }
479
+ }
480
+ const providerPriorityRank = buildModelProviderPriorityRank(preferences?.providerOrder);
481
+ const deprioritizedProviders = new Set(preferences?.deprioritizeProviders ?? []);
482
+ const modelOrder = new Map<string, number>();
483
+ for (let i = 0; i < availableModels.length; i += 1) {
484
+ modelOrder.set(formatModelString(availableModels[i]), i);
485
+ }
486
+
487
+ return { modelUsageRank, providerUsageRank, providerPriorityRank, deprioritizedProviders, modelOrder };
488
+ }
489
+
490
+ export function getModelMatchPreferences(
491
+ settings?: Partial<Pick<Settings, "get" | "getStorage">>,
492
+ ): ModelMatchPreferences {
493
+ return {
494
+ usageOrder: settings?.getStorage?.()?.getModelUsageOrder(),
495
+ providerOrder: settings?.get?.("modelProviderOrder"),
496
+ };
497
+ }
498
+
499
+ function mergeModelMatchPreferences(
500
+ settings: Settings | undefined,
501
+ preferences: ModelMatchPreferences | undefined,
502
+ ): ModelMatchPreferences {
503
+ const settingsPreferences = getModelMatchPreferences(settings);
504
+ return {
505
+ usageOrder: preferences?.usageOrder ?? settingsPreferences.usageOrder,
506
+ providerOrder: preferences?.providerOrder ?? settingsPreferences.providerOrder,
507
+ deprioritizeProviders: preferences?.deprioritizeProviders,
508
+ };
509
+ }
510
+
511
+ function pickPreferredModel(candidates: Model<Api>[], context: ModelPreferenceContext): Model<Api> {
512
+ if (candidates.length <= 1) return candidates[0];
513
+ return [...candidates].sort((a, b) => {
514
+ const aKey = formatModelString(a);
515
+ const bKey = formatModelString(b);
516
+ const aUsage = context.modelUsageRank.get(aKey);
517
+ const bUsage = context.modelUsageRank.get(bKey);
518
+ if (aUsage !== undefined || bUsage !== undefined) {
519
+ return (aUsage ?? Number.POSITIVE_INFINITY) - (bUsage ?? Number.POSITIVE_INFINITY);
520
+ }
521
+
522
+ const aProviderPriority = context.providerPriorityRank.get(a.provider.toLowerCase());
523
+ const bProviderPriority = context.providerPriorityRank.get(b.provider.toLowerCase());
524
+ if (aProviderPriority !== undefined || bProviderPriority !== undefined) {
525
+ return (aProviderPriority ?? Number.POSITIVE_INFINITY) - (bProviderPriority ?? Number.POSITIVE_INFINITY);
526
+ }
527
+
528
+ const aProviderUsage = context.providerUsageRank.get(a.provider);
529
+ const bProviderUsage = context.providerUsageRank.get(b.provider);
530
+ if (aProviderUsage !== undefined || bProviderUsage !== undefined) {
531
+ return (aProviderUsage ?? Number.POSITIVE_INFINITY) - (bProviderUsage ?? Number.POSITIVE_INFINITY);
532
+ }
533
+
534
+ const aDeprioritized = context.deprioritizedProviders.has(a.provider);
535
+ const bDeprioritized = context.deprioritizedProviders.has(b.provider);
536
+ if (aDeprioritized !== bDeprioritized) {
537
+ return aDeprioritized ? 1 : -1;
538
+ }
539
+
540
+ const aOrder = context.modelOrder.get(aKey) ?? 0;
541
+ const bOrder = context.modelOrder.get(bKey) ?? 0;
542
+ return aOrder - bOrder;
543
+ })[0];
544
+ }
545
+
546
+ /**
547
+ * Helper to check if a model ID looks like an alias (no date suffix)
548
+ * Dates are typically in format: -20241022 or -20250929
549
+ */
550
+ function isAlias(id: string): boolean {
551
+ // Check if ID ends with -latest
552
+ if (id.endsWith("-latest")) return true;
553
+
554
+ // Check if ID ends with a date pattern (-YYYYMMDD)
555
+ const datePattern = /-\d{8}$/;
556
+ return !datePattern.test(id);
557
+ }
558
+
559
+ function includeSyntheticAllowedModels(available: Model<Api>[], allowedModels: Iterable<Model<Api>>): Model<Api>[] {
560
+ const allowedByKey = new Map<string, Model<Api>>();
561
+ for (const model of allowedModels) {
562
+ const key = formatModelString(model);
563
+ if (!allowedByKey.has(key)) {
564
+ allowedByKey.set(key, model);
565
+ }
566
+ }
567
+ if (allowedByKey.size === 0) return [];
568
+
569
+ const result: Model<Api>[] = [];
570
+ for (const model of available) {
571
+ if (allowedByKey.delete(formatModelString(model))) {
572
+ result.push(model);
573
+ }
574
+ }
575
+
576
+ result.push(...allowedByKey.values());
577
+ return result;
578
+ }
579
+
580
+ /**
581
+ * Find an exact explicit provider/model match.
582
+ * Bare model ids are handled separately so canonical ids can coalesce variants.
583
+ */
584
+ function findExactModelReferenceMatch(modelReference: string, availableModels: Model<Api>[]): Model<Api> | undefined {
585
+ const trimmedReference = modelReference.trim();
586
+ if (!trimmedReference) {
587
+ return undefined;
588
+ }
589
+
590
+ const slashIndex = trimmedReference.indexOf("/");
591
+ if (slashIndex !== -1) {
592
+ const provider = trimmedReference.substring(0, slashIndex).trim();
593
+ const modelId = trimmedReference.substring(slashIndex + 1).trim();
594
+ if (provider && modelId) {
595
+ return resolveProviderModelReference(provider, modelId, availableModels);
596
+ }
597
+ }
598
+ return undefined;
599
+ }
600
+
601
+ function findExactCanonicalModelMatch(
602
+ modelReference: string,
603
+ availableModels: Model<Api>[],
604
+ modelRegistry: CanonicalModelRegistry | undefined,
605
+ ): Model<Api> | undefined {
606
+ if (!modelRegistry) {
607
+ return undefined;
608
+ }
609
+ const trimmedReference = modelReference.trim();
610
+ if (!trimmedReference || trimmedReference.includes("/")) {
611
+ return undefined;
612
+ }
613
+ return modelRegistry.resolveCanonicalModel?.(trimmedReference, {
614
+ availableOnly: false,
615
+ candidates: availableModels,
616
+ });
617
+ }
618
+
619
+ /**
620
+ * The single model-matching engine. Tries, in order:
621
+ * 1. exact `provider/id` reference (variant-alias and OpenRouter routed/date
622
+ * fallbacks included),
623
+
624
+ * 2. exact canonical id (coalesces provider variants),
625
+ * 3. exact bare id (preference-ranked),
626
+ * 4. retired effort-tier variant alias (collapsed catalog entries),
627
+ * 5. provider-scoped fuzzy match,
628
+ * 6. substring match with the alias-vs-dated pick.
629
+ * Returns the matched model or undefined if no match found.
630
+ */
631
+ function matchModel(
632
+ modelPattern: string,
633
+ availableModels: Model<Api>[],
634
+ context: ModelPreferenceContext,
635
+ options?: { modelRegistry?: CanonicalModelRegistry },
636
+ ): Model<Api> | undefined {
637
+ const exactRefMatch = findExactModelReferenceMatch(modelPattern, availableModels);
638
+ if (exactRefMatch) {
639
+ return exactRefMatch;
640
+ }
641
+ // Exact canonical ids coalesce provider variants before bare-id matching.
642
+ const exactCanonicalMatch = findExactCanonicalModelMatch(modelPattern, availableModels, options?.modelRegistry);
643
+ if (exactCanonicalMatch) {
644
+ return exactCanonicalMatch;
645
+ }
646
+
647
+ // Exact ID match (case-insensitive) — this must happen before provider-scoped
648
+ // fuzzy matching so raw IDs that contain slashes (for example OpenRouter model
649
+ // IDs like "openai/gpt-4o:extended") still resolve as IDs instead of being
650
+ // misread as a provider-qualified selector.
651
+ const exactMatches = availableModels.filter(m => m.id.toLowerCase() === modelPattern.toLowerCase());
652
+ if (exactMatches.length > 0) {
653
+ return pickPreferredModel(exactMatches, context);
654
+ }
655
+
656
+ const bedrockInferenceProfile = resolveBedrockInferenceProfileModelId(modelPattern, availableModels);
657
+ if (bedrockInferenceProfile) {
658
+ return bedrockInferenceProfile;
659
+ }
660
+
661
+ // Retired effort-tier variant ids (bare, no provider prefix) resolve to
662
+ // their collapsed logical model; models from the providers whose table
663
+ // declared the alias win ties. Auto-derived `X-thinking` pairs resolve
664
+ // through the grammar fallback.
665
+ const bareAlias = resolveBareVariantAlias(modelPattern);
666
+ const bareAliasTargetId = bareAlias?.id ?? stripThinkingVariantToken(modelPattern);
667
+ if (bareAliasTargetId) {
668
+ const aliasMatches = availableModels.filter(m => m.id.toLowerCase() === bareAliasTargetId.toLowerCase());
669
+ if (aliasMatches.length > 0) {
670
+ const preferred = bareAlias ? aliasMatches.filter(m => bareAlias.providers.includes(m.provider)) : [];
671
+ return pickPreferredModel(preferred.length > 0 ? preferred : aliasMatches, context);
672
+ }
673
+ }
674
+ // Check for provider/modelId format — fuzzy match within provider only.
675
+ const slashIndex = modelPattern.indexOf("/");
676
+ if (slashIndex !== -1) {
677
+ const provider = modelPattern.substring(0, slashIndex);
678
+ const modelId = modelPattern.substring(slashIndex + 1);
679
+ const providerModels = availableModels.filter(m => m.provider.toLowerCase() === provider.toLowerCase());
680
+ if (providerModels.length === 0) {
681
+ // The prefix is not a known provider in this candidate set, so treat the
682
+ // slash as part of the raw model ID and continue with generic matching.
683
+ } else {
684
+ // Let the routing fallback apply `@upstream` before fuzzy matching can consume the
685
+ // slug — but only for aggregator providers (OpenRouter / Vercel Gateway). Other
686
+ // providers have ids that legitimately end in `@` (Vertex `claude-opus-4-8@default`),
687
+ // and the fallback never routes them, so they must keep fuzzy matching.
688
+ if (splitUpstreamRouting(modelId) && providerModels.some(supportsUpstreamRouting)) {
689
+ return undefined;
690
+ }
691
+ const scored = providerModels
692
+ .map(model => ({ model, match: fuzzyMatch(modelId, model.id) }))
693
+ .filter(entry => entry.match.matches);
694
+ if (scored.length === 0) {
695
+ return undefined;
696
+ }
697
+
698
+ scored.sort((a, b) => {
699
+ if (a.match.score !== b.match.score) return a.match.score - b.match.score;
700
+ const aKey = formatModelString(a.model);
701
+ const bKey = formatModelString(b.model);
702
+ const aUsage = context.modelUsageRank.get(aKey) ?? Number.POSITIVE_INFINITY;
703
+ const bUsage = context.modelUsageRank.get(bKey) ?? Number.POSITIVE_INFINITY;
704
+ if (aUsage !== bUsage) return aUsage - bUsage;
705
+
706
+ const aProviderUsage = context.providerUsageRank.get(a.model.provider) ?? Number.POSITIVE_INFINITY;
707
+ const bProviderUsage = context.providerUsageRank.get(b.model.provider) ?? Number.POSITIVE_INFINITY;
708
+ if (aProviderUsage !== bProviderUsage) return aProviderUsage - bProviderUsage;
709
+
710
+ const aOrder = context.modelOrder.get(aKey) ?? 0;
711
+ const bOrder = context.modelOrder.get(bKey) ?? 0;
712
+ return aOrder - bOrder;
713
+ });
714
+ return scored[0]?.model;
715
+ }
716
+ }
717
+
718
+ // No exact match - fall back to partial matching
719
+ const matches = availableModels.filter(
720
+ m =>
721
+ m.id.toLowerCase().includes(modelPattern.toLowerCase()) ||
722
+ m.name?.toLowerCase().includes(modelPattern.toLowerCase()),
723
+ );
724
+
725
+ if (matches.length === 0) {
726
+ return undefined;
727
+ }
728
+
729
+ // Separate into aliases and dated versions
730
+ const aliases = matches.filter(m => isAlias(m.id));
731
+ const datedVersions = matches.filter(m => !isAlias(m.id));
732
+
733
+ if (aliases.length > 0) {
734
+ return pickPreferredModel(aliases, context);
735
+ }
736
+ if (datedVersions.length === 0) return undefined;
737
+
738
+ if (datedVersions.length === 1) {
739
+ return datedVersions[0];
740
+ }
741
+
742
+ const sortedById = [...datedVersions].sort((a, b) => b.id.localeCompare(a.id));
743
+ const topId = sortedById[0]?.id;
744
+ if (!topId) return undefined;
745
+ const topCandidates = sortedById.filter(model => model.id === topId);
746
+ return pickPreferredModel(topCandidates, context);
747
+ }
748
+
749
+ export interface ParsedModelResult {
750
+ model: Model<Api> | undefined;
751
+ /** Thinking level if explicitly specified in pattern, undefined otherwise */
752
+ thinkingLevel?: ThinkingLevel;
753
+ /** Upstream provider slug from an `@upstream` routing selector, if present. */
754
+ upstream?: string;
755
+ warning: string | undefined;
756
+ explicitThinkingLevel: boolean;
757
+ }
758
+
759
+ /**
760
+ * Parse a pattern to extract model and thinking level.
761
+ * Handles models with colons in their IDs (e.g., OpenRouter's :exacto suffix).
762
+ *
763
+ * Algorithm:
764
+ * 1. Try to match full pattern as a model
765
+ * 2. If found, return it with undefined thinking level
766
+ * 3. If not found and has colons, split on last colon:
767
+ * - If suffix is valid thinking level, use it and recurse on prefix
768
+ * - If suffix is invalid, warn and recurse on prefix
769
+ *
770
+ * @internal Exported for testing
771
+ */
772
+ function parseModelPatternWithContext(
773
+ pattern: string,
774
+ availableModels: Model<Api>[],
775
+ context: ModelPreferenceContext,
776
+ options?: { allowInvalidThinkingSelectorFallback?: boolean; modelRegistry?: CanonicalModelRegistry },
777
+ ): ParsedModelResult {
778
+ // Try exact match first
779
+ const exactMatch = matchModel(pattern, availableModels, context, options);
780
+ if (exactMatch) {
781
+ return { model: exactMatch, thinkingLevel: undefined, warning: undefined, explicitThinkingLevel: false };
782
+ }
783
+
784
+ // No match - try stripping a valid thinking suffix and recursing.
785
+ // `max` is accepted only after the full pattern failed, so literal model IDs
786
+ // ending in `:max` keep winning over the alias.
787
+ const { base, level } = splitThinkingSuffix(pattern, -1, MAX_THINKING_SUFFIX_OPTIONS);
788
+ if (level) {
789
+ const result = parseModelPatternWithContext(base, availableModels, context, options);
790
+ if (result.model) {
791
+ // Only use this thinking level if no warning from inner recursion
792
+ const explicitThinkingLevel = !result.warning;
793
+ return {
794
+ model: result.model,
795
+ thinkingLevel: explicitThinkingLevel ? level : undefined,
796
+ warning: result.warning,
797
+ explicitThinkingLevel,
798
+ };
799
+ }
800
+ return result;
801
+ }
802
+
803
+ const lastColonIndex = pattern.lastIndexOf(":");
804
+ if (lastColonIndex === -1) {
805
+ // No colons, pattern simply doesn't match any model
806
+ return { model: undefined, thinkingLevel: undefined, warning: undefined, explicitThinkingLevel: false };
807
+ }
808
+ const prefix = pattern.substring(0, lastColonIndex);
809
+ const suffix = pattern.substring(lastColonIndex + 1);
810
+
811
+ const allowFallback = options?.allowInvalidThinkingSelectorFallback ?? true;
812
+ if (!allowFallback) {
813
+ return { model: undefined, thinkingLevel: undefined, warning: undefined, explicitThinkingLevel: false };
814
+ }
815
+
816
+ // Invalid suffix - recurse on prefix and warn
817
+ const result = parseModelPatternWithContext(prefix, availableModels, context, options);
818
+ if (result.model) {
819
+ return {
820
+ model: result.model,
821
+ thinkingLevel: undefined,
822
+ warning: `Invalid thinking level "${suffix}" in pattern "${pattern}". Using default instead.`,
823
+ explicitThinkingLevel: false,
824
+ };
825
+ }
826
+ return result;
827
+ }
828
+
829
+ export function parseModelPattern(
830
+ pattern: string,
831
+ availableModels: Model<Api>[],
832
+ preferences?: ModelMatchPreferences,
833
+ options?: { allowInvalidThinkingSelectorFallback?: boolean; modelRegistry?: CanonicalModelRegistry },
834
+ ): ParsedModelResult {
835
+ const context = buildPreferenceContext(availableModels, preferences);
836
+ const direct = parseModelPatternWithContext(pattern, availableModels, context, options);
837
+ if (direct.model) return direct;
838
+
839
+ // No direct match: a trailing `@upstream` may be a provider-routing selector.
840
+ // Only honor it when the base resolves to an aggregator model (OpenRouter /
841
+ // Vercel Gateway); otherwise `@` stays part of the id and `direct` stands.
842
+ const routing = splitUpstreamRouting(pattern);
843
+ if (routing) {
844
+ const routed = parseModelPatternWithContext(routing.base, availableModels, context, options);
845
+ if (routed.model && supportsUpstreamRouting(routed.model)) {
846
+ return { ...routed, model: applyUpstreamRouting(routed.model, routing.upstream), upstream: routing.upstream };
847
+ }
848
+ }
849
+ return direct;
850
+ }
851
+
852
+ const PREFIX_MODEL_ROLE = "pi/";
853
+ const DEFAULT_MODEL_ROLE = "default";
854
+
855
+ function getModelRoleAlias(value: string): ModelRole | undefined {
856
+ const normalized = value.trim();
857
+ if (!normalized.startsWith(PREFIX_MODEL_ROLE)) return undefined;
858
+
859
+ const candidate = normalized.slice(PREFIX_MODEL_ROLE.length);
860
+ for (const role of MODEL_ROLE_IDS) {
861
+ if (candidate === role) return role;
862
+ }
863
+ return undefined;
864
+ }
865
+
866
+ function normalizeModelPatternList(value: string | string[] | undefined): string[] {
867
+ if (!value) return [];
868
+ const patterns = Array.isArray(value) ? value : value.split(",");
869
+ return patterns.map(pattern => pattern.trim()).filter(Boolean);
870
+ }
871
+
872
+ function isSessionInheritedAgentPattern(value: string): boolean {
873
+ return value === DEFAULT_MODEL_ROLE || value === `${PREFIX_MODEL_ROLE}${DEFAULT_MODEL_ROLE}` || value === "pi/task";
874
+ }
875
+
876
+ function shouldInheritDefaultBeforePriority(role: ModelRole): boolean {
877
+ return role === "smol" || role === "slow" || role === "designer";
878
+ }
879
+
880
+ function resolveDefaultInheritedPatterns(
881
+ role: ModelRole,
882
+ configuredDefault: string | undefined,
883
+ roleDefaults: string[],
884
+ settings: Settings | undefined,
885
+ visited: Set<ModelRole>,
886
+ ): string[] {
887
+ if (!shouldInheritDefaultBeforePriority(role) || !configuredDefault) return [];
888
+
889
+ const resolved: string[] = [];
890
+ for (const pattern of normalizeModelPatternList(configuredDefault)) {
891
+ const { base: aliasCandidate, level: thinkingLevel } = splitThinkingSuffix(
892
+ pattern,
893
+ PREFIX_MODEL_ROLE.length,
894
+ MAX_THINKING_SUFFIX_OPTIONS,
895
+ );
896
+ const aliasRole = getModelRoleAlias(aliasCandidate);
897
+ if (aliasRole === role) {
898
+ // Self-alias (e.g. modelRoles.default = "pi/smol") would loop back to the
899
+ // same unset role; collapse straight to the built-in priority chain.
900
+ resolved.push(
901
+ ...(thinkingLevel
902
+ ? roleDefaults.map(defaultPattern => `${defaultPattern}:${thinkingLevel}`)
903
+ : roleDefaults),
904
+ );
905
+ continue;
906
+ }
907
+ if (aliasRole && !visited.has(aliasRole)) {
908
+ // Cross-role alias (e.g. modelRoles.default = "pi/slow"): resolve the
909
+ // target role's patterns now so downstream one-layer expanders see
910
+ // concrete model patterns instead of another role alias.
911
+ const recursed = resolveConfiguredRolePattern(pattern, settings, new Set(visited));
912
+ if (recursed && recursed.length > 0) {
913
+ resolved.push(...recursed);
914
+ continue;
915
+ }
916
+ }
917
+ resolved.push(pattern);
918
+ }
919
+ return resolved;
920
+ }
921
+
922
+ function resolveConfiguredRolePattern(
923
+ value: string,
924
+ settings?: Settings,
925
+ visited: Set<ModelRole> = new Set(),
926
+ ): string[] | undefined {
927
+ const normalized = value.trim();
928
+ if (!normalized) return undefined;
929
+
930
+ const { base: aliasCandidate, level: thinkingLevel } = splitThinkingSuffix(
931
+ normalized,
932
+ PREFIX_MODEL_ROLE.length,
933
+ MAX_THINKING_SUFFIX_OPTIONS,
934
+ );
935
+ const role = getModelRoleAlias(aliasCandidate);
936
+ if (!role) return [normalized];
937
+ if (visited.has(role)) return undefined;
938
+ visited.add(role);
939
+
940
+ const configured = settings?.getModelRole(role)?.trim();
941
+ const configuredDefault = settings?.getModelRole(DEFAULT_MODEL_ROLE)?.trim();
942
+ const roleDefaults = normalizeModelPatternList(MODEL_PRIO[role as keyof typeof MODEL_PRIO]);
943
+ const resolved = configured
944
+ ? normalizeModelPatternList(configured)
945
+ : resolveDefaultInheritedPatterns(role, configuredDefault, roleDefaults, settings, visited);
946
+ if (resolved.length === 0) {
947
+ resolved.push(...roleDefaults);
948
+ }
949
+ if (resolved.length === 0) {
950
+ return undefined;
951
+ }
952
+
953
+ return thinkingLevel ? resolved.map(pattern => `${pattern}:${thinkingLevel}`) : resolved;
954
+ }
955
+
956
+ /**
957
+ * Expand a role alias like "pi/smol" to the configured model string.
958
+ */
959
+ export function expandRoleAlias(value: string, settings?: Settings): string {
960
+ const normalized = value.trim();
961
+ if (normalized === DEFAULT_MODEL_ROLE) {
962
+ return settings?.getModelRole("default") ?? value;
963
+ }
964
+
965
+ const resolved = resolveConfiguredRolePattern(value, settings)?.[0];
966
+ return resolved ?? value;
967
+ }
968
+
969
+ export function resolveConfiguredModelPatterns(value: string | string[] | undefined, settings?: Settings): string[] {
970
+ const patterns = normalizeModelPatternList(value);
971
+ return patterns.flatMap(pattern => {
972
+ const resolved = resolveConfiguredRolePattern(pattern, settings);
973
+ return resolved ?? [];
974
+ });
975
+ }
976
+ export interface AgentModelPatternResolutionOptions {
977
+ settingsOverride?: string | string[];
978
+ agentModel?: string | string[];
979
+ settings?: Settings;
980
+ activeModelPattern?: string;
981
+ fallbackModelPattern?: string;
982
+ }
983
+
984
+ export function resolveAgentModelPatterns(options: AgentModelPatternResolutionOptions): string[] {
985
+ const { settingsOverride, agentModel, settings, activeModelPattern, fallbackModelPattern } = options;
986
+
987
+ const overridePatterns = resolveConfiguredModelPatterns(settingsOverride, settings);
988
+ if (overridePatterns.length > 0) return overridePatterns;
989
+
990
+ const normalizedAgentPatterns = normalizeModelPatternList(agentModel);
991
+ const configuredAgentPatterns = resolveConfiguredModelPatterns(agentModel, settings);
992
+ const singleAgentPattern = normalizedAgentPatterns.length === 1 ? normalizedAgentPatterns[0] : undefined;
993
+ const agentInheritsSessionModel = singleAgentPattern ? isSessionInheritedAgentPattern(singleAgentPattern) : false;
994
+ if (configuredAgentPatterns.length > 0) {
995
+ if (!agentInheritsSessionModel) return configuredAgentPatterns;
996
+ if (singleAgentPattern === "pi/task") return configuredAgentPatterns;
997
+ }
998
+
999
+ const fallback =
1000
+ activeModelPattern?.trim() || fallbackModelPattern?.trim() || settings?.getModelRole("default")?.trim() || "";
1001
+ return resolveConfiguredModelPatterns(fallback, settings);
1002
+ }
1003
+
1004
+ /**
1005
+ * Resolve a model role value into a concrete model and thinking metadata.
1006
+ */
1007
+ export interface ResolvedModelRoleValue {
1008
+ model: Model<Api> | undefined;
1009
+ thinkingLevel?: ThinkingLevel;
1010
+ explicitThinkingLevel: boolean;
1011
+ warning: string | undefined;
1012
+ }
1013
+
1014
+ export function resolveModelRoleValue(
1015
+ roleValue: string | undefined,
1016
+ availableModels: Model<Api>[],
1017
+ options?: { settings?: Settings; matchPreferences?: ModelMatchPreferences; modelRegistry?: CanonicalModelRegistry },
1018
+ ): ResolvedModelRoleValue {
1019
+ if (!roleValue) {
1020
+ return { model: undefined, thinkingLevel: undefined, explicitThinkingLevel: false, warning: undefined };
1021
+ }
1022
+
1023
+ const normalized = roleValue.trim();
1024
+ if (!normalized || normalized === DEFAULT_MODEL_ROLE) {
1025
+ return { model: undefined, thinkingLevel: undefined, explicitThinkingLevel: false, warning: undefined };
1026
+ }
1027
+
1028
+ const effectivePatterns = resolveConfiguredModelPatterns(normalized, options?.settings);
1029
+ if (!effectivePatterns || effectivePatterns.length === 0) {
1030
+ return { model: undefined, thinkingLevel: undefined, explicitThinkingLevel: false, warning: undefined };
1031
+ }
1032
+
1033
+ let warning: string | undefined;
1034
+ const matchPreferences = mergeModelMatchPreferences(options?.settings, options?.matchPreferences);
1035
+ for (const effectivePattern of effectivePatterns) {
1036
+ const resolved = parseModelPattern(effectivePattern, availableModels, matchPreferences, {
1037
+ modelRegistry: options?.modelRegistry,
1038
+ });
1039
+ if (resolved.model) {
1040
+ return {
1041
+ model: resolved.model,
1042
+ thinkingLevel: resolved.explicitThinkingLevel
1043
+ ? (resolveThinkingLevelForModel(resolved.model, resolved.thinkingLevel) ?? resolved.thinkingLevel)
1044
+ : resolved.thinkingLevel,
1045
+ explicitThinkingLevel: resolved.explicitThinkingLevel,
1046
+ warning: resolved.warning,
1047
+ };
1048
+ }
1049
+ if (!warning && resolved.warning) {
1050
+ warning = resolved.warning;
1051
+ }
1052
+ }
1053
+
1054
+ return { model: undefined, thinkingLevel: undefined, explicitThinkingLevel: false, warning };
1055
+ }
1056
+
1057
+ interface ExplicitThinkingSelectorOptions {
1058
+ isLiteralModelId?: (provider: string, id: string) => boolean;
1059
+ }
1060
+
1061
+ function isLiteralModelSelector(value: string, options?: ExplicitThinkingSelectorOptions): boolean {
1062
+ const parsed = parseModelString(value);
1063
+ return parsed !== undefined && options?.isLiteralModelId?.(parsed.provider, parsed.id) === true;
1064
+ }
1065
+
1066
+ export function extractExplicitThinkingSelector(
1067
+ value: string | undefined,
1068
+ settings?: Settings,
1069
+ options?: ExplicitThinkingSelectorOptions,
1070
+ ): ThinkingLevel | undefined {
1071
+ if (!value) return undefined;
1072
+ const normalized = value.trim();
1073
+ if (!normalized || normalized === DEFAULT_MODEL_ROLE) return undefined;
1074
+
1075
+ const visited = new Set<string>();
1076
+ let current = normalized;
1077
+ while (!visited.has(current)) {
1078
+ visited.add(current);
1079
+ const strictSelector = splitThinkingSuffix(current, PREFIX_MODEL_ROLE.length).level;
1080
+ if (strictSelector) {
1081
+ return strictSelector;
1082
+ }
1083
+ const maxSelector = splitThinkingSuffix(current, PREFIX_MODEL_ROLE.length, MAX_THINKING_SUFFIX_OPTIONS).level;
1084
+ if (maxSelector && (current.startsWith(PREFIX_MODEL_ROLE) || !isLiteralModelSelector(current, options))) {
1085
+ return maxSelector;
1086
+ }
1087
+ const expanded = expandRoleAlias(current, settings).trim();
1088
+ if (!expanded || expanded === current) break;
1089
+ if (expanded === DEFAULT_MODEL_ROLE) return undefined;
1090
+ current = expanded;
1091
+ }
1092
+
1093
+ return undefined;
1094
+ }
1095
+
1096
+ /**
1097
+ * Resolve a model identifier or pattern to a Model instance.
1098
+ */
1099
+ export function resolveModelFromString(
1100
+ value: string,
1101
+ available: Model<Api>[],
1102
+ matchPreferences?: ModelMatchPreferences,
1103
+ modelRegistry?: CanonicalModelRegistry,
1104
+ ): Model<Api> | undefined {
1105
+ const exact = available.find(model => `${model.provider}/${model.id}` === value);
1106
+ if (exact) return exact;
1107
+ const parsed = parseModelString(value, {
1108
+ ...MAX_THINKING_SUFFIX_OPTIONS,
1109
+ isLiteralModelId: (provider, id) => available.some(model => model.provider === provider && model.id === id),
1110
+ });
1111
+ if (parsed) {
1112
+ const parsedExact = available.find(model => model.provider === parsed.provider && model.id === parsed.id);
1113
+ if (parsedExact) return parsedExact;
1114
+ }
1115
+ return parseModelPattern(value, available, matchPreferences, { modelRegistry }).model;
1116
+ }
1117
+
1118
+ /**
1119
+ * Resolve a model from configured roles, honoring order and overrides.
1120
+ */
1121
+ export function resolveModelFromSettings(options: {
1122
+ settings: Settings;
1123
+ availableModels: Model<Api>[];
1124
+ matchPreferences?: ModelMatchPreferences;
1125
+ roleOrder?: readonly ModelRole[];
1126
+ modelRegistry?: CanonicalModelRegistry;
1127
+ }): Model<Api> | undefined {
1128
+ const { settings, availableModels, matchPreferences, roleOrder, modelRegistry } = options;
1129
+ const roles = roleOrder ?? MODEL_ROLE_IDS;
1130
+ let sawConfiguredProviderQualifiedRole = false;
1131
+ for (const role of roles) {
1132
+ const configured = settings.getModelRole(role);
1133
+ if (!configured) continue;
1134
+ const expanded = expandRoleAlias(configured, settings).trim();
1135
+ if (expanded.includes("/")) {
1136
+ sawConfiguredProviderQualifiedRole = true;
1137
+ }
1138
+ const resolved = resolveModelFromString(expanded, availableModels, matchPreferences, modelRegistry);
1139
+ if (resolved) return resolved;
1140
+ }
1141
+ return sawConfiguredProviderQualifiedRole ? undefined : availableModels[0];
1142
+ }
1143
+
1144
+ /**
1145
+ * Resolve a list of override patterns to the first matching model.
1146
+ */
1147
+ export function resolveModelOverride(
1148
+ modelPatterns: string[],
1149
+ modelRegistry: ModelLookupRegistry,
1150
+ settings?: Settings,
1151
+ ): { model?: Model<Api>; thinkingLevel?: ThinkingLevel; explicitThinkingLevel: boolean } {
1152
+ if (modelPatterns.length === 0) return { explicitThinkingLevel: false };
1153
+ const availableModels = modelRegistry.getAvailable();
1154
+ const matchPreferences = getModelMatchPreferences(settings);
1155
+ for (const pattern of modelPatterns) {
1156
+ const { model, thinkingLevel, explicitThinkingLevel } = resolveModelRoleValue(pattern, availableModels, {
1157
+ settings,
1158
+ matchPreferences,
1159
+ modelRegistry,
1160
+ });
1161
+ if (model) {
1162
+ return { model, thinkingLevel, explicitThinkingLevel };
1163
+ }
1164
+ }
1165
+ return { explicitThinkingLevel: false };
1166
+ }
1167
+
1168
+ /**
1169
+ * Resolve a list of override patterns to the first matching model, with an
1170
+ * auth-aware fallback to the parent session's active model.
1171
+ *
1172
+ * If the resolved subagent model has no working credentials (provider has no
1173
+ * usable auth), and the parent's active model resolves with working auth,
1174
+ * use the parent's model instead. This prevents subagent dispatch from
1175
+ * silently routing to a provider the user can't actually call (e.g.
1176
+ * `modelRoles.task` pointing at an unqualified id whose only available
1177
+ * provider variant has no configured credentials — see #985).
1178
+ *
1179
+ * Keyless-by-design providers (llama.cpp, ollama, lm-studio) advertise the
1180
+ * `kNoAuth` sentinel from `getApiKey` to signal that they do not require
1181
+ * credentials. Those are treated as authenticated here so an explicitly
1182
+ * configured local model is never silently rerouted to the parent's remote
1183
+ * provider (see #1008).
1184
+ *
1185
+ * If neither the subagent nor the parent has working auth, returns the
1186
+ * primary resolution unchanged so the existing error path still surfaces
1187
+ * a meaningful failure downstream.
1188
+ */
1189
+ export async function resolveModelOverrideWithAuthFallback(
1190
+ modelPatterns: string[],
1191
+ parentActiveModelPattern: string | undefined,
1192
+ modelRegistry: ModelLookupRegistry & Pick<ModelRegistry, "getApiKey">,
1193
+ settings?: Settings,
1194
+ ): Promise<{
1195
+ model?: Model<Api>;
1196
+ thinkingLevel?: ThinkingLevel;
1197
+ explicitThinkingLevel: boolean;
1198
+ authFallbackUsed: boolean;
1199
+ }> {
1200
+ const primary = resolveModelOverride(modelPatterns, modelRegistry, settings);
1201
+ if (!primary.model || !parentActiveModelPattern) {
1202
+ return { ...primary, authFallbackUsed: false };
1203
+ }
1204
+
1205
+ const primaryKey = await modelRegistry.getApiKey(primary.model);
1206
+ if (primaryKey === kNoAuth || isAuthenticated(primaryKey)) {
1207
+ return { ...primary, authFallbackUsed: false };
1208
+ }
1209
+
1210
+ const fallback = resolveModelOverride([parentActiveModelPattern], modelRegistry, settings);
1211
+ if (!fallback.model) {
1212
+ return { ...primary, authFallbackUsed: false };
1213
+ }
1214
+ if (modelsAreEqual(fallback.model, primary.model)) {
1215
+ return { ...primary, authFallbackUsed: false };
1216
+ }
1217
+ const fallbackKey = await modelRegistry.getApiKey(fallback.model);
1218
+ if (!isAuthenticated(fallbackKey)) {
1219
+ return { ...primary, authFallbackUsed: false };
1220
+ }
1221
+
1222
+ return { ...fallback, authFallbackUsed: true };
1223
+ }
1224
+
1225
+ /**
1226
+ * Resolve a list of role patterns to the first matching model.
1227
+ */
1228
+ export function resolveRoleSelection(
1229
+ roles: readonly string[],
1230
+ settings: Settings,
1231
+ availableModels: Model<Api>[],
1232
+ modelRegistry?: CanonicalModelRegistry,
1233
+ ): { model: Model<Api>; thinkingLevel?: ThinkingLevel } | undefined {
1234
+ const matchPreferences = getModelMatchPreferences(settings);
1235
+ for (const role of roles) {
1236
+ const resolved = resolveModelRoleValue(settings.getModelRole(role), availableModels, {
1237
+ settings,
1238
+ matchPreferences,
1239
+ modelRegistry,
1240
+ });
1241
+ if (resolved.model) {
1242
+ return { model: resolved.model, thinkingLevel: resolved.thinkingLevel };
1243
+ }
1244
+ }
1245
+ return undefined;
1246
+ }
1247
+
1248
+ function resolveExactCanonicalScopePattern(
1249
+ pattern: string,
1250
+ modelRegistry: Pick<ModelRegistry, "getCanonicalVariants">,
1251
+ availableModels: Model<Api>[],
1252
+ ): { models: Model<Api>[]; thinkingLevel?: ThinkingLevel; explicitThinkingLevel: boolean } | undefined {
1253
+ if (pattern.endsWith(":max") && hasExactModelPattern(pattern, availableModels)) {
1254
+ return undefined;
1255
+ }
1256
+ const { base: canonicalId, level: thinkingLevel } = splitThinkingSuffix(pattern, -1, MAX_THINKING_SUFFIX_OPTIONS);
1257
+ const explicitThinkingLevel = thinkingLevel !== undefined;
1258
+
1259
+ const variants = modelRegistry
1260
+ .getCanonicalVariants(canonicalId, { availableOnly: true, candidates: availableModels })
1261
+ .map(variant => variant.model);
1262
+ if (variants.length === 0) {
1263
+ return undefined;
1264
+ }
1265
+
1266
+ return { models: variants, thinkingLevel, explicitThinkingLevel };
1267
+ }
1268
+
1269
+ /**
1270
+ * Resolve model patterns to actual Model objects with optional thinking levels
1271
+ * Format: "pattern:level" where :level is optional
1272
+ * For each pattern, finds all matching models and picks the best version:
1273
+ * 1. Prefer alias (e.g., claude-sonnet-4-5) over dated versions (claude-sonnet-4-5-20250929)
1274
+ * 2. If no alias, pick the latest dated version
1275
+ *
1276
+ * Supports models with colons in their IDs (e.g., OpenRouter's model:exacto).
1277
+ * The algorithm tries to match the full pattern first, then progressively
1278
+ * strips colon-suffixes to find a match.
1279
+ */
1280
+ export async function resolveModelScope(
1281
+ patterns: string[],
1282
+ modelRegistry: Pick<ModelRegistry, "getAvailable" | "getCanonicalVariants">,
1283
+ preferences?: ModelMatchPreferences,
1284
+ ): Promise<ScopedModel[]> {
1285
+ const availableModels = modelRegistry.getAvailable();
1286
+ const context = buildPreferenceContext(availableModels, preferences);
1287
+ const scopedModels: ScopedModel[] = [];
1288
+ const addScopedModel = (model: Model<Api>, thinkingLevel: ThinkingLevel | undefined, explicit: boolean) => {
1289
+ if (scopedModels.some(sm => modelsAreEqual(sm.model, model))) return;
1290
+ scopedModels.push({
1291
+ model,
1292
+ thinkingLevel: explicit
1293
+ ? (resolveThinkingLevelForModel(model, thinkingLevel) ?? thinkingLevel)
1294
+ : thinkingLevel,
1295
+ explicitThinkingLevel: explicit,
1296
+ });
1297
+ };
1298
+
1299
+ for (const pattern of patterns) {
1300
+ // Check if pattern contains glob characters
1301
+ if (pattern.includes("*") || pattern.includes("?") || pattern.includes("[")) {
1302
+ // Extract optional thinking level suffix (e.g., "provider/*:high") only
1303
+ // after literal `:max` globs had a chance to match real model IDs.
1304
+ const {
1305
+ models: matchingModels,
1306
+ thinkingLevel,
1307
+ explicitThinkingLevel,
1308
+ } = resolveGlobScopePattern(pattern, availableModels);
1309
+
1310
+ if (matchingModels.length === 0) {
1311
+ logger.warn(`No models match pattern "${pattern}"`);
1312
+ continue;
1313
+ }
1314
+
1315
+ for (const model of matchingModels) {
1316
+ addScopedModel(model, thinkingLevel, explicitThinkingLevel);
1317
+ }
1318
+ continue;
1319
+ }
1320
+
1321
+ const exactCanonical = resolveExactCanonicalScopePattern(pattern, modelRegistry, availableModels);
1322
+ if (exactCanonical) {
1323
+ for (const model of exactCanonical.models) {
1324
+ addScopedModel(model, exactCanonical.thinkingLevel, exactCanonical.explicitThinkingLevel);
1325
+ }
1326
+ continue;
1327
+ }
1328
+
1329
+ const { model, thinkingLevel, warning, explicitThinkingLevel } = parseModelPatternWithContext(
1330
+ pattern,
1331
+ availableModels,
1332
+ context,
1333
+ { modelRegistry },
1334
+ );
1335
+
1336
+ if (warning) {
1337
+ logger.warn(warning);
1338
+ }
1339
+
1340
+ if (!model) {
1341
+ logger.warn(`No models match pattern "${pattern}"`);
1342
+ continue;
1343
+ }
1344
+
1345
+ addScopedModel(model, thinkingLevel, explicitThinkingLevel);
1346
+ }
1347
+
1348
+ return scopedModels;
1349
+ }
1350
+
1351
+ /**
1352
+ * Resolve the set of models a session is allowed to use, given the active
1353
+ * settings. Starts from `modelRegistry.getAvailable()` (so disabled providers
1354
+ * and providers without credentials are already filtered out) and, when
1355
+ * `enabledModels` is configured for the current path scope, further restricts
1356
+ * the result to models matching those patterns.
1357
+ *
1358
+ * Returns the unfiltered available list when `enabledModels` is empty.
1359
+ * Returns an empty list when `enabledModels` is configured but no model matches
1360
+ * any pattern — callers MUST treat this as "no usable model" rather than
1361
+ * falling back to the global default (see issue #1022).
1362
+ */
1363
+ export async function resolveAllowedModels(
1364
+ modelRegistry: Pick<ModelRegistry, "getAvailable" | "getCanonicalVariants">,
1365
+ settings: Settings | undefined,
1366
+ preferences?: ModelMatchPreferences,
1367
+ ): Promise<Model<Api>[]> {
1368
+ const available = modelRegistry.getAvailable();
1369
+ const patterns = settings?.get("enabledModels");
1370
+ if (!patterns || patterns.length === 0) {
1371
+ return available;
1372
+ }
1373
+ const scoped = await resolveModelScope(patterns, modelRegistry, preferences);
1374
+ if (scoped.length === 0) {
1375
+ return [];
1376
+ }
1377
+ return includeSyntheticAllowedModels(
1378
+ available,
1379
+ scoped.map(entry => entry.model),
1380
+ );
1381
+ }
1382
+
1383
+ /**
1384
+ * Synchronous subset of {@link resolveAllowedModels} for contexts where async is unavailable
1385
+ * (e.g. `getAvailableModels()` which is called from the ACP model-list advertisement, RPC
1386
+ * `get_available_models`, and the `/model` slash command). Uses the same effective
1387
+ * `enabledModels` scope semantics as startup resolution:
1388
+ *
1389
+ * - Glob selectors match `provider/modelId` and bare model id
1390
+ * - Exact canonical ids expand to all available concrete variants
1391
+ * - Exact `provider/modelId`, bare ids, provider-scoped fuzzy, and substring selectors
1392
+ * resolve through the shared model-pattern matcher
1393
+ * - Optional `:thinkingLevel` suffixes are stripped only when valid
1394
+ *
1395
+ * When no pattern resolves to any model (misconfiguration / typo) an empty list is returned,
1396
+ * consistent with the empty-list contract of {@link resolveAllowedModels}. Callers that render
1397
+ * a UI picker should treat an empty list as "hide the picker entry", matching how the SDK
1398
+ * surfaces the same misconfiguration during session initialization.
1399
+ */
1400
+ export function filterAvailableModelsByEnabledPatterns(
1401
+ available: Model<Api>[],
1402
+ patterns: readonly string[],
1403
+ registry: Pick<ModelRegistry, "getCanonicalVariants">,
1404
+ ): Model<Api>[] {
1405
+ if (patterns.length === 0) return available;
1406
+
1407
+ const context = buildPreferenceContext(available, undefined);
1408
+ const allowedModels: Model<Api>[] = [];
1409
+ const addAllowed = (model: Model<Api>) => {
1410
+ allowedModels.push(model);
1411
+ };
1412
+
1413
+ for (const pattern of patterns) {
1414
+ if (pattern.includes("*") || pattern.includes("?") || pattern.includes("[")) {
1415
+ for (const model of resolveGlobScopePattern(pattern, available).models) {
1416
+ addAllowed(model);
1417
+ }
1418
+ continue;
1419
+ }
1420
+
1421
+ const exactCanonical = resolveExactCanonicalScopePattern(pattern, registry, available);
1422
+ if (exactCanonical) {
1423
+ for (const model of exactCanonical.models) {
1424
+ addAllowed(model);
1425
+ }
1426
+ continue;
1427
+ }
1428
+
1429
+ const { model } = parseModelPatternWithContext(pattern, available, context, { modelRegistry: registry });
1430
+ if (model) {
1431
+ addAllowed(model);
1432
+ }
1433
+ }
1434
+
1435
+ return includeSyntheticAllowedModels(available, allowedModels);
1436
+ }
1437
+
1438
+ export interface ResolveCliModelResult {
1439
+ model: Model<Api> | undefined;
1440
+ selector?: string;
1441
+ thinkingLevel?: ThinkingLevel;
1442
+ warning: string | undefined;
1443
+ error: string | undefined;
1444
+ }
1445
+
1446
+ /**
1447
+ * Resolve a single model from CLI flags.
1448
+ */
1449
+ export function resolveCliModel(options: {
1450
+ cliProvider?: string;
1451
+ cliModel?: string;
1452
+ modelRegistry: CliModelRegistry;
1453
+ preferences?: ModelMatchPreferences;
1454
+ }): ResolveCliModelResult {
1455
+ const { cliProvider, cliModel, modelRegistry, preferences } = options;
1456
+
1457
+ if (!cliModel) {
1458
+ return { model: undefined, selector: undefined, warning: undefined, error: undefined };
1459
+ }
1460
+
1461
+ const availableModels = modelRegistry.getAll();
1462
+ if (availableModels.length === 0) {
1463
+ return {
1464
+ model: undefined,
1465
+ selector: undefined,
1466
+ warning: undefined,
1467
+ error: "No models available. Check your installation or add models to models.json.",
1468
+ };
1469
+ }
1470
+
1471
+ const providerMap = new Map<string, string>();
1472
+ for (const model of availableModels) {
1473
+ providerMap.set(model.provider.toLowerCase(), model.provider);
1474
+ }
1475
+
1476
+ let provider = cliProvider ? providerMap.get(cliProvider.toLowerCase()) : undefined;
1477
+ if (cliProvider && !provider) {
1478
+ return {
1479
+ model: undefined,
1480
+ selector: undefined,
1481
+ warning: undefined,
1482
+ error: `Unknown provider "${cliProvider}". Run "omp models" to see available providers/models.`,
1483
+ };
1484
+ }
1485
+
1486
+ const trimmedModel = cliModel.trim();
1487
+ if (!provider) {
1488
+ const lower = trimmedModel.toLowerCase();
1489
+ // When input has provider/id format (e.g. "zai/glm-5"), prefer decomposed
1490
+ // provider+id match over flat id match. Without this, a model with id
1491
+ // "zai/glm-5" on provider "vercel-ai-gateway" wins over provider "zai"
1492
+ // with id "glm-5", because Array.find returns the first catalog hit.
1493
+ let exact = findExactModelReferenceMatch(trimmedModel, availableModels);
1494
+ if (!exact && !trimmedModel.includes(":")) {
1495
+ // CLI flags address the full catalog, so unlike the engine's canonical
1496
+ // step this lookup is unrestricted; the `:`-guard defers suffixed
1497
+ // selectors (thinking levels, ollama-style ids) to the grammar below.
1498
+ const canonicalMatch = modelRegistry.resolveCanonicalModel?.(trimmedModel, { availableOnly: false });
1499
+ if (canonicalMatch) {
1500
+ return {
1501
+ model: canonicalMatch,
1502
+ selector: modelRegistry.getCanonicalId?.(canonicalMatch) ?? trimmedModel,
1503
+ warning: undefined,
1504
+ thinkingLevel: undefined,
1505
+ error: undefined,
1506
+ };
1507
+ }
1508
+ }
1509
+ if (!exact) {
1510
+ // Flat exact id (or full selector) by catalog order: CLI resolution
1511
+ // stays deterministic across runs regardless of usage-based ranking.
1512
+ exact = availableModels.find(
1513
+ model => model.id.toLowerCase() === lower || `${model.provider}/${model.id}`.toLowerCase() === lower,
1514
+ );
1515
+ }
1516
+ if (exact) {
1517
+ return {
1518
+ model: exact,
1519
+ selector: formatModelString(exact),
1520
+ warning: undefined,
1521
+ thinkingLevel: undefined,
1522
+ error: undefined,
1523
+ };
1524
+ }
1525
+ }
1526
+
1527
+ let pattern = trimmedModel;
1528
+
1529
+ if (!provider) {
1530
+ const slashIndex = cliModel.indexOf("/");
1531
+ if (slashIndex !== -1) {
1532
+ const maybeProvider = cliModel.substring(0, slashIndex);
1533
+ const canonical = providerMap.get(maybeProvider.toLowerCase());
1534
+ if (canonical) {
1535
+ provider = canonical;
1536
+ pattern = cliModel.substring(slashIndex + 1);
1537
+ }
1538
+ }
1539
+ } else {
1540
+ const prefix = `${provider}/`;
1541
+ if (cliModel.toLowerCase().startsWith(prefix.toLowerCase())) {
1542
+ pattern = cliModel.substring(prefix.length);
1543
+ }
1544
+ }
1545
+
1546
+ if (provider) {
1547
+ const exactProviderMatch = resolveProviderModelReference(provider, pattern, availableModels);
1548
+ if (exactProviderMatch) {
1549
+ return {
1550
+ model: exactProviderMatch,
1551
+ selector: formatModelString(exactProviderMatch),
1552
+ warning: undefined,
1553
+ thinkingLevel: undefined,
1554
+ error: undefined,
1555
+ };
1556
+ }
1557
+ }
1558
+
1559
+ const candidates = provider ? availableModels.filter(model => model.provider === provider) : availableModels;
1560
+ const { model, thinkingLevel, warning, upstream } = parseModelPattern(pattern, candidates, preferences, {
1561
+ allowInvalidThinkingSelectorFallback: false,
1562
+ modelRegistry,
1563
+ });
1564
+
1565
+ if (!model) {
1566
+ const display = provider ? `${provider}/${pattern}` : cliModel;
1567
+ return {
1568
+ model: undefined,
1569
+ selector: undefined,
1570
+ thinkingLevel: undefined,
1571
+ warning,
1572
+ error: `Model "${display}" not found. Run "omp models" to see available models.`,
1573
+ };
1574
+ }
1575
+
1576
+ let selector = provider ? formatModelString(model) : undefined;
1577
+ if (!provider) {
1578
+ const canonicalCandidate = splitThinkingSuffix(pattern).base;
1579
+ if (!canonicalCandidate.includes("/")) {
1580
+ const canonicalResolved = modelRegistry.resolveCanonicalModel?.(canonicalCandidate, { availableOnly: false });
1581
+ if (canonicalResolved && canonicalResolved.provider === model.provider && canonicalResolved.id === model.id) {
1582
+ selector = modelRegistry.getCanonicalId?.(canonicalResolved) ?? canonicalCandidate;
1583
+ }
1584
+ }
1585
+ }
1586
+ if (selector !== undefined && upstream) {
1587
+ selector = `${selector}@${upstream}`;
1588
+ }
1589
+
1590
+ return {
1591
+ model,
1592
+ selector,
1593
+ thinkingLevel,
1594
+ warning,
1595
+ error: undefined,
1596
+ };
1597
+ }
1598
+
1599
+ export interface InitialModelResult {
1600
+ model: Model<Api> | undefined;
1601
+ thinkingLevel?: ThinkingLevel;
1602
+ fallbackMessage: string | undefined;
1603
+ }
1604
+
1605
+ /**
1606
+ * Find the initial model to use based on priority:
1607
+ * 1. CLI args (provider + model)
1608
+ * 2. First model from scoped models (if not continuing/resuming)
1609
+ * 3. Restored from session (if continuing/resuming)
1610
+ * 4. Saved default from settings
1611
+ * 5. First available model with valid API key
1612
+ */
1613
+ export async function findInitialModel(options: {
1614
+ cliProvider?: string;
1615
+ cliModel?: string;
1616
+ scopedModels: ScopedModel[];
1617
+ isContinuing: boolean;
1618
+ defaultProvider?: string;
1619
+ defaultModelId?: string;
1620
+ defaultThinkingSelector?: Effort;
1621
+ modelRegistry: InitialModelRegistry;
1622
+ }): Promise<InitialModelResult> {
1623
+ const {
1624
+ cliProvider,
1625
+ cliModel,
1626
+ scopedModels,
1627
+ isContinuing,
1628
+ defaultProvider,
1629
+ defaultModelId,
1630
+ defaultThinkingSelector,
1631
+ modelRegistry,
1632
+ } = options;
1633
+
1634
+ let model: Model<Api> | undefined;
1635
+ let thinkingLevel: Effort | undefined;
1636
+
1637
+ // 1. CLI args take priority
1638
+ if (cliProvider && cliModel) {
1639
+ const found = modelRegistry.find(cliProvider, cliModel);
1640
+ if (!found) {
1641
+ console.error(chalk.red(`Model ${cliProvider}/${cliModel} not found`));
1642
+ process.exit(1);
1643
+ }
1644
+ return { model: found, thinkingLevel: undefined, fallbackMessage: undefined };
1645
+ }
1646
+
1647
+ // 2. Use first model from scoped models (skip if continuing/resuming)
1648
+ if (scopedModels.length > 0 && !isContinuing) {
1649
+ const scoped = scopedModels[0];
1650
+ const scopedThinkingSelector =
1651
+ scoped.thinkingLevel === ThinkingLevel.Inherit
1652
+ ? defaultThinkingSelector
1653
+ : (scoped.thinkingLevel ?? defaultThinkingSelector);
1654
+ return {
1655
+ model: scoped.model,
1656
+ thinkingLevel:
1657
+ scopedThinkingSelector === ThinkingLevel.Off
1658
+ ? ThinkingLevel.Off
1659
+ : clampThinkingLevelForModel(scoped.model, scopedThinkingSelector),
1660
+ fallbackMessage: undefined,
1661
+ };
1662
+ }
1663
+
1664
+ // 3. Try saved default from settings
1665
+ if (defaultProvider && defaultModelId) {
1666
+ const found = modelRegistry.find(defaultProvider, defaultModelId);
1667
+ if (found) {
1668
+ model = found;
1669
+ thinkingLevel = clampThinkingLevelForModel(found, defaultThinkingSelector);
1670
+ return { model, thinkingLevel, fallbackMessage: undefined };
1671
+ }
1672
+ }
1673
+
1674
+ // 4. Try first available model with valid API key
1675
+ const availableModels = modelRegistry.getAvailable();
1676
+
1677
+ const fallback = pickDefaultAvailableModel(availableModels);
1678
+ if (fallback) {
1679
+ return { model: fallback, thinkingLevel: undefined, fallbackMessage: undefined };
1680
+ }
1681
+
1682
+ // 5. No model found
1683
+ return { model: undefined, thinkingLevel: undefined, fallbackMessage: undefined };
1684
+ }
1685
+
1686
+ /**
1687
+ * Restore model from session, with fallback to available models
1688
+ */
1689
+ export async function restoreModelFromSession(
1690
+ savedProvider: string,
1691
+ savedModelId: string,
1692
+ currentModel: Model<Api> | undefined,
1693
+ shouldPrintMessages: boolean,
1694
+ modelRegistry: RestorableModelRegistry,
1695
+ ): Promise<{ model: Model<Api> | undefined; fallbackMessage: string | undefined }> {
1696
+ const restoredModel = modelRegistry.find(savedProvider, savedModelId);
1697
+
1698
+ // Check if restored model exists and has a valid API key
1699
+ const hasApiKey = restoredModel ? !!(await modelRegistry.getApiKey(restoredModel)) : false;
1700
+
1701
+ if (restoredModel && hasApiKey) {
1702
+ if (shouldPrintMessages) {
1703
+ console.log(chalk.dim(`Restored model: ${savedProvider}/${savedModelId}`));
1704
+ }
1705
+ return { model: restoredModel, fallbackMessage: undefined };
1706
+ }
1707
+
1708
+ // Model not found or no API key - fall back
1709
+ const reason = !restoredModel ? "model no longer exists" : "no API key available";
1710
+
1711
+ if (shouldPrintMessages) {
1712
+ console.error(chalk.yellow(`Warning: Could not restore model ${savedProvider}/${savedModelId} (${reason}).`));
1713
+ }
1714
+
1715
+ // If we already have a model, use it as fallback
1716
+ if (currentModel) {
1717
+ if (shouldPrintMessages) {
1718
+ console.log(chalk.dim(`Falling back to: ${currentModel.provider}/${currentModel.id}`));
1719
+ }
1720
+ return {
1721
+ model: currentModel,
1722
+ fallbackMessage: `Could not restore model ${savedProvider}/${savedModelId} (${reason}). Using ${currentModel.provider}/${currentModel.id}.`,
1723
+ };
1724
+ }
1725
+
1726
+ // Try to find any available model
1727
+ const availableModels = modelRegistry.getAvailable();
1728
+
1729
+ const fallbackModel = pickDefaultAvailableModel(availableModels);
1730
+ if (fallbackModel) {
1731
+ if (shouldPrintMessages) {
1732
+ console.log(chalk.dim(`Falling back to: ${fallbackModel.provider}/${fallbackModel.id}`));
1733
+ }
1734
+
1735
+ return {
1736
+ model: fallbackModel,
1737
+ fallbackMessage: `Could not restore model ${savedProvider}/${savedModelId} (${reason}). Using ${fallbackModel.provider}/${fallbackModel.id}.`,
1738
+ };
1739
+ }
1740
+
1741
+ // No models available
1742
+ return { model: undefined, fallbackMessage: undefined };
1743
+ }
1744
+
1745
+ /**
1746
+ * Find a smol/fast model using the priority chain.
1747
+ * Tries exact matches first, then fuzzy matches.
1748
+ *
1749
+ * @param modelRegistry The model registry to search
1750
+ * @param savedModel Optional saved model string from settings (provider/modelId)
1751
+ * @returns The best available smol model, or undefined if none found
1752
+ */
1753
+ export async function findSmolModel(
1754
+ modelRegistry: ModelLookupRegistry,
1755
+ savedModel?: string,
1756
+ ): Promise<Model<Api> | undefined> {
1757
+ const availableModels = modelRegistry.getAvailable();
1758
+ if (availableModels.length === 0) return undefined;
1759
+
1760
+ // 1. Try saved model from settings
1761
+ if (savedModel) {
1762
+ const match = resolveModelFromString(savedModel, availableModels, undefined, modelRegistry);
1763
+ if (match) return match;
1764
+ }
1765
+
1766
+ // 2. Try priority chain
1767
+ for (const pattern of MODEL_PRIO.smol) {
1768
+ // Try exact match with provider prefix
1769
+ const providerMatch = availableModels.find(m => `${m.provider}/${m.id}`.toLowerCase() === pattern);
1770
+ if (providerMatch) return providerMatch;
1771
+
1772
+ // Try exact match first
1773
+ const exactMatch = parseModelPattern(pattern, availableModels, undefined, { modelRegistry }).model;
1774
+ if (exactMatch) return exactMatch;
1775
+
1776
+ // Try fuzzy match (substring)
1777
+ const fuzzyMatch = availableModels.find(m => m.id.toLowerCase().includes(pattern));
1778
+ if (fuzzyMatch) return fuzzyMatch;
1779
+ }
1780
+
1781
+ // 3. Fallback to first available (same as default)
1782
+ return availableModels[0];
1783
+ }
1784
+
1785
+ /**
1786
+ * Find a slow/comprehensive model using the priority chain.
1787
+ * Prioritizes reasoning and codex models for thorough analysis.
1788
+ *
1789
+ * @param modelRegistry The model registry to search
1790
+ * @param savedModel Optional saved model string from settings (provider/modelId)
1791
+ * @returns The best available slow model, or undefined if none found
1792
+ */
1793
+ export async function findSlowModel(
1794
+ modelRegistry: ModelLookupRegistry,
1795
+ savedModel?: string,
1796
+ ): Promise<Model<Api> | undefined> {
1797
+ const availableModels = modelRegistry.getAvailable();
1798
+ if (availableModels.length === 0) return undefined;
1799
+
1800
+ // 1. Try saved model from settings
1801
+ if (savedModel) {
1802
+ const match = resolveModelFromString(savedModel, availableModels, undefined, modelRegistry);
1803
+ if (match) return match;
1804
+ }
1805
+
1806
+ // 2. Try priority chain
1807
+ for (const pattern of MODEL_PRIO.slow) {
1808
+ // Try exact match first
1809
+ const exactMatch = parseModelPattern(pattern, availableModels, undefined, { modelRegistry }).model;
1810
+ if (exactMatch) return exactMatch;
1811
+
1812
+ // Try fuzzy match (substring)
1813
+ const fuzzyMatch = availableModels.find(m => m.id.toLowerCase().includes(pattern.toLowerCase()));
1814
+ if (fuzzyMatch) return fuzzyMatch;
1815
+ }
1816
+
1817
+ // 3. Fallback to first available (same as default)
1818
+ return availableModels[0];
1819
+ }