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
package/src/sdk.ts ADDED
@@ -0,0 +1,2919 @@
1
+ import {
2
+ Agent,
3
+ type AgentEvent,
4
+ type AgentMessage,
5
+ type AgentTelemetryConfig,
6
+ type AgentTool,
7
+ AppendOnlyContextManager,
8
+ type ThinkingLevel,
9
+ } from "@oh-my-pi/pi-agent-core";
10
+ import {
11
+ type Context,
12
+ type CredentialDisabledEvent,
13
+ type Message,
14
+ type Model,
15
+ type SimpleStreamOptions,
16
+ streamSimple,
17
+ } from "@oh-my-pi/pi-ai";
18
+ import type { Dialect } from "@oh-my-pi/pi-ai/dialect";
19
+ import {
20
+ getOpenAICodexTransportDetails,
21
+ prewarmOpenAICodexResponses,
22
+ } from "@oh-my-pi/pi-ai/providers/openai-codex-responses";
23
+ import { FALLBACK_DIALECT, preferredDialect } from "@oh-my-pi/pi-catalog/identity";
24
+ import type { Component } from "@oh-my-pi/pi-tui";
25
+ import { $env, $flag, getAgentDir, getProjectDir, logger, postmortem, prompt, Snowflake } from "@oh-my-pi/pi-utils";
26
+ import { INTENT_FIELD } from "@oh-my-pi/pi-wire";
27
+ import { ADVISOR_READONLY_TOOL_NAMES, discoverWatchdogFiles } from "./advisor";
28
+ import { type AsyncJob, AsyncJobManager } from "./async";
29
+ import { AutoLearnController, buildAutoLearnInstructions } from "./autolearn/controller";
30
+ import { loadCapability } from "./capability";
31
+ import { type Rule, ruleCapability, setActiveRules } from "./capability/rule";
32
+ import { bucketRules } from "./capability/rule-buckets";
33
+ import { shouldEnableAppendOnlyContext } from "./config/append-only-context-mode";
34
+ import { ModelRegistry } from "./config/model-registry";
35
+ import {
36
+ formatModelString,
37
+ getModelMatchPreferences,
38
+ parseModelPattern,
39
+ parseModelString,
40
+ pickDefaultAvailableModel,
41
+ resolveAllowedModels,
42
+ resolveModelRoleValue,
43
+ } from "./config/model-resolver";
44
+ import { loadPromptTemplates as loadPromptTemplatesInternal, type PromptTemplate } from "./config/prompt-templates";
45
+ import { Settings, type SkillsSettings } from "./config/settings";
46
+ import { CursorExecHandlers } from "./cursor";
47
+ import "./discovery";
48
+ import { initializeWithSettings } from "./discovery";
49
+ import { disposeAllKernelSessions, disposeKernelSessionsByOwner } from "./eval/py/executor";
50
+ import { defaultEvalSessionId } from "./eval/session-id";
51
+ import {
52
+ type CustomCommandsLoadResult,
53
+ type LoadedCustomCommand,
54
+ loadCustomCommands as loadCustomCommandsInternal,
55
+ } from "./extensibility/custom-commands";
56
+ import { discoverCustomToolPaths, loadCustomTools, type ToolPathWithSource } from "./extensibility/custom-tools";
57
+ import type { CustomTool, CustomToolContext, CustomToolSessionEvent } from "./extensibility/custom-tools/types";
58
+ import {
59
+ discoverAndLoadExtensions,
60
+ discoverExtensionPaths,
61
+ type ExtensionContext,
62
+ type ExtensionFactory,
63
+ ExtensionRunner,
64
+ ExtensionToolWrapper,
65
+ type ExtensionUIContext,
66
+ type LoadExtensionsResult,
67
+ loadExtensionFromFactory,
68
+ loadExtensions,
69
+ type ToolDefinition,
70
+ wrapRegisteredTools,
71
+ } from "./extensibility/extensions";
72
+ import {
73
+ loadSkills as loadSkillsInternal,
74
+ type Skill,
75
+ type SkillWarning,
76
+ setActiveSkills,
77
+ } from "./extensibility/skills";
78
+ import { type FileSlashCommand, loadSlashCommands as loadSlashCommandsInternal } from "./extensibility/slash-commands";
79
+ import type { HindsightSessionState } from "./hindsight/state";
80
+ import { LocalProtocolHandler, type LocalProtocolOptions } from "./internal-urls";
81
+ import { LSP_STARTUP_EVENT_CHANNEL, type LspStartupEvent } from "./lsp/startup-events";
82
+ import {
83
+ discoverAndLoadMCPTools,
84
+ type MCPLoadResult,
85
+ MCPManager,
86
+ MCPToolCache,
87
+ type MCPToolsLoadResult,
88
+ parseMCPToolName,
89
+ } from "./mcp";
90
+ import { MCP_CONNECTING_EVENT_CHANNEL, type McpConnectingEvent } from "./mcp/startup-events";
91
+ import { createSessionMemoryRuntimeContext, resolveMemoryBackend } from "./memory-backend";
92
+ import type { MnemopiSessionState } from "./mnemopi/state";
93
+ import asyncResultTemplate from "./prompts/tools/async-result.md" with { type: "text" };
94
+ import lateDiagnosticTemplate from "./prompts/tools/lsp-late-diagnostic.md" with { type: "text" };
95
+ import { AgentLifecycleManager } from "./registry/agent-lifecycle";
96
+ import { AgentRegistry, MAIN_AGENT_ID } from "./registry/agent-registry";
97
+ import {
98
+ collectEnvSecrets,
99
+ deobfuscateSessionContext,
100
+ loadSecrets,
101
+ obfuscateMessages,
102
+ obfuscateProviderContext,
103
+ SecretObfuscator,
104
+ } from "./secrets";
105
+ import { AgentSession } from "./session/agent-session";
106
+ import { discoverAuthStorage as discoverAuthStorageFromConfig } from "./session/auth-broker-config";
107
+ import type { AuthStorage } from "./session/auth-storage";
108
+ import {
109
+ type CustomMessage,
110
+ convertToLlm,
111
+ LSP_LATE_DIAGNOSTIC_MESSAGE_TYPE,
112
+ USER_INTERRUPT_LABEL,
113
+ wrapSteeringForModel,
114
+ } from "./session/messages";
115
+ import { getRestorableSessionModels } from "./session/session-context";
116
+ import { SessionManager } from "./session/session-manager";
117
+ import { SnapcompactInlineTransformer } from "./session/snapcompact-inline";
118
+ import { createSnapcompactSavingsRecorder } from "./session/snapcompact-savings-journal";
119
+ import { closeAllConnections } from "./ssh/connection-manager";
120
+ import { unmountAll } from "./ssh/sshfs-mount";
121
+ import {
122
+ type BuildSystemPromptResult,
123
+ buildSystemPrompt as buildSystemPromptInternal,
124
+ buildSystemPromptToolMetadata,
125
+ loadProjectContextFiles as loadContextFilesInternal,
126
+ } from "./system-prompt";
127
+ import { AgentOutputManager } from "./task/output-manager";
128
+ import {
129
+ AUTO_THINKING,
130
+ type ConfiguredThinkingLevel,
131
+ parseConfiguredThinkingLevel,
132
+ parseThinkingLevel,
133
+ resolveProvisionalAutoLevel,
134
+ resolveThinkingLevelForModel,
135
+ shouldDisableReasoning,
136
+ toReasoningEffort,
137
+ } from "./thinking";
138
+ import { countToolsForAutoDiscovery, resolveEffectiveToolDiscoveryMode } from "./tool-discovery/mode";
139
+ import {
140
+ collectDiscoverableTools,
141
+ type DiscoverableTool,
142
+ filterBySource,
143
+ formatDiscoverableToolServerSummary,
144
+ isMCPToolName,
145
+ selectDiscoverableToolNamesByServer,
146
+ summarizeDiscoverableTools,
147
+ } from "./tool-discovery/tool-index";
148
+ import {
149
+ BashTool,
150
+ BUILTIN_TOOLS,
151
+ computeEssentialBuiltinNames,
152
+ createTools,
153
+ type DeferredDiagnosticsEntry,
154
+ discoverStartupLspServers,
155
+ EditTool,
156
+ EvalTool,
157
+ FindTool,
158
+ filterInitialToolsForDiscoveryAll,
159
+ getSearchTools,
160
+ HIDDEN_TOOLS,
161
+ isImageProviderPreference,
162
+ isSearchProviderId,
163
+ isSearchProviderPreference,
164
+ type LspStartupServerInfo,
165
+ loadSshTool,
166
+ ReadTool,
167
+ ResolveTool,
168
+ renderSearchToolBm25Description,
169
+ SearchTool,
170
+ SearchToolBm25Tool,
171
+ setExcludedSearchProviders,
172
+ setPreferredImageProvider,
173
+ setPreferredSearchProvider,
174
+ type Tool,
175
+ type ToolSession,
176
+ WebSearchTool,
177
+ WriteTool,
178
+ warmupLspServers,
179
+ } from "./tools";
180
+ import { ToolContextStore } from "./tools/context";
181
+ import { getImageGenTools } from "./tools/image-gen";
182
+ import { wrapToolWithMetaNotice } from "./tools/output-meta";
183
+ import { queueResolveHandler } from "./tools/resolve";
184
+ import { ttsTool } from "./tools/tts";
185
+ import { EventBus } from "./utils/event-bus";
186
+ import { buildNamedToolChoice } from "./utils/tool-choice";
187
+ import { buildWorkspaceTree, type WorkspaceTree } from "./workspace-tree";
188
+
189
+ type AsyncResultEntry = {
190
+ jobId: string;
191
+ result: string;
192
+ job: AsyncJob | undefined;
193
+ durationMs: number | undefined;
194
+ };
195
+
196
+ type AsyncResultJobDetails = {
197
+ jobId: string;
198
+ type?: "bash" | "task";
199
+ label?: string;
200
+ durationMs?: number;
201
+ };
202
+
203
+ type AsyncResultDetails = {
204
+ jobs: AsyncResultJobDetails[];
205
+ };
206
+
207
+ type McpNotificationEntry = {
208
+ serverName: string;
209
+ uri: string;
210
+ };
211
+
212
+ function buildAsyncResultBatchMessage(entries: AsyncResultEntry[]): CustomMessage<AsyncResultDetails> | null {
213
+ if (entries.length === 0) return null;
214
+ const jobs = entries.map(entry => ({
215
+ jobId: entry.jobId,
216
+ result: entry.result,
217
+ type: entry.job?.type,
218
+ label: entry.job?.label,
219
+ durationMs: entry.durationMs,
220
+ }));
221
+ const details: AsyncResultDetails = {
222
+ jobs: jobs.map(job => ({
223
+ jobId: job.jobId,
224
+ type: job.type,
225
+ label: job.label,
226
+ durationMs: job.durationMs,
227
+ })),
228
+ };
229
+ return {
230
+ role: "custom",
231
+ customType: "async-result",
232
+ content: prompt.render(asyncResultTemplate, {
233
+ multiple: jobs.length > 1,
234
+ jobs,
235
+ }),
236
+ display: true,
237
+ attribution: "agent",
238
+ details,
239
+ timestamp: Date.now(),
240
+ };
241
+ }
242
+
243
+ type LateDiagnosticsDetails = {
244
+ files: Array<{ path: string; summary: string; errored: boolean; messages: string[] }>;
245
+ };
246
+
247
+ function buildLateDiagnosticsBatchMessage(
248
+ entries: DeferredDiagnosticsEntry[],
249
+ ): CustomMessage<LateDiagnosticsDetails> | null {
250
+ if (entries.length === 0) return null;
251
+ const files = entries.map(entry => ({
252
+ path: entry.path,
253
+ summary: entry.summary,
254
+ messages: entry.messages,
255
+ errored: entry.errored,
256
+ }));
257
+ const details: LateDiagnosticsDetails = {
258
+ files: files.map(file => ({
259
+ path: file.path,
260
+ summary: file.summary,
261
+ errored: file.errored,
262
+ messages: file.messages,
263
+ })),
264
+ };
265
+ return {
266
+ role: "custom",
267
+ customType: LSP_LATE_DIAGNOSTIC_MESSAGE_TYPE,
268
+ content: prompt.render(lateDiagnosticTemplate, {
269
+ multiple: files.length > 1,
270
+ files,
271
+ }),
272
+ display: true,
273
+ attribution: "agent",
274
+ details,
275
+ timestamp: Date.now(),
276
+ };
277
+ }
278
+
279
+ function buildMcpNotificationBatchMessage(entries: McpNotificationEntry[]): AgentMessage | null {
280
+ const resources: McpNotificationEntry[] = [];
281
+ const seen = new Set<string>();
282
+ for (const entry of entries) {
283
+ const key = `${entry.serverName}\0${entry.uri}`;
284
+ if (seen.has(key)) continue;
285
+ seen.add(key);
286
+ resources.push(entry);
287
+ }
288
+ if (resources.length === 0) return null;
289
+ const lines = [`[MCP notification] ${resources.length} resource(s) updated:`];
290
+ for (const resource of resources) {
291
+ lines.push(`- server="${resource.serverName}" uri=${resource.uri}`);
292
+ }
293
+ lines.push('Use read(path="mcp://<uri>") to inspect if relevant.');
294
+ return {
295
+ role: "user",
296
+ content: [{ type: "text", text: lines.join("\n") }],
297
+ attribution: "agent",
298
+ timestamp: Date.now(),
299
+ };
300
+ }
301
+
302
+ type DeferredMCPActivation = {
303
+ mcpDiscoveryEnabled: boolean;
304
+ explicitlyRequestedMCPToolNames: string[];
305
+ activateAllMCPTools: boolean;
306
+ };
307
+
308
+ function createPendingMCPTool(name: string): Tool {
309
+ const parsed = parseMCPToolName(name);
310
+ const serverName = parsed?.serverName;
311
+ const mcpToolName = parsed?.toolName ?? name;
312
+ const label = serverName ? `${serverName}/${mcpToolName}` : name;
313
+ const message = serverName
314
+ ? `MCP server "${serverName}" is still connecting; tool "${name}" is not yet available. Retry after the MCP connection completes.`
315
+ : `MCP discovery is still in progress; tool "${name}" is not yet available. Retry after MCP connection completes.`;
316
+ const tool: Tool & { mcpServerName?: string; mcpToolName?: string } = {
317
+ name,
318
+ label,
319
+ description: `Pending MCP tool. ${message}`,
320
+ parameters: {
321
+ type: "object",
322
+ properties: {},
323
+ additionalProperties: true,
324
+ },
325
+ approval: "write",
326
+ intent: "omit",
327
+ mcpServerName: serverName,
328
+ mcpToolName,
329
+ async execute() {
330
+ return {
331
+ content: [{ type: "text", text: message }],
332
+ details: { serverName, mcpToolName, isError: true },
333
+ isError: true,
334
+ };
335
+ },
336
+ };
337
+ return tool;
338
+ }
339
+
340
+ function collectPendingMCPToolNames(
341
+ explicitToolNames: readonly string[] | undefined,
342
+ restoredSelectedToolNames: readonly string[],
343
+ ): string[] {
344
+ const names = new Set<string>();
345
+ for (const name of explicitToolNames ?? []) {
346
+ const normalized = name.toLowerCase();
347
+ if (isMCPToolName(normalized)) names.add(normalized);
348
+ }
349
+ for (const name of restoredSelectedToolNames) {
350
+ const normalized = name.toLowerCase();
351
+ if (isMCPToolName(normalized)) names.add(normalized);
352
+ }
353
+ return [...names];
354
+ }
355
+
356
+ function logMCPLoadErrors(errors: MCPLoadResult["errors"]): void {
357
+ for (const [serverName, error] of errors) {
358
+ logger.error("MCP tool load failed", { path: `mcp:${serverName}`, error });
359
+ }
360
+ }
361
+
362
+ function applyMCPEnvironment(result: { exaApiKeys: string[] }): void {
363
+ if (result.exaApiKeys.length > 0 && !$env.EXA_API_KEY) {
364
+ Bun.env.EXA_API_KEY = result.exaApiKeys[0];
365
+ }
366
+ }
367
+
368
+ // Types
369
+ export interface CreateAgentSessionOptions {
370
+ /** Working directory for project-local discovery. Default: getProjectDir() */
371
+ cwd?: string;
372
+ /** Global config directory. Default: ~/.omp/agent */
373
+ agentDir?: string;
374
+ /** Spawns to allow. Default: "*" */
375
+ spawns?: string;
376
+
377
+ /** Auth storage for credentials. Default: discoverAuthStorage(agentDir) */
378
+ authStorage?: AuthStorage;
379
+ /** Model registry. Default: discoverModels(authStorage, agentDir) */
380
+ modelRegistry?: ModelRegistry;
381
+
382
+ /** Model to use. Default: from settings, else first available */
383
+ model?: Model;
384
+ /** Raw model pattern string (e.g. from --model CLI flag) to resolve after extensions load.
385
+ * Used when model lookup is deferred because extension-provided models aren't registered yet. */
386
+ modelPattern?: string;
387
+ /** Thinking selector. Default: from settings, else unset */
388
+ thinkingLevel?: ConfiguredThinkingLevel;
389
+ /** Models available for cycling (Ctrl+P in interactive mode) */
390
+ scopedModels?: Array<{ model: Model; thinkingLevel?: ThinkingLevel }>;
391
+
392
+ /** Provider-facing system prompt override. Replaces the fully rendered default blocks. */
393
+ systemPrompt?: string | string[] | ((defaultPrompt: string[]) => string | string[]);
394
+ /** Already-loaded custom prompt text rendered through the bundled custom system prompt template. */
395
+ customSystemPrompt?: string;
396
+ /** Already-loaded text appended through the bundled system prompt templates. */
397
+ appendSystemPrompt?: string;
398
+ /** Optional provider-facing session identifier for prompt caches and sticky auth selection.
399
+ * Keeps persisted session files isolated while reusing provider-side caches. */
400
+ providerSessionId?: string;
401
+ /** Optional provider-facing prompt cache key, distinct from request lineage. */
402
+ providerPromptCacheKey?: string;
403
+ /** Absolute wall-clock deadline in Unix epoch milliseconds. */
404
+ deadline?: number;
405
+
406
+ /** Custom tools to register (in addition to built-in tools). Accepts both CustomTool and ToolDefinition. */
407
+ customTools?: (CustomTool | ToolDefinition)[];
408
+ /** Inline extensions (merged with discovery). */
409
+ extensions?: ExtensionFactory[];
410
+ /** Additional extension paths to load (merged with discovery). */
411
+ additionalExtensionPaths?: string[];
412
+ /** Disable extension discovery (explicit paths still load). */
413
+ disableExtensionDiscovery?: boolean;
414
+ /**
415
+ * Pre-loaded extensions (skips file discovery and the per-session factory
416
+ * call). Used by the CLI when extensions are loaded early to parse custom
417
+ * flags — the same process owns the returned instances, so reusing them is
418
+ * safe.
419
+ *
420
+ * NEVER pass this across session boundaries (e.g. parent → subagent).
421
+ * `Extension` instances close over a parent-bound `ExtensionAPI` (cwd,
422
+ * eventBus, runtime), and reusing them would route tools/handlers/commands
423
+ * back through the parent. For subagents, forward
424
+ * {@link preloadedExtensionPaths} instead.
425
+ *
426
+ * @internal
427
+ */
428
+ preloadedExtensions?: LoadExtensionsResult;
429
+ /**
430
+ * Pre-discovered extension source paths. When provided, the filesystem-scan
431
+ * inside `discoverExtensionPaths()` is skipped — the session still calls
432
+ * `loadExtensions()` itself so each `Extension` is bound to THIS session's
433
+ * `ExtensionAPI` (cwd, eventBus, runtime).
434
+ *
435
+ * This is the safe pass-through for parent → subagent forwarding.
436
+ */
437
+ preloadedExtensionPaths?: string[];
438
+ /**
439
+ * Pre-discovered custom-tool source paths from `.omp/tools/`, `.claude/tools/`,
440
+ * plugins, etc. When provided, the filesystem-scan inside
441
+ * `discoverCustomToolPaths()` is skipped — subagents inherit the parent's
442
+ * scan result and call `loadCustomTools()` themselves so each session binds
443
+ * tools to its OWN `CustomToolAPI` (cwd, exec, pushPendingAction, UI).
444
+ *
445
+ * Forwarding the loaded `LoadedCustomTool[]` instances directly would reuse
446
+ * the parent's session-bound API and route tool execution back through the
447
+ * parent — wrong for isolated tasks and for pending-action routing.
448
+ */
449
+ preloadedCustomToolPaths?: ToolPathWithSource[];
450
+
451
+ /** Shared event bus for tool/extension communication. Default: creates new bus. */
452
+ eventBus?: EventBus;
453
+
454
+ /** Skills. Default: discovered from multiple locations */
455
+ skills?: Skill[];
456
+ /** Rules. Default: discovered from multiple locations */
457
+ rules?: Rule[];
458
+ /** Context files (AGENTS.md content). Default: discovered walking up from cwd */
459
+ contextFiles?: Array<{ path: string; content: string }>;
460
+ /** Pre-built workspace tree (skips re-scanning; passed by parents to subagents). */
461
+ workspaceTree?: WorkspaceTree;
462
+ /** Prompt templates. Default: discovered from cwd/.omp/prompts/ + agentDir/prompts/ */
463
+ promptTemplates?: PromptTemplate[];
464
+ /** File-based slash commands. Default: discovered from commands/ directories */
465
+ slashCommands?: FileSlashCommand[];
466
+
467
+ /** Enable MCP server discovery from .mcp.json files. Default: true */
468
+ enableMCP?: boolean;
469
+ /** Existing MCP manager to reuse (skips discovery, propagates to toolSession). */
470
+ mcpManager?: MCPManager;
471
+
472
+ /** Enable LSP integration (tool, formatting, diagnostics, warmup). Default: true */
473
+ enableLsp?: boolean;
474
+ /** Skip Python kernel availability check and prelude warmup */
475
+ skipPythonPreflight?: boolean;
476
+ /** Tool names explicitly requested (enables disabled-by-default tools) */
477
+ toolNames?: string[];
478
+
479
+ /** Output schema for structured completion (subagents) */
480
+ outputSchema?: unknown;
481
+ /** Whether to include the yield tool by default */
482
+ requireYieldTool?: boolean;
483
+ /** Task recursion depth (for subagent sessions). Default: 0 */
484
+ taskDepth?: number;
485
+ /** Parent Hindsight state to alias for subagent memory tools. */
486
+ parentHindsightSessionState?: HindsightSessionState;
487
+ /** Parent Mnemopi state to alias for subagent memory tools. */
488
+ parentMnemopiSessionState?: MnemopiSessionState;
489
+ /** Pre-allocated agent identity for IRC routing. Default: "Main" for top-level, parentTaskPrefix-derived for sub. */
490
+ agentId?: string;
491
+ /** Display name for the agent in IRC. Default: "main" or "sub". */
492
+ agentDisplayName?: string;
493
+ /** Optional shared agent registry for IRC routing. Default: AgentRegistry.global(). */
494
+ agentRegistry?: AgentRegistry;
495
+ /** Parent task ID prefix for nested artifact naming (e.g., "Extensions") */
496
+ parentTaskPrefix?: string;
497
+ /**
498
+ * Registry id of the spawning agent, recorded as this subagent's parent in
499
+ * the agent registry. Distinct from `parentTaskPrefix`, which is this agent's
500
+ * own artifact/output-id prefix (the executor passes the child's own id
501
+ * there, so it must never double as the parent link). Undefined for the
502
+ * top-level "Main" session, which has no parent.
503
+ */
504
+ parentAgentId?: string;
505
+ /** Inherited eval executor session id for subagents sharing parent eval state. */
506
+ parentEvalSessionId?: string;
507
+
508
+ /** Session manager. Default: session stored under the configured agentDir sessions root */
509
+ sessionManager?: SessionManager;
510
+
511
+ /** Override local:// protocol options for subagent local:// sharing. Default: uses the session's own artifacts dir and session ID. */
512
+ localProtocolOptions?: LocalProtocolOptions;
513
+
514
+ /** Settings instance. Default: Settings.init({ cwd, agentDir }) */
515
+ settings?: Settings;
516
+ /**
517
+ * Legacy alias for `settings`. Older Pi extensions pass SettingsManager.create(...)
518
+ * through this field; accept it so their SDK calls keep the configured settings.
519
+ */
520
+ settingsManager?: Settings | Promise<Settings>;
521
+
522
+ /** Whether UI is available (enables interactive tools like ask). Default: false */
523
+ hasUI?: boolean;
524
+
525
+ /**
526
+ * Opt-in OpenTelemetry instrumentation forwarded to the underlying Agent.
527
+ * Passing `{}` enables the loop's GenAI-semantic-convention spans. See
528
+ * {@link AgentTelemetryConfig} for the full surface (hooks, content capture,
529
+ * cost estimator, agent identity).
530
+ *
531
+ * Safe to enable without an OTEL SDK registered in the host: the
532
+ * `@opentelemetry/api` package returns a no-op tracer in that case.
533
+ */
534
+ telemetry?: AgentTelemetryConfig;
535
+
536
+ /**
537
+ * Fired once, when the agent loop hands its first request to the provider
538
+ * transport (i.e. the `streamFn` wrapper is first invoked). Used to measure
539
+ * subagent launch latency — the boundary between "session built" and "model
540
+ * call dispatched". This is the loop's dispatch point, slightly before the
541
+ * actual provider HTTP call (per-request prep, identical across all
542
+ * requests, follows it), which is the right granularity for launch timing.
543
+ */
544
+ onFirstChatDispatch?: () => void;
545
+
546
+ /** Whether to auto-approve all tool calls (--auto-approve CLI flag). Default: false */
547
+ autoApprove?: boolean;
548
+ }
549
+
550
+ /** Result from createAgentSession */
551
+ export interface CreateAgentSessionResult {
552
+ /** The created session */
553
+ session: AgentSession;
554
+ /** Extensions result (loaded extensions + runtime) */
555
+ extensionsResult: LoadExtensionsResult;
556
+ /** Update tool UI context (interactive mode) */
557
+ setToolUIContext: (uiContext: ExtensionUIContext, hasUI: boolean) => void;
558
+ /** MCP manager for server lifecycle management (undefined if MCP disabled) */
559
+ mcpManager?: MCPManager;
560
+ /** Warning if session was restored with a different model than saved */
561
+ modelFallbackMessage?: string;
562
+ /** LSP servers detected for startup; warmup may continue in the background */
563
+ lspServers?: LspStartupServerInfo[];
564
+ /** Shared event bus for tool/extension communication */
565
+ eventBus: EventBus;
566
+ }
567
+
568
+ export type DialectFormat = "auto" | "native" | Dialect;
569
+
570
+ export function resolveDialect(
571
+ format: DialectFormat,
572
+ model: (Pick<Model, "supportsTools"> & Partial<Pick<Model, "id">>) | undefined,
573
+ ): Dialect | undefined {
574
+ if (format === "native") return undefined;
575
+ if (format === "auto") {
576
+ if (model?.supportsTools !== false) return undefined;
577
+ if (!model.id) return "glm";
578
+ const preferred = preferredDialect(model.id);
579
+ return preferred === FALLBACK_DIALECT ? "glm" : preferred;
580
+ }
581
+ return format;
582
+ }
583
+
584
+ // Re-exports
585
+
586
+ export type { PromptTemplate } from "./config/prompt-templates";
587
+ export { Settings, type SkillsSettings } from "./config/settings";
588
+ export type { CustomCommand, CustomCommandFactory } from "./extensibility/custom-commands/types";
589
+ export type { CustomTool, CustomToolFactory } from "./extensibility/custom-tools/types";
590
+ export type * from "./extensibility/extensions";
591
+ export type { Skill } from "./extensibility/skills";
592
+ export type { FileSlashCommand } from "./extensibility/slash-commands";
593
+ export type { MCPManager, MCPServerConfig, MCPServerConnection, MCPToolsLoadResult } from "./mcp";
594
+ export type { Tool } from "./tools";
595
+ export { buildDirectoryTree, buildWorkspaceTree, type DirectoryTree, type WorkspaceTree } from "./workspace-tree";
596
+
597
+ export {
598
+ // Individual tool classes (for custom usage)
599
+ BashTool,
600
+ // Tool classes and factories
601
+ BUILTIN_TOOLS,
602
+ createTools,
603
+ EditTool,
604
+ EvalTool,
605
+ FindTool,
606
+ HIDDEN_TOOLS,
607
+ loadSshTool,
608
+ ReadTool,
609
+ ResolveTool,
610
+ SearchTool,
611
+ type ToolSession,
612
+ WebSearchTool,
613
+ WriteTool,
614
+ };
615
+
616
+ // Helper Functions
617
+
618
+ // Discovery Functions
619
+
620
+ /**
621
+ * Create an AuthStorage instance.
622
+ *
623
+ * Default: local SQLite store at `<agentDir>/agent.db`.
624
+ *
625
+ * Broker mode: when `OMP_AUTH_BROKER_URL` is set, credentials are pulled from
626
+ * a remote auth-broker over the wire. Refresh tokens never leave the broker;
627
+ * the client receives access tokens with `refresh = "__remote__"` and calls
628
+ * back into the broker through the {@link AuthStorageOptions.refreshOAuthCredential}
629
+ * override to re-mint access tokens when needed.
630
+ *
631
+ * Delegates to {@link ./session/auth-broker-config} so the TUI and the catalog
632
+ * generator share the same credential-discovery logic.
633
+ */
634
+ export async function discoverAuthStorage(agentDir: string = getAgentDir()): Promise<AuthStorage> {
635
+ return discoverAuthStorageFromConfig(agentDir);
636
+ }
637
+
638
+ /**
639
+ * Discover extensions from cwd.
640
+ */
641
+ export async function discoverExtensions(cwd?: string): Promise<LoadExtensionsResult> {
642
+ const resolvedCwd = cwd ?? getProjectDir();
643
+
644
+ return discoverAndLoadExtensions([], resolvedCwd);
645
+ }
646
+
647
+ /**
648
+ * Path-only counterpart of {@link loadSessionExtensions}: the FS-heavy scan
649
+ * without the per-session module load. Subagents reuse the parent's path list
650
+ * (cached on {@link ToolSession.extensionPaths}) and rebuild Extension
651
+ * instances themselves so each session's `ExtensionAPI` (cwd, eventBus,
652
+ * runtime) is its own.
653
+ */
654
+ export async function discoverSessionExtensionPaths(
655
+ options: Pick<CreateAgentSessionOptions, "disableExtensionDiscovery" | "additionalExtensionPaths">,
656
+ cwd: string,
657
+ settings: Settings,
658
+ ): Promise<string[]> {
659
+ if (options.disableExtensionDiscovery) {
660
+ return options.additionalExtensionPaths ?? [];
661
+ }
662
+ const configuredPaths = [...(options.additionalExtensionPaths ?? []), ...(settings.get("extensions") ?? [])];
663
+ const disabledExtensionIds = settings.get("disabledExtensions") ?? [];
664
+ return discoverExtensionPaths(configuredPaths, cwd, disabledExtensionIds);
665
+ }
666
+
667
+ /**
668
+ * Load the discovered/configured extensions for a session — everything {@link
669
+ * createAgentSession} would load except the inline factory extensions it appends
670
+ * itself. Extracted so the CLI can resolve extension-registered flags (and thus
671
+ * classify `@file` arguments extension-aware) *before* a session — and its
672
+ * terminal breadcrumb — is created, then hand the result back through
673
+ * {@link CreateAgentSessionOptions.preloadedExtensions} so the work is not
674
+ * repeated. Keep this the single source of the discovery branch logic.
675
+ */
676
+ export async function loadSessionExtensions(
677
+ options: Pick<CreateAgentSessionOptions, "disableExtensionDiscovery" | "additionalExtensionPaths">,
678
+ cwd: string,
679
+ settings: Settings,
680
+ eventBus: EventBus,
681
+ ): Promise<LoadExtensionsResult> {
682
+ const paths = await discoverSessionExtensionPaths(options, cwd, settings);
683
+ const result = await logger.time("loadExtensions", loadExtensions, paths, cwd, eventBus);
684
+ for (const { path, error } of result.errors) {
685
+ logger.error("Failed to load extension", { path, error });
686
+ }
687
+ return result;
688
+ }
689
+
690
+ /**
691
+ * Load discovered/configured extensions and register their providers into
692
+ * `modelRegistry`, then discover the dynamic provider catalogs. One-shot CLIs
693
+ * (`omp bench`, dry-balance) build a bare {@link ModelRegistry} that only knows
694
+ * built-in catalog providers; without this, providers contributed by an
695
+ * extension (e.g. a custom OpenAI-compatible provider under
696
+ * `~/.omp/agent/extensions/`) never reach model resolution. Mirrors the
697
+ * session / `omp models` path: drain the queued provider registrations, then
698
+ * `refreshRuntimeProviders` so dynamically-discovered models exist before
699
+ * selectors are resolved.
700
+ */
701
+ export async function loadCliExtensionProviders(
702
+ modelRegistry: ModelRegistry,
703
+ settings: Settings,
704
+ cwd: string,
705
+ options: Pick<CreateAgentSessionOptions, "disableExtensionDiscovery" | "additionalExtensionPaths"> = {},
706
+ ): Promise<void> {
707
+ const eventBus = new EventBus();
708
+ const extensionsResult = await loadSessionExtensions(options, cwd, settings, eventBus);
709
+ const activeSources = extensionsResult.extensions.map(extension => extension.path);
710
+ modelRegistry.syncExtensionSources(activeSources);
711
+ for (const sourceId of new Set(activeSources)) {
712
+ modelRegistry.clearSourceRegistrations(sourceId);
713
+ }
714
+ for (const { name, config, sourceId } of extensionsResult.runtime.pendingProviderRegistrations) {
715
+ modelRegistry.registerProvider(name, config, sourceId);
716
+ }
717
+ extensionsResult.runtime.pendingProviderRegistrations = [];
718
+ await modelRegistry.refreshRuntimeProviders();
719
+ }
720
+
721
+ /**
722
+ * Discover skills from cwd and agentDir.
723
+ */
724
+ export async function discoverSkills(
725
+ cwd?: string,
726
+ _agentDir?: string,
727
+ settings?: SkillsSettings,
728
+ ): Promise<{ skills: Skill[]; warnings: SkillWarning[] }> {
729
+ return await loadSkillsInternal({
730
+ ...settings,
731
+ cwd: cwd ?? getProjectDir(),
732
+ });
733
+ }
734
+
735
+ /**
736
+ * Discover context files (AGENTS.md) walking up from cwd.
737
+ * Returns files sorted by depth (farther from cwd first, so closer files appear last/more prominent).
738
+ */
739
+ export async function discoverContextFiles(
740
+ cwd?: string,
741
+ _agentDir?: string,
742
+ ): Promise<Array<{ path: string; content: string; depth?: number }>> {
743
+ return await loadContextFilesInternal({
744
+ cwd: cwd ?? getProjectDir(),
745
+ });
746
+ }
747
+
748
+ /**
749
+ * Discover prompt templates from cwd and agentDir.
750
+ */
751
+ export async function discoverPromptTemplates(cwd?: string, agentDir?: string): Promise<PromptTemplate[]> {
752
+ return await loadPromptTemplatesInternal({
753
+ cwd: cwd ?? getProjectDir(),
754
+ agentDir: agentDir ?? getAgentDir(),
755
+ });
756
+ }
757
+
758
+ /**
759
+ * Discover file-based slash commands from commands/ directories.
760
+ */
761
+ export async function discoverSlashCommands(cwd?: string): Promise<FileSlashCommand[]> {
762
+ return loadSlashCommandsInternal({ cwd: cwd ?? getProjectDir() });
763
+ }
764
+
765
+ /**
766
+ * Discover custom commands (TypeScript slash commands) from cwd and agentDir.
767
+ */
768
+ export async function discoverCustomTSCommands(cwd?: string, agentDir?: string): Promise<CustomCommandsLoadResult> {
769
+ const resolvedCwd = cwd ?? getProjectDir();
770
+ const resolvedAgentDir = agentDir ?? getAgentDir();
771
+
772
+ return loadCustomCommandsInternal({
773
+ cwd: resolvedCwd,
774
+ agentDir: resolvedAgentDir,
775
+ });
776
+ }
777
+
778
+ /**
779
+ * Discover MCP servers from .mcp.json files.
780
+ * Returns the manager and loaded tools.
781
+ */
782
+ export async function discoverMCPServers(cwd?: string): Promise<MCPToolsLoadResult> {
783
+ const resolvedCwd = cwd ?? getProjectDir();
784
+ return discoverAndLoadMCPTools(resolvedCwd);
785
+ }
786
+
787
+ // API Key Helpers
788
+
789
+ // System Prompt
790
+
791
+ export interface BuildSystemPromptOptions {
792
+ tools?: Tool[];
793
+ skills?: Skill[];
794
+ contextFiles?: Array<{ path: string; content: string }>;
795
+ cwd?: string;
796
+ customPrompt?: string;
797
+ appendPrompt?: string;
798
+ inlineToolDescriptors?: boolean;
799
+ }
800
+
801
+ /**
802
+ * Build the default provider-facing system prompt blocks.
803
+ *
804
+ * The returned `systemPrompt` preserves the stable harness prompt and dynamic project context
805
+ * as separate entries so providers can cache prompt prefixes without concatenating blocks.
806
+ */
807
+ export async function buildSystemPrompt(options: BuildSystemPromptOptions = {}): Promise<BuildSystemPromptResult> {
808
+ return await buildSystemPromptInternal({
809
+ cwd: options.cwd,
810
+ customPrompt: options.customPrompt,
811
+ skills: options.skills,
812
+ contextFiles: options.contextFiles,
813
+ appendSystemPrompt: options.appendPrompt,
814
+ inlineToolDescriptors: options.inlineToolDescriptors,
815
+ });
816
+ }
817
+
818
+ // Internal Helpers
819
+
820
+ function createCustomToolContext(ctx: ExtensionContext): CustomToolContext {
821
+ return {
822
+ sessionManager: ctx.sessionManager,
823
+ modelRegistry: ctx.modelRegistry,
824
+ model: ctx.model,
825
+ isIdle: ctx.isIdle,
826
+ hasQueuedMessages: ctx.hasPendingMessages,
827
+ abort: ctx.abort,
828
+ };
829
+ }
830
+
831
+ function isCustomTool(tool: CustomTool | ToolDefinition): tool is CustomTool {
832
+ // To distinguish, we mark converted tools with a hidden symbol property.
833
+ // If the tool doesn't have this marker, it's a CustomTool that needs conversion.
834
+ return !(tool as any).__isToolDefinition;
835
+ }
836
+
837
+ function isLegacyBuiltinToolDefinition(tool: CustomTool | ToolDefinition): boolean {
838
+ return !isCustomTool(tool) && "__ompLegacyBuiltinTool" in tool && tool.__ompLegacyBuiltinTool === true;
839
+ }
840
+
841
+ const TOOL_DEFINITION_MARKER = Symbol("__isToolDefinition");
842
+
843
+ /** Matches the truncation applied to per-server instructions inside `rebuildSystemPrompt`. */
844
+ const MAX_MCP_INSTRUCTIONS_LENGTH = 4000;
845
+
846
+ let sshCleanupRegistered = false;
847
+
848
+ async function cleanupSshResources(): Promise<void> {
849
+ const results = await Promise.allSettled([closeAllConnections(), unmountAll()]);
850
+ for (const result of results) {
851
+ if (result.status === "rejected") {
852
+ logger.warn("SSH cleanup failed", { error: String(result.reason) });
853
+ }
854
+ }
855
+ }
856
+
857
+ function registerSshCleanup(): void {
858
+ if (sshCleanupRegistered) return;
859
+ sshCleanupRegistered = true;
860
+ postmortem.register("ssh-cleanup", cleanupSshResources);
861
+ }
862
+
863
+ let pythonCleanupRegistered = false;
864
+
865
+ function registerPythonCleanup(): void {
866
+ if (pythonCleanupRegistered) return;
867
+ pythonCleanupRegistered = true;
868
+ postmortem.register("python-cleanup", disposeAllKernelSessions);
869
+ }
870
+
871
+ function customToolToDefinition(tool: CustomTool): ToolDefinition {
872
+ const definition: ToolDefinition & { [TOOL_DEFINITION_MARKER]: true } = {
873
+ name: tool.name,
874
+ label: tool.label,
875
+ description: tool.description,
876
+ parameters: tool.parameters,
877
+ hidden: tool.hidden,
878
+ deferrable: tool.deferrable,
879
+ approval: typeof tool.approval === "function" ? tool.approval.bind(tool) : tool.approval,
880
+ mcpServerName: tool.mcpServerName,
881
+ mcpToolName: tool.mcpToolName,
882
+ execute: (toolCallId, params, signal, onUpdate, ctx) =>
883
+ tool.execute(toolCallId, params, onUpdate, createCustomToolContext(ctx), signal),
884
+ onSession: tool.onSession ? (event, ctx) => tool.onSession?.(event, createCustomToolContext(ctx)) : undefined,
885
+ renderCall: tool.renderCall,
886
+ renderResult: tool.renderResult
887
+ ? (result, options, theme): Component => {
888
+ const component = tool.renderResult?.(
889
+ result,
890
+ { expanded: options.expanded, isPartial: options.isPartial, spinnerFrame: options.spinnerFrame },
891
+ theme,
892
+ );
893
+ // Return empty component if undefined to match Component type requirement
894
+ return component ?? ({ render: () => [] } as unknown as Component);
895
+ }
896
+ : undefined,
897
+ [TOOL_DEFINITION_MARKER]: true,
898
+ };
899
+ return definition;
900
+ }
901
+
902
+ function createCustomToolsExtension(tools: CustomTool[]): ExtensionFactory {
903
+ return api => {
904
+ for (const tool of tools) {
905
+ api.registerTool(customToolToDefinition(tool));
906
+ }
907
+
908
+ const runOnSession = async (event: CustomToolSessionEvent, ctx: ExtensionContext) => {
909
+ for (const tool of tools) {
910
+ if (!tool.onSession) continue;
911
+ try {
912
+ await tool.onSession(event, createCustomToolContext(ctx));
913
+ } catch (err) {
914
+ logger.warn("Custom tool onSession error", { tool: tool.name, error: String(err) });
915
+ }
916
+ }
917
+ };
918
+
919
+ api.on("session_start", async (_event, ctx) =>
920
+ runOnSession({ reason: "start", previousSessionFile: undefined }, ctx),
921
+ );
922
+ api.on("session_switch", async (event, ctx) =>
923
+ runOnSession({ reason: "switch", previousSessionFile: event.previousSessionFile }, ctx),
924
+ );
925
+ api.on("session_branch", async (event, ctx) =>
926
+ runOnSession({ reason: "branch", previousSessionFile: event.previousSessionFile }, ctx),
927
+ );
928
+ api.on("session_tree", async (_event, ctx) =>
929
+ runOnSession({ reason: "tree", previousSessionFile: undefined }, ctx),
930
+ );
931
+ api.on("session_shutdown", async (_event, ctx) =>
932
+ runOnSession({ reason: "shutdown", previousSessionFile: undefined }, ctx),
933
+ );
934
+ api.on("auto_compaction_start", async (event, ctx) =>
935
+ runOnSession({ reason: "auto_compaction_start", trigger: event.reason, action: event.action }, ctx),
936
+ );
937
+ api.on("auto_compaction_end", async (event, ctx) =>
938
+ runOnSession(
939
+ {
940
+ reason: "auto_compaction_end",
941
+ action: event.action,
942
+ result: event.result,
943
+ aborted: event.aborted,
944
+ willRetry: event.willRetry,
945
+ errorMessage: event.errorMessage,
946
+ },
947
+ ctx,
948
+ ),
949
+ );
950
+ api.on("auto_retry_start", async (event, ctx) =>
951
+ runOnSession(
952
+ {
953
+ reason: "auto_retry_start",
954
+ attempt: event.attempt,
955
+ maxAttempts: event.maxAttempts,
956
+ delayMs: event.delayMs,
957
+ errorMessage: event.errorMessage,
958
+ },
959
+ ctx,
960
+ ),
961
+ );
962
+ api.on("auto_retry_end", async (event, ctx) =>
963
+ runOnSession(
964
+ {
965
+ reason: "auto_retry_end",
966
+ success: event.success,
967
+ attempt: event.attempt,
968
+ finalError: event.finalError,
969
+ },
970
+ ctx,
971
+ ),
972
+ );
973
+ api.on("ttsr_triggered", async (event, ctx) =>
974
+ runOnSession({ reason: "ttsr_triggered", rules: event.rules }, ctx),
975
+ );
976
+ api.on("todo_reminder", async (event, ctx) =>
977
+ runOnSession(
978
+ {
979
+ reason: "todo_reminder",
980
+ todos: event.todos,
981
+ attempt: event.attempt,
982
+ maxAttempts: event.maxAttempts,
983
+ },
984
+ ctx,
985
+ ),
986
+ );
987
+ };
988
+ }
989
+
990
+ // Factory
991
+
992
+ /**
993
+ * Build LoadedCustomCommand entries for all MCP prompts across connected servers.
994
+ * These are re-created whenever prompts change (setOnPromptsChanged callback).
995
+ */
996
+ function buildMCPPromptCommands(manager: MCPManager): LoadedCustomCommand[] {
997
+ const commands: LoadedCustomCommand[] = [];
998
+ for (const serverName of manager.getConnectedServers()) {
999
+ const prompts = manager.getServerPrompts(serverName);
1000
+ if (!prompts?.length) continue;
1001
+ for (const prompt of prompts) {
1002
+ const commandName = `${serverName}:${prompt.name}`;
1003
+ commands.push({
1004
+ path: `mcp:${commandName}`,
1005
+ resolvedPath: `mcp:${commandName}`,
1006
+ source: "bundled",
1007
+ command: {
1008
+ name: commandName,
1009
+ description: prompt.description ?? `MCP prompt from ${serverName}`,
1010
+ async execute(args: string[]) {
1011
+ const promptArgs: Record<string, string> = {};
1012
+ for (const arg of args) {
1013
+ const eqIdx = arg.indexOf("=");
1014
+ if (eqIdx > 0) {
1015
+ promptArgs[arg.slice(0, eqIdx)] = arg.slice(eqIdx + 1);
1016
+ }
1017
+ }
1018
+ const result = await manager.executePrompt(serverName, prompt.name, promptArgs);
1019
+ if (!result) return "";
1020
+ const parts: string[] = [];
1021
+ for (const msg of result.messages) {
1022
+ const contentItems = Array.isArray(msg.content) ? msg.content : [msg.content];
1023
+ for (const item of contentItems) {
1024
+ if (item.type === "text") {
1025
+ parts.push(item.text);
1026
+ } else if (item.type === "resource") {
1027
+ const resource = item.resource;
1028
+ if (resource.text) parts.push(resource.text);
1029
+ }
1030
+ }
1031
+ }
1032
+ return parts.join("\n\n");
1033
+ },
1034
+ },
1035
+ });
1036
+ }
1037
+ }
1038
+ return commands;
1039
+ }
1040
+ /**
1041
+ * Create an AgentSession with the specified options.
1042
+ *
1043
+ * @example
1044
+ * ```typescript
1045
+ * // Minimal - uses defaults
1046
+ * const { session } = await createAgentSession();
1047
+ *
1048
+ * // With explicit model
1049
+ * import { getModel } from '@oh-my-pi/pi-ai';
1050
+ * const { session } = await createAgentSession({
1051
+ * model: getModel('anthropic', 'claude-opus-4-5'),
1052
+ * thinkingLevel: 'high',
1053
+ * });
1054
+ *
1055
+ * // Continue previous session
1056
+ * const { session, modelFallbackMessage } = await createAgentSession({
1057
+ * continueSession: true,
1058
+ * });
1059
+ *
1060
+ * // Full control
1061
+ * const { session } = await createAgentSession({
1062
+ * model: myModel,
1063
+ * getApiKey: async () => Bun.env.MY_KEY,
1064
+ * systemPrompt: ['You are helpful.'],
1065
+ * tools: codingTools({ cwd: getProjectDir() }),
1066
+ * skills: [],
1067
+ * sessionManager: SessionManager.inMemory(),
1068
+ * });
1069
+ * ```
1070
+ */
1071
+ export async function createAgentSession(options: CreateAgentSessionOptions = {}): Promise<CreateAgentSessionResult> {
1072
+ const cwd = options.cwd ?? getProjectDir();
1073
+ const agentDir = options.agentDir ?? getAgentDir();
1074
+ const eventBus = options.eventBus ?? new EventBus();
1075
+
1076
+ registerSshCleanup();
1077
+ registerPythonCleanup();
1078
+
1079
+ // Pin authStorage to modelRegistry.authStorage: ModelRegistry.getApiKey() routes refresh
1080
+ // failures through that instance, so any divergent storage handed to the bridge / mcpManager
1081
+ // / session would silently miss credential_disabled events.
1082
+ const modelRegistry =
1083
+ options.modelRegistry ??
1084
+ new ModelRegistry(options.authStorage ?? (await logger.time("discoverModels", discoverAuthStorage, agentDir)));
1085
+ // Track whether we internally created the authStorage so we can close it
1086
+ // if construction fails before the session takes ownership.
1087
+ const ownsAuthStorage = !options.authStorage && !options.modelRegistry;
1088
+ const authStorage = modelRegistry.authStorage;
1089
+ if (options.authStorage && options.authStorage !== authStorage) {
1090
+ throw new Error(
1091
+ "options.authStorage and options.modelRegistry.authStorage must be the same instance when both are provided",
1092
+ );
1093
+ }
1094
+ // Subscribe before any getApiKey() call so startup model probes can't fire a
1095
+ // credential_disabled event past us. An embedder's constructor handler makes the
1096
+ // listener set non-empty from construction, which defeats AuthStorage's no-listener
1097
+ // buffer — so we can't rely on it to catch startup events for the extension runner.
1098
+ const startupCredentialDisabledEvents: CredentialDisabledEvent[] = [];
1099
+ let credentialDisabledTarget: ExtensionRunner | undefined;
1100
+ const unsubscribeCredentialDisabled: (() => void) | undefined = authStorage.onCredentialDisabled(event => {
1101
+ if (credentialDisabledTarget) {
1102
+ // Discard return: any handler error is routed through runner.onError listeners.
1103
+ void credentialDisabledTarget.emitCredentialDisabled(event);
1104
+ } else {
1105
+ startupCredentialDisabledEvents.push(event);
1106
+ }
1107
+ });
1108
+ const settings = await (options.settings ??
1109
+ options.settingsManager ??
1110
+ logger.time("settings", Settings.init, { cwd, agentDir }));
1111
+ logger.time("initializeWithSettings", initializeWithSettings, settings);
1112
+ if (!options.modelRegistry) {
1113
+ modelRegistry.refreshInBackground();
1114
+ }
1115
+ // Kick off workspace tree discovery early. The native workspace scan returns
1116
+ // both the rendered-tree input and the AGENTS.md directory-context index, so
1117
+ // startup does not perform a second recursive filesystem search. Subagents
1118
+ // inherit the parent's resolved values via options.
1119
+ const STARTUP_SCAN_DEADLINE_MS = 5000;
1120
+ const workspaceTreePromise: Promise<WorkspaceTree> = options.workspaceTree
1121
+ ? Promise.resolve(options.workspaceTree)
1122
+ : logger.time("buildWorkspaceTree", () => buildWorkspaceTree(cwd, { timeoutMs: STARTUP_SCAN_DEADLINE_MS }));
1123
+ workspaceTreePromise.catch(() => {});
1124
+
1125
+ // Independent discoveries that depend only on cwd/agentDir — kicked off in parallel and awaited
1126
+ // at their respective consumer sites. Their work can overlap with model resolution, secret loading,
1127
+ // session-context build, tool creation, MCP discovery, and extension discovery.
1128
+ const contextFilesPromise = options.contextFiles
1129
+ ? Promise.resolve(options.contextFiles)
1130
+ : logger.time("discoverContextFiles", discoverContextFiles, cwd, agentDir);
1131
+ contextFilesPromise.catch(() => {});
1132
+ const watchdogFilesPromise = logger.time("discoverWatchdogFiles", () => discoverWatchdogFiles(cwd, agentDir));
1133
+ watchdogFilesPromise.catch(() => {});
1134
+ const promptTemplatesPromise = options.promptTemplates
1135
+ ? Promise.resolve(options.promptTemplates)
1136
+ : logger.time("discoverPromptTemplates", discoverPromptTemplates, cwd, agentDir);
1137
+ promptTemplatesPromise.catch(() => {});
1138
+ const slashCommandsPromise = options.slashCommands
1139
+ ? Promise.resolve(options.slashCommands)
1140
+ : logger.time("discoverSlashCommands", discoverSlashCommands, cwd);
1141
+ slashCommandsPromise.catch(() => {});
1142
+ const skillsSettings = settings.getGroup("skills");
1143
+ const disabledExtensionIds = settings.get("disabledExtensions") ?? [];
1144
+ const discoveredSkillsPromise =
1145
+ options.skills === undefined
1146
+ ? logger.time("discoverSkills", discoverSkills, cwd, agentDir, {
1147
+ ...skillsSettings,
1148
+ disabledExtensions: disabledExtensionIds,
1149
+ })
1150
+ : undefined;
1151
+ discoveredSkillsPromise?.catch(() => {});
1152
+
1153
+ // Initialize provider preferences from settings
1154
+ const excludedWebSearchProviders = settings.get("providers.webSearchExclude");
1155
+ if (Array.isArray(excludedWebSearchProviders)) {
1156
+ setExcludedSearchProviders(excludedWebSearchProviders.filter(isSearchProviderId));
1157
+ }
1158
+
1159
+ const webSearchProvider = settings.get("providers.webSearch");
1160
+ if (typeof webSearchProvider === "string" && isSearchProviderPreference(webSearchProvider)) {
1161
+ setPreferredSearchProvider(webSearchProvider);
1162
+ }
1163
+
1164
+ const imageProvider = settings.get("providers.image");
1165
+ if (isImageProviderPreference(imageProvider)) {
1166
+ setPreferredImageProvider(imageProvider);
1167
+ }
1168
+
1169
+ const sessionManager =
1170
+ options.sessionManager ??
1171
+ logger.time("sessionManager", () =>
1172
+ SessionManager.create(cwd, SessionManager.getDefaultSessionDir(cwd, agentDir)),
1173
+ );
1174
+ const providerSessionId = options.providerSessionId ?? sessionManager.getSessionId();
1175
+ // Startup model *selection* only needs to know whether auth is configured for
1176
+ // a candidate's provider — never the resolved key bytes. Use the synchronous,
1177
+ // side-effect-free probe (`hasConfiguredAuth`): it refreshes no OAuth tokens,
1178
+ // executes no `!command` keys, and issues no auth-broker requests. Resolving the
1179
+ // real key here (`getApiKey`) blocks resume on those network paths — a slow or
1180
+ // unreachable OAuth/broker endpoint stalls startup for the full ~10s refresh
1181
+ // timeout per candidate (observed as a hang in `restoreSessionModel`). The real
1182
+ // key is resolved lazily per request via ModelRegistry.resolver.
1183
+ const hasModelAuth = (candidate: Model): boolean => modelRegistry.hasConfiguredAuth(candidate);
1184
+
1185
+ // Load and create secret obfuscator early so resumed session state and prompt warnings
1186
+ // reflect actual loaded secrets, not just the setting toggle.
1187
+ let obfuscator: SecretObfuscator | undefined;
1188
+ if (settings.get("secrets.enabled")) {
1189
+ const fileEntries = await logger.time("loadSecrets", loadSecrets, cwd, agentDir);
1190
+ const envEntries = collectEnvSecrets();
1191
+ const allEntries = [...envEntries, ...fileEntries];
1192
+ if (allEntries.length > 0) {
1193
+ obfuscator = new SecretObfuscator(allEntries);
1194
+ }
1195
+ }
1196
+ const secretsEnabled = obfuscator?.hasSecrets() === true;
1197
+
1198
+ // Check if session has existing data to restore
1199
+ const existingSession = logger.time("loadSessionContext", () =>
1200
+ deobfuscateSessionContext(sessionManager.buildSessionContext(), obfuscator),
1201
+ );
1202
+ const existingBranch = logger.time("getSessionBranch", () => sessionManager.getBranch());
1203
+ const hasExistingSession = existingBranch.length > 0;
1204
+ const hasThinkingEntry = existingBranch.some(entry => entry.type === "thinking_level_change");
1205
+ const hasServiceTierEntry = existingBranch.some(entry => entry.type === "service_tier_change");
1206
+
1207
+ const hasExplicitModel = options.model !== undefined || options.modelPattern !== undefined;
1208
+ const modelMatchPreferences = getModelMatchPreferences(settings);
1209
+ const allowedModels = await logger.time("resolveAllowedModels", () =>
1210
+ resolveAllowedModels(modelRegistry, settings, modelMatchPreferences),
1211
+ );
1212
+ const defaultRoleSpec = logger.time("resolveDefaultModelRole", () =>
1213
+ resolveModelRoleValue(settings.getModelRole("default"), allowedModels, {
1214
+ settings,
1215
+ matchPreferences: modelMatchPreferences,
1216
+ modelRegistry,
1217
+ }),
1218
+ );
1219
+ let model = options.model;
1220
+ let modelFallbackMessage: string | undefined;
1221
+ // Identify session model strings to restore in fallback order. We do an
1222
+ // initial pass here so model-dependent setup (thinking-level resolution,
1223
+ // host preconnect) can use the restored model; extension-registered
1224
+ // providers aren't visible yet, so we retry the preferred candidates once
1225
+ // extensions register below.
1226
+ const sessionModelStrings =
1227
+ !hasExplicitModel && hasExistingSession
1228
+ ? getRestorableSessionModels(existingSession.models, sessionManager.getLastModelChangeRole())
1229
+ : [];
1230
+ let restoredSessionModelIndex = -1;
1231
+ let restoredSessionThinkingLevel: ThinkingLevel | undefined;
1232
+ if (!hasExplicitModel && !model && sessionModelStrings.length > 0) {
1233
+ logger.time("restoreSessionModel", () => {
1234
+ let failedSessionModel: string | undefined;
1235
+ for (let i = 0; i < sessionModelStrings.length; i++) {
1236
+ const sessionModelStr = sessionModelStrings[i];
1237
+ const parsedModel = parseModelString(sessionModelStr, {
1238
+ allowMaxAlias: true,
1239
+ isLiteralModelId: (provider, id) => modelRegistry.find(provider, id) !== undefined,
1240
+ });
1241
+ if (!parsedModel) {
1242
+ failedSessionModel ??= sessionModelStr;
1243
+ continue;
1244
+ }
1245
+
1246
+ const restoredModel = modelRegistry.find(parsedModel.provider, parsedModel.id);
1247
+ if (restoredModel && hasModelAuth(restoredModel)) {
1248
+ model = restoredModel;
1249
+ restoredSessionModelIndex = i;
1250
+ restoredSessionThinkingLevel = parsedModel.thinkingLevel;
1251
+ break;
1252
+ }
1253
+ failedSessionModel ??= sessionModelStr;
1254
+ }
1255
+ if (failedSessionModel) {
1256
+ modelFallbackMessage = `Could not restore model ${failedSessionModel}`;
1257
+ }
1258
+ });
1259
+ }
1260
+
1261
+ // If still no model, try settings default.
1262
+ // Skip settings fallback when an explicit model was requested.
1263
+ if (!hasExplicitModel && !model && defaultRoleSpec.model) {
1264
+ const settingsDefaultModel = defaultRoleSpec.model;
1265
+ logger.time("resolveSettingsDefaultModel", () => {
1266
+ // defaultRoleSpec.model already comes from modelRegistry.getAvailable(),
1267
+ // so re-validating auth here just repeats the expensive lookup path.
1268
+ model = settingsDefaultModel;
1269
+ });
1270
+ }
1271
+
1272
+ const taskDepth = options.taskDepth ?? 0;
1273
+
1274
+ // Resolves the session/agent thinking level using the same precedence we
1275
+ // apply at startup: explicit option → persisted session entry → restored
1276
+ // model selector suffix → default role's explicit selector → selected
1277
+ // model's defaultLevel → global settings default. Run again after extension
1278
+ // role reclaim so the final model's own defaults aren't masked by an earlier
1279
+ // fallback model's.
1280
+ const pickInitialThinkingLevel = (selectedModel: Model | undefined): ConfiguredThinkingLevel | undefined => {
1281
+ let level = options.thinkingLevel;
1282
+ if (level === undefined && hasExistingSession && hasThinkingEntry) {
1283
+ level = parseThinkingLevel(existingSession.thinkingLevel);
1284
+ }
1285
+ if (level === undefined && !hasThinkingEntry && restoredSessionThinkingLevel !== undefined) {
1286
+ level = restoredSessionThinkingLevel;
1287
+ }
1288
+ if (level === undefined && !hasExplicitModel && !hasThinkingEntry && defaultRoleSpec.explicitThinkingLevel) {
1289
+ level = defaultRoleSpec.thinkingLevel;
1290
+ }
1291
+ if (level === undefined && selectedModel?.thinking?.defaultLevel !== undefined) {
1292
+ level = selectedModel.thinking.defaultLevel;
1293
+ }
1294
+ if (level === undefined) {
1295
+ level = parseConfiguredThinkingLevel(settings.get("defaultThinkingLevel"));
1296
+ }
1297
+ return level;
1298
+ };
1299
+ let thinkingLevel = pickInitialThinkingLevel(model);
1300
+ let autoThinking = thinkingLevel === AUTO_THINKING;
1301
+ // Concrete level the agent/session start with. With `auto` this is the
1302
+ // provisional level shown until the first per-turn classification resolves;
1303
+ // `auto` itself stays a session-only concept handled by AgentSession.
1304
+ let effectiveThinkingLevel: ThinkingLevel | undefined = thinkingLevel === AUTO_THINKING ? undefined : thinkingLevel;
1305
+ if (model) {
1306
+ const resolvedModel = model;
1307
+ effectiveThinkingLevel = logger.time("resolveThinkingLevelForModel", () =>
1308
+ autoThinking
1309
+ ? resolveProvisionalAutoLevel(resolvedModel)
1310
+ : resolveThinkingLevelForModel(resolvedModel, effectiveThinkingLevel),
1311
+ );
1312
+ // Fire-and-forget TLS+H2 handshake to the model's host so it overlaps
1313
+ // with the rest of session setup (extension/skill load, tool registry,
1314
+ // system prompt build). Without this, the first `fetch(...)` pays the
1315
+ // full handshake serially — 100–300 ms transcontinental for
1316
+ // api.anthropic.com from a residential IP. Every mode benefits
1317
+ // (interactive, print, rpc, acp).
1318
+ preconnectModelHost(model.baseUrl);
1319
+ }
1320
+
1321
+ let skills: Skill[];
1322
+ let skillWarnings: SkillWarning[];
1323
+ if (options.skills !== undefined) {
1324
+ skills = options.skills;
1325
+ skillWarnings = [];
1326
+ } else {
1327
+ const discovered = await (discoveredSkillsPromise ?? Promise.resolve({ skills: [], warnings: [] }));
1328
+ skills = discovered.skills;
1329
+ skillWarnings = discovered.warnings;
1330
+ }
1331
+
1332
+ // Discover rules and bucket them in one pass to avoid repeated scans over large rule sets.
1333
+ const { ttsrManager, rulebookRules, alwaysApplyRules, allRules } = await logger.time(
1334
+ "discoverTtsrRules",
1335
+ async () => {
1336
+ const { TtsrManager } = await import("./export/ttsr");
1337
+ const ttsrSettings = settings.getGroup("ttsr");
1338
+ const ttsrManager = new TtsrManager(ttsrSettings);
1339
+ const rulesResult =
1340
+ options.rules !== undefined
1341
+ ? { items: options.rules, warnings: undefined }
1342
+ : await loadCapability<Rule>(ruleCapability.id, { cwd });
1343
+ const { rulebookRules, alwaysApplyRules } = bucketRules(rulesResult.items, ttsrManager, {
1344
+ builtinRules: ttsrSettings.builtinRules,
1345
+ disabledRules: ttsrSettings.disabledRules,
1346
+ });
1347
+ if (existingSession.injectedTtsrRules.length > 0) {
1348
+ ttsrManager.restoreInjected(existingSession.injectedTtsrRules);
1349
+ }
1350
+ return { ttsrManager, rulebookRules, alwaysApplyRules, allRules: rulesResult.items };
1351
+ },
1352
+ );
1353
+
1354
+ // Resolve contextFiles up-front (it's needed before tool creation). The
1355
+ // workspace tree scan is slow on large repos and we MUST NOT block startup on
1356
+ // it. On timeout we forward `undefined` to ToolSession; buildSystemPromptInternal
1357
+ // will re-race the same promise through its own withDeadline path. Background
1358
+ // work continues so caches still warm.
1359
+ const raceWithDeadline = async <T>(name: string, work: Promise<T>): Promise<T | undefined> => {
1360
+ let timedOut = false;
1361
+ const result = await Promise.race([
1362
+ work,
1363
+ Bun.sleep(STARTUP_SCAN_DEADLINE_MS).then(() => {
1364
+ timedOut = true;
1365
+ return undefined;
1366
+ }),
1367
+ ]);
1368
+ if (timedOut) {
1369
+ logger.warn("Startup scan exceeded deadline; deferring to system prompt fallback", {
1370
+ name,
1371
+ timeoutMs: STARTUP_SCAN_DEADLINE_MS,
1372
+ cwd,
1373
+ });
1374
+ }
1375
+ return result;
1376
+ };
1377
+ const [contextFiles, resolvedWorkspaceTree, watchdogFiles] = await Promise.all([
1378
+ contextFilesPromise,
1379
+ raceWithDeadline("buildWorkspaceTree", workspaceTreePromise),
1380
+ watchdogFilesPromise,
1381
+ ]);
1382
+
1383
+ let agent: Agent;
1384
+ let session!: AgentSession;
1385
+ let hasSession = false;
1386
+ let hasRegistered = false;
1387
+ const enableLsp = options.enableLsp ?? true;
1388
+ const asyncMaxJobs = Math.min(100, Math.max(1, settings.get("async.maxJobs") ?? 100));
1389
+ const ASYNC_INLINE_RESULT_MAX_CHARS = 12_000;
1390
+ const ASYNC_PREVIEW_MAX_CHARS = 4_000;
1391
+ const formatAsyncResultForFollowUp = async (result: string): Promise<string> => {
1392
+ if (result.length <= ASYNC_INLINE_RESULT_MAX_CHARS) {
1393
+ return result;
1394
+ }
1395
+
1396
+ const preview = `${result.slice(0, ASYNC_PREVIEW_MAX_CHARS)}\n\n[Output truncated. Showing first ${ASYNC_PREVIEW_MAX_CHARS.toLocaleString()} characters.]`;
1397
+ try {
1398
+ const { path: artifactPath, id: artifactId } = await sessionManager.allocateArtifactPath("async");
1399
+ if (artifactPath && artifactId) {
1400
+ await Bun.write(artifactPath, result);
1401
+ return `${preview}\nFull output: artifact://${artifactId}`;
1402
+ }
1403
+ } catch (error) {
1404
+ logger.warn("Failed to persist async follow-up artifact", {
1405
+ error: error instanceof Error ? error.message : String(error),
1406
+ });
1407
+ }
1408
+
1409
+ return preview;
1410
+ };
1411
+ // Only the first top-level session in a process owns an AsyncJobManager.
1412
+ // Subagents inherit the parent's manager via `AsyncJobManager.instance()`
1413
+ // (set below), and any additional top-level session spun up in-process
1414
+ // (e.g. the agent-creation architect in `agent-dashboard.ts`) must share
1415
+ // the live singleton — otherwise its dispose path would clobber the
1416
+ // owning session's manager and break the `task`/`bash` async paths
1417
+ // (issue #1923). The `instance()` guard means later sessions also skip
1418
+ // constructing an orphaned manager that nothing would ever route to.
1419
+ const asyncJobManager =
1420
+ !options.parentTaskPrefix && !AsyncJobManager.instance()
1421
+ ? new AsyncJobManager({
1422
+ maxRunningJobs: asyncMaxJobs,
1423
+ onJobComplete: async (jobId, result, job) => {
1424
+ if (!session || asyncJobManager!.isDeliverySuppressed(jobId)) return;
1425
+ const formattedResult = await formatAsyncResultForFollowUp(result);
1426
+ if (asyncJobManager!.isDeliverySuppressed(jobId)) return;
1427
+
1428
+ const durationMs = job ? Math.max(0, Date.now() - job.startTime) : undefined;
1429
+ session.yieldQueue.enqueue<AsyncResultEntry>("async-result", {
1430
+ jobId,
1431
+ result: formattedResult,
1432
+ job,
1433
+ durationMs,
1434
+ });
1435
+ },
1436
+ })
1437
+ : undefined;
1438
+
1439
+ const scopedAsyncJobManager = asyncJobManager ?? (options.parentTaskPrefix ? AsyncJobManager.instance() : undefined);
1440
+
1441
+ const agentRegistry = options.agentRegistry ?? AgentRegistry.global();
1442
+ const resolvedAgentId = options.agentId ?? options.parentTaskPrefix ?? MAIN_AGENT_ID;
1443
+ const resolvedAgentDisplayName =
1444
+ options.agentDisplayName ?? ((options.taskDepth ?? 0) > 0 || options.parentTaskPrefix ? "sub" : "main");
1445
+ const agentKind = (options.taskDepth ?? 0) > 0 || options.parentTaskPrefix ? ("sub" as const) : ("main" as const);
1446
+ /**
1447
+ * Forget the agent ref on teardown — unless the agent is being parked (or is
1448
+ * already parked). Parking disposes the session but keeps the ref addressable
1449
+ * (history://, revive); only process teardown / explicit kill unregisters.
1450
+ */
1451
+ const unregisterUnlessParked = (): void => {
1452
+ if (agentRegistry.get(resolvedAgentId)?.status === "parked") return;
1453
+ if (AgentLifecycleManager.global().isParking(resolvedAgentId)) return;
1454
+ agentRegistry.unregister(resolvedAgentId);
1455
+ };
1456
+ const evalKernelOwnerId = `agent-session:${Snowflake.next()}`;
1457
+
1458
+ try {
1459
+ const getActiveModelString = (): string | undefined => {
1460
+ const activeModel = agent?.state.model;
1461
+ if (activeModel) return formatModelString(activeModel);
1462
+ if (model) return formatModelString(model);
1463
+ return undefined;
1464
+ };
1465
+ // Per-path mutation counter shared across edit/write tools. Late-diagnostics
1466
+ // entries capture it at fetch time and are dropped at injection if a newer
1467
+ // mutation (any tool) bumped it in the meantime.
1468
+ const fileMutationVersions = new Map<string, number>();
1469
+ const toolSession: ToolSession = {
1470
+ get cwd() {
1471
+ return sessionManager.getCwd();
1472
+ },
1473
+ hasUI: options.hasUI ?? false,
1474
+ enableLsp,
1475
+ get hasEditTool() {
1476
+ const requestedToolNames = options.toolNames
1477
+ ? [...new Set(options.toolNames.map(name => name.toLowerCase()))]
1478
+ : undefined;
1479
+ return !requestedToolNames || requestedToolNames.includes("edit");
1480
+ },
1481
+ skipPythonPreflight: options.skipPythonPreflight,
1482
+ contextFiles,
1483
+ workspaceTree: resolvedWorkspaceTree,
1484
+ skills,
1485
+ rules: allRules,
1486
+ eventBus,
1487
+ outputSchema: options.outputSchema,
1488
+ requireYieldTool: options.requireYieldTool,
1489
+ taskDepth: options.taskDepth ?? 0,
1490
+ getSessionFile: () => sessionManager.getSessionFile() ?? null,
1491
+ getEvalKernelOwnerId: () => evalKernelOwnerId,
1492
+ getEvalSessionId: () =>
1493
+ session?.getEvalSessionId() ?? options.parentEvalSessionId ?? defaultEvalSessionId(toolSession),
1494
+ assertEvalExecutionAllowed: () => session?.assertEvalExecutionAllowed(),
1495
+ trackEvalExecution: (execution, abortController) =>
1496
+ session ? session.trackEvalExecution(execution, abortController) : execution,
1497
+ getSessionId: () => sessionManager.getSessionId?.() ?? null,
1498
+ getHindsightSessionState: () => session?.getHindsightSessionState(),
1499
+ getMnemopiSessionState: () => session?.getMnemopiSessionState(),
1500
+ getAgentId: () => resolvedAgentId,
1501
+ getToolByName: name => session?.getToolByName(name),
1502
+ agentRegistry,
1503
+ getSessionSpawns: () => options.spawns ?? "*",
1504
+ getModelString: () => (hasExplicitModel && model ? formatModelString(model) : undefined),
1505
+ getActiveModelString,
1506
+ getActiveModel: () => agent?.state.model ?? model,
1507
+ getImageAttachments: () => session?.getImageAttachments() ?? [],
1508
+ getPlanModeState: () => session?.getPlanModeState(),
1509
+ getPlanReferencePath: () => session?.getPlanReferencePath() ?? "local://PLAN.md",
1510
+ getGoalModeState: () => session?.getGoalModeState(),
1511
+ getGoalRuntime: () => session?.goalRuntime,
1512
+ getUsageStatistics: () => sessionManager.getUsageStatistics(),
1513
+ getTurnBudget: () => sessionManager.getTurnBudget(),
1514
+ recordEvalSubagentUsage: output => sessionManager.recordEvalSubagentOutput(output),
1515
+ getClientBridge: () => session?.clientBridge,
1516
+ queueDeferredDiagnostics: entry => session?.yieldQueue.enqueue(LSP_LATE_DIAGNOSTIC_MESSAGE_TYPE, entry),
1517
+ bumpFileMutationVersion: path => {
1518
+ const next = (fileMutationVersions.get(path) ?? 0) + 1;
1519
+ fileMutationVersions.set(path, next);
1520
+ return next;
1521
+ },
1522
+ getFileMutationVersion: path => fileMutationVersions.get(path) ?? 0,
1523
+ getTodoPhases: () => session.getTodoPhases(),
1524
+ setTodoPhases: phases => session.setTodoPhases(phases),
1525
+ isMCPDiscoveryEnabled: () => session.isMCPDiscoveryEnabled(),
1526
+ getSelectedMCPToolNames: () => session.getSelectedMCPToolNames(),
1527
+ activateDiscoveredMCPTools: toolNames => session.activateDiscoveredMCPTools(toolNames),
1528
+ // Generic tool discovery (unified — covers built-in + MCP + extension)
1529
+ isToolDiscoveryEnabled: () => session.isToolDiscoveryEnabled(),
1530
+ getDiscoverableTools: filter => session.getDiscoverableTools(filter),
1531
+ getDiscoverableToolSearchIndex: () => session.getDiscoverableToolSearchIndex(),
1532
+ getSelectedDiscoveredToolNames: () => session.getSelectedDiscoveredToolNames(),
1533
+ activateDiscoveredTools: toolNames => session.activateDiscoveredTools(toolNames),
1534
+ getCheckpointState: () => session.getCheckpointState(),
1535
+ setCheckpointState: state => session.setCheckpointState(state ?? undefined),
1536
+ getToolChoiceQueue: () => session.toolChoiceQueue,
1537
+ buildToolChoice: name => {
1538
+ const m = session.model;
1539
+ return m ? buildNamedToolChoice(name, m) : undefined;
1540
+ },
1541
+ steer: msg =>
1542
+ session.agent.steer({
1543
+ role: "custom",
1544
+ customType: msg.customType,
1545
+ content: msg.content,
1546
+ display: false,
1547
+ details: msg.details,
1548
+ attribution: "agent",
1549
+ timestamp: Date.now(),
1550
+ }),
1551
+ peekQueueInvoker: () => session.peekQueueInvoker(),
1552
+ peekPendingInvoker: () => session.peekPendingInvoker(),
1553
+ clearPendingInvokers: () => session.clearPendingInvokers(),
1554
+ peekStandingResolveHandler: () => session.peekStandingResolveHandler(),
1555
+ setStandingResolveHandler: handler => session.setStandingResolveHandler(handler),
1556
+ allocateOutputArtifact: async toolType => {
1557
+ try {
1558
+ return await sessionManager.allocateArtifactPath(toolType);
1559
+ } catch {
1560
+ return {};
1561
+ }
1562
+ },
1563
+ getArtifactManager: () => sessionManager.getArtifactManager(),
1564
+ settings,
1565
+ authStorage,
1566
+ modelRegistry,
1567
+ getTelemetry: () => agent?.telemetry,
1568
+ // Subagents inherit the singleton (the parent's manager) so their bash/task
1569
+ // completions still flow into the spawning conversation's yieldQueue.
1570
+ // Secondary in-process top-level sessions (no parentTaskPrefix, no
1571
+ // constructed manager because the singleton was already installed) leave
1572
+ // this undefined so tools and session job snapshots refuse async work
1573
+ // instead of silently routing into the owning session (issue #1923).
1574
+ asyncJobManager: scopedAsyncJobManager,
1575
+ };
1576
+
1577
+ // Wire process-wide internal URL singletons owned by their real classes.
1578
+ // Top-level sessions install the active snapshots; subagents inherit them.
1579
+ // Artifact and agent-output URLs resolve via `AgentRegistry.global()` —
1580
+ // the protocol handlers walk each ref's `sessionManager.getArtifactsDir()`,
1581
+ // which collapses to the parent's dir for subagents (they adopt the
1582
+ // parent's ArtifactManager) so one lookup hits everything.
1583
+ const getArtifactsDir = () => sessionManager.getArtifactsDir();
1584
+ if (!options.parentTaskPrefix) {
1585
+ setActiveSkills(skills);
1586
+ // Include TTSR rules so `rule://<name>` can resolve them too. They are
1587
+ // registered with the manager and bucketed out before rulebook/always,
1588
+ // so without this a TTSR-only rule (e.g. a triggered builtin) is not
1589
+ // addressable and `rule://` reports "Available: none".
1590
+ setActiveRules([...rulebookRules, ...alwaysApplyRules, ...ttsrManager.getRules()]);
1591
+ if (asyncJobManager) AsyncJobManager.setInstance(asyncJobManager);
1592
+ }
1593
+ const localProtocolOptions = options.localProtocolOptions ?? {
1594
+ getArtifactsDir,
1595
+ getSessionId: () => sessionManager.getSessionId?.() ?? null,
1596
+ };
1597
+ if (options.localProtocolOptions) {
1598
+ LocalProtocolHandler.setOverride(options.localProtocolOptions);
1599
+ }
1600
+ toolSession.getArtifactsDir = getArtifactsDir;
1601
+ toolSession.localProtocolOptions = localProtocolOptions;
1602
+ toolSession.agentOutputManager = new AgentOutputManager(
1603
+ getArtifactsDir,
1604
+ options.parentTaskPrefix ? { parentPrefix: options.parentTaskPrefix } : undefined,
1605
+ );
1606
+
1607
+ // Create built-in tools (already wrapped with meta notice formatting)
1608
+ const builtinTools = await logger.time("createAllTools", createTools, toolSession, options.toolNames);
1609
+
1610
+ // Discover MCP tools from .mcp.json files
1611
+ let mcpManager: MCPManager | undefined = options.mcpManager;
1612
+ toolSession.mcpManager = mcpManager;
1613
+ const enableMCP = options.enableMCP ?? true;
1614
+ const deferMCPDiscoveryForUI = enableMCP && !mcpManager && options.hasUI === true;
1615
+ const customTools: CustomTool[] = [];
1616
+ let startDeferredMCPDiscovery:
1617
+ | ((liveSession: AgentSession, activation: DeferredMCPActivation) => void)
1618
+ | undefined;
1619
+ const startupQuiet = settings.get("startup.quiet");
1620
+ const onMCPConnecting = (serverNames: string[]) => {
1621
+ if (!options.hasUI || startupQuiet || serverNames.length === 0) return;
1622
+ eventBus.emit(MCP_CONNECTING_EVENT_CHANNEL, { serverNames } satisfies McpConnectingEvent);
1623
+ };
1624
+ const mcpDiscoverOptions = {
1625
+ onConnecting: onMCPConnecting,
1626
+ enableProjectConfig: settings.get("mcp.enableProjectConfig") ?? true,
1627
+ // Always filter Exa - we have native integration
1628
+ filterExa: true,
1629
+ // Filter browser MCP servers when builtin browser tool is active
1630
+ filterBrowser: settings.get("browser.enabled") ?? false,
1631
+ };
1632
+ if (enableMCP && !mcpManager) {
1633
+ if (deferMCPDiscoveryForUI) {
1634
+ const cacheStorage = settings.getStorage();
1635
+ mcpManager = new MCPManager(cwd, cacheStorage ? new MCPToolCache(cacheStorage) : null);
1636
+ mcpManager.setAuthStorage(authStorage);
1637
+ toolSession.mcpManager = mcpManager;
1638
+
1639
+ if (settings.get("mcp.notifications")) {
1640
+ mcpManager.setNotificationsEnabled(true);
1641
+ }
1642
+
1643
+ const deferredMCPManager = mcpManager;
1644
+ startDeferredMCPDiscovery = (liveSession, activation) => {
1645
+ void (async () => {
1646
+ try {
1647
+ const mcpResult = await logger.time("discoverAndLoadMCPTools", () =>
1648
+ deferredMCPManager.discoverAndConnect(mcpDiscoverOptions),
1649
+ );
1650
+ // The session can be torn down while servers are still connecting.
1651
+ // Don't resurrect tools on a disposed session, and don't leak the
1652
+ // transports/subprocesses the connect just spawned.
1653
+ if (liveSession.isDisposed) {
1654
+ await deferredMCPManager.disconnectAll();
1655
+ return;
1656
+ }
1657
+ applyMCPEnvironment(mcpResult);
1658
+ logMCPLoadErrors(mcpResult.errors);
1659
+ // `tools.discoveryMode: "auto"` was resolved against a registry that
1660
+ // held only built-ins plus persisted placeholder names. Recompute with
1661
+ // the real MCP tool count: a large toolset must flip discovery on
1662
+ // BEFORE the refresh, or activateAll would dump every MCP tool into
1663
+ // the active set with no search_tool_bm25 registered.
1664
+ let discoveryEnabled = activation.mcpDiscoveryEnabled;
1665
+ let activateAll = activation.activateAllMCPTools;
1666
+ if (!discoveryEnabled) {
1667
+ const nonMCPToolNames = [...toolRegistry.keys()].filter(name => !isMCPToolName(name));
1668
+ const projectedMode = resolveEffectiveToolDiscoveryMode(
1669
+ settings,
1670
+ countToolsForAutoDiscovery([...nonMCPToolNames, ...mcpResult.tools.map(tool => tool.name)]),
1671
+ );
1672
+ if (projectedMode !== "off") {
1673
+ effectiveDiscoveryMode = projectedMode;
1674
+ mcpDiscoveryEnabled = true;
1675
+ discoveryEnabled = true;
1676
+ activateAll = false;
1677
+ liveSession.enableMCPDiscovery();
1678
+ if (!toolRegistry.has("search_tool_bm25")) {
1679
+ const searchTool: Tool = new SearchToolBm25Tool(toolSession);
1680
+ toolRegistry.set(
1681
+ searchTool.name,
1682
+ new ExtensionToolWrapper(wrapToolWithMetaNotice(searchTool), extensionRunner) as Tool,
1683
+ );
1684
+ }
1685
+ await liveSession.setActiveToolsByName([
1686
+ ...liveSession.getActiveToolNames(),
1687
+ "search_tool_bm25",
1688
+ ]);
1689
+ }
1690
+ }
1691
+ await liveSession.refreshMCPTools(mcpResult.tools, { activateAll });
1692
+ if (activation.explicitlyRequestedMCPToolNames.length > 0) {
1693
+ if (discoveryEnabled && !activation.mcpDiscoveryEnabled) {
1694
+ // Discovery flipped on mid-flight: route the explicit request
1695
+ // through discovery-aware activation so selection persists.
1696
+ await liveSession.activateDiscoveredMCPTools(activation.explicitlyRequestedMCPToolNames);
1697
+ } else if (!discoveryEnabled) {
1698
+ await liveSession.setActiveToolsByName([
1699
+ ...liveSession.getActiveToolNames(),
1700
+ ...activation.explicitlyRequestedMCPToolNames,
1701
+ ]);
1702
+ }
1703
+ }
1704
+ } catch (error) {
1705
+ logger.error("MCP tool load failed", {
1706
+ path: ".mcp.json",
1707
+ error: error instanceof Error ? error.message : String(error),
1708
+ });
1709
+ }
1710
+ })();
1711
+ };
1712
+ } else {
1713
+ const mcpResult = await logger.time("discoverAndLoadMCPTools", discoverAndLoadMCPTools, cwd, {
1714
+ ...mcpDiscoverOptions,
1715
+ cacheStorage: settings.getStorage(),
1716
+ authStorage,
1717
+ });
1718
+ mcpManager = mcpResult.manager;
1719
+ toolSession.mcpManager = mcpManager;
1720
+
1721
+ if (settings.get("mcp.notifications")) {
1722
+ mcpManager.setNotificationsEnabled(true);
1723
+ }
1724
+ applyMCPEnvironment(mcpResult);
1725
+
1726
+ // Log MCP errors
1727
+ for (const { path, error } of mcpResult.errors) {
1728
+ logger.error("MCP tool load failed", { path, error });
1729
+ }
1730
+
1731
+ if (mcpResult.tools.length > 0) {
1732
+ // MCP tools are LoadedCustomTool, extract the tool property
1733
+ customTools.push(...mcpResult.tools.map(loaded => loaded.tool));
1734
+ }
1735
+ }
1736
+ }
1737
+ // Only top-level sessions own the global MCPManager. Subagents already
1738
+ // receive the parent's manager via `options.mcpManager`, and reassigning
1739
+ // the singleton to the same value is a no-op — keep the gate explicit
1740
+ // to mirror the AsyncJobManager ownership rule.
1741
+ if (mcpManager && !options.parentTaskPrefix) MCPManager.setInstance(mcpManager);
1742
+
1743
+ // Add image tools when the active model or configured image providers can generate images.
1744
+ const imageGenTools = await logger.time("getImageGenTools", () => getImageGenTools(modelRegistry, model));
1745
+ if (imageGenTools.length > 0) {
1746
+ customTools.push(...(imageGenTools as unknown as CustomTool[]));
1747
+ }
1748
+
1749
+ if (settings.get("speechgen.enabled")) {
1750
+ customTools.push(ttsTool as unknown as CustomTool);
1751
+ }
1752
+
1753
+ // Add web search tools
1754
+ if (options.toolNames?.includes("web_search")) {
1755
+ customTools.push(...getSearchTools());
1756
+ }
1757
+
1758
+ // Discover custom tools from `.omp/tools/`, `.claude/tools/`, plugins, etc.
1759
+ // Subagents reuse the parent's scan via `preloadedCustomToolPaths` to skip
1760
+ // the FS walk, but ALWAYS re-call `loadCustomTools` here so factories bind
1761
+ // to THIS session's `CustomToolAPI` (cwd, exec, pushPendingAction, UI).
1762
+ // Forwarding the parent's `LoadedCustomTool[]` directly would route tool
1763
+ // execution back through the parent — wrong for isolated tasks and for
1764
+ // pending-action queueing.
1765
+ const builtInToolNames = builtinTools.map(t => t.name);
1766
+ const customToolPaths: ToolPathWithSource[] =
1767
+ options.preloadedCustomToolPaths ??
1768
+ (await logger.time("discoverCustomToolPaths", () => discoverCustomToolPaths([], cwd)));
1769
+ const customToolsLoadResult = await logger.time("loadCustomTools", () =>
1770
+ loadCustomTools(customToolPaths, cwd, builtInToolNames, action => queueResolveHandler(toolSession, action)),
1771
+ );
1772
+ for (const { path, error } of customToolsLoadResult.errors) {
1773
+ logger.error("Custom tool load failed", { path, error });
1774
+ }
1775
+ if (customToolsLoadResult.tools.length > 0) {
1776
+ customTools.push(...customToolsLoadResult.tools.map(loaded => loaded.tool));
1777
+ }
1778
+ // Forward the path list (NOT the loaded tools) to subagents so they
1779
+ // re-bind under their own `CustomToolAPI` while skipping the FS scan.
1780
+ toolSession.customToolPaths = customToolPaths;
1781
+
1782
+ const inlineExtensions: ExtensionFactory[] = options.extensions ? [...options.extensions] : [];
1783
+ inlineExtensions.push((await import("./autoresearch")).createAutoresearchExtension);
1784
+ if (customTools.length > 0) {
1785
+ inlineExtensions.push(createCustomToolsExtension(customTools));
1786
+ }
1787
+
1788
+ // Load extensions. Three paths:
1789
+ // 1. `preloadedExtensions` (CLI): caller already loaded — reuse the
1790
+ // Extension instances. Shallow-clone `extensions` so the inline
1791
+ // push below cannot mutate the caller's array. `runtime` is shared
1792
+ // so flag values set pre-creation flow into the live session.
1793
+ // 2. `preloadedExtensionPaths` (subagent): caller resolved paths;
1794
+ // skip the FS scan but always re-call `loadExtensions` here so
1795
+ // each `Extension` binds to THIS session's `ExtensionAPI`
1796
+ // (cwd, eventBus, runtime).
1797
+ // 3. No preload: run the full session discovery.
1798
+ // `disableExtensionDiscovery` is honored implicitly: a caller that set
1799
+ // the flag and pre-resolved the result already reflects that choice.
1800
+ let extensionPaths: string[];
1801
+ let extensionsResult: LoadExtensionsResult;
1802
+ if (options.preloadedExtensions) {
1803
+ extensionsResult = {
1804
+ ...options.preloadedExtensions,
1805
+ extensions: [...options.preloadedExtensions.extensions],
1806
+ };
1807
+ // Capture paths for downstream forwarding; filter inline-factory
1808
+ // entries (`<inline-N>`) — those are per-session, not source paths.
1809
+ extensionPaths = extensionsResult.extensions
1810
+ .map(ext => ext.resolvedPath)
1811
+ .filter(p => !p.startsWith("<inline"));
1812
+ } else if (options.preloadedExtensionPaths) {
1813
+ extensionPaths = options.preloadedExtensionPaths;
1814
+ extensionsResult = await logger.time("loadExtensions", loadExtensions, extensionPaths, cwd, eventBus);
1815
+ for (const { path, error } of extensionsResult.errors) {
1816
+ logger.error("Failed to load extension", { path, error });
1817
+ }
1818
+ } else {
1819
+ extensionPaths = await logger.time("discoverSessionExtensionPaths", () =>
1820
+ discoverSessionExtensionPaths(options, cwd, settings),
1821
+ );
1822
+ extensionsResult = await logger.time("loadExtensions", loadExtensions, extensionPaths, cwd, eventBus);
1823
+ for (const { path, error } of extensionsResult.errors) {
1824
+ logger.error("Failed to load extension", { path, error });
1825
+ }
1826
+ }
1827
+ // Forward the source-path list (NOT the loaded instances) so subagents
1828
+ // rebuild their own session-scoped extensions.
1829
+ toolSession.extensionPaths = extensionPaths;
1830
+
1831
+ // Load inline extensions from factories
1832
+ if (inlineExtensions.length > 0) {
1833
+ for (let i = 0; i < inlineExtensions.length; i++) {
1834
+ const factory = inlineExtensions[i];
1835
+ const loaded = await loadExtensionFromFactory(
1836
+ factory,
1837
+ cwd,
1838
+ eventBus,
1839
+ extensionsResult.runtime,
1840
+ `<inline-${i}>`,
1841
+ );
1842
+ extensionsResult.extensions.push(loaded);
1843
+ }
1844
+ }
1845
+
1846
+ // Process provider registrations queued during extension loading.
1847
+ // This must happen before the runner is created so that models registered by
1848
+ // extensions are available for model selection on session resume / fallback.
1849
+ const activeExtensionSources = extensionsResult.extensions.map(extension => extension.path);
1850
+ modelRegistry.syncExtensionSources(activeExtensionSources);
1851
+ for (const sourceId of new Set(activeExtensionSources)) {
1852
+ modelRegistry.clearSourceRegistrations(sourceId);
1853
+ }
1854
+ if (extensionsResult.runtime.pendingProviderRegistrations.length > 0) {
1855
+ for (const { name, config, sourceId } of extensionsResult.runtime.pendingProviderRegistrations) {
1856
+ modelRegistry.registerProvider(name, config, sourceId);
1857
+ }
1858
+ extensionsResult.runtime.pendingProviderRegistrations = [];
1859
+ }
1860
+ // Discover runtime (extension) provider catalogs now that they are
1861
+ // registered. The startup refreshInBackground() ran before extensions
1862
+ // loaded, so dynamic extension providers are only discovered here. Runs in
1863
+ // the background (cache-aware) so startup is never blocked on the fetch; the
1864
+ // model list re-renders when the catalog arrives, like other dynamic providers.
1865
+ void modelRegistry.refreshRuntimeProviders().catch(error => {
1866
+ logger.warn("runtime provider discovery failed", {
1867
+ error: error instanceof Error ? error.message : String(error),
1868
+ });
1869
+ });
1870
+
1871
+ // Retry session-model candidates now that extension providers are
1872
+ // registered. The initial restore runs before extensions load, so a role
1873
+ // model supplied by an extension would have either fallen back to the
1874
+ // saved default (`restoredSessionModelIndex > 0`) or failed entirely
1875
+ // (`restoredSessionModelIndex === -1`, with the settings default or
1876
+ // downstream fallback filling `model`). Reclaim it here so resume
1877
+ // honors the last active role in either case.
1878
+ const sessionRetryLimit = restoredSessionModelIndex >= 0 ? restoredSessionModelIndex : sessionModelStrings.length;
1879
+ if (!hasExplicitModel && sessionRetryLimit > 0) {
1880
+ for (let i = 0; i < sessionRetryLimit; i++) {
1881
+ const sessionModelStr = sessionModelStrings[i];
1882
+ const parsedModel = parseModelString(sessionModelStr, {
1883
+ allowMaxAlias: true,
1884
+ isLiteralModelId: (provider, id) => modelRegistry.find(provider, id) !== undefined,
1885
+ });
1886
+ if (!parsedModel) continue;
1887
+ const restoredModel = modelRegistry.find(parsedModel.provider, parsedModel.id);
1888
+ if (restoredModel && hasModelAuth(restoredModel)) {
1889
+ model = restoredModel;
1890
+ modelFallbackMessage = undefined;
1891
+ restoredSessionModelIndex = i;
1892
+ restoredSessionThinkingLevel = parsedModel.thinkingLevel;
1893
+ // Recompute thinking-level from scratch against the reclaimed
1894
+ // model: any value derived from the earlier fallback model's
1895
+ // `thinking.defaultLevel` must not become sticky.
1896
+ thinkingLevel = pickInitialThinkingLevel(restoredModel);
1897
+ autoThinking = thinkingLevel === AUTO_THINKING;
1898
+ effectiveThinkingLevel = thinkingLevel === AUTO_THINKING ? undefined : thinkingLevel;
1899
+ effectiveThinkingLevel = logger.time("resolveThinkingLevelForModel", () =>
1900
+ autoThinking
1901
+ ? resolveProvisionalAutoLevel(restoredModel)
1902
+ : resolveThinkingLevelForModel(restoredModel, effectiveThinkingLevel),
1903
+ );
1904
+ preconnectModelHost(restoredModel.baseUrl);
1905
+ break;
1906
+ }
1907
+ }
1908
+ }
1909
+ // Resolve deferred --model pattern now that extension models are registered.
1910
+ if (!model && options.modelPattern) {
1911
+ const availableModels = modelRegistry.getAll();
1912
+ const matchPreferences = getModelMatchPreferences(settings);
1913
+ const { model: resolved } = parseModelPattern(options.modelPattern, availableModels, matchPreferences, {
1914
+ modelRegistry,
1915
+ });
1916
+ if (resolved) {
1917
+ model = resolved;
1918
+ modelFallbackMessage = undefined;
1919
+ } else {
1920
+ modelFallbackMessage = `Model "${options.modelPattern}" not found`;
1921
+ }
1922
+ }
1923
+
1924
+ // Fall back to first available model with a valid API key, honoring the
1925
+ // path-scoped `enabledModels` allow-list when configured. Skip when the
1926
+ // user explicitly requested a model via --model that wasn't found.
1927
+ if (!model && !options.modelPattern) {
1928
+ // Re-resolve the allowed set: extension factories above may have
1929
+ // registered providers/models that weren't visible at startup.
1930
+ const fallbackCandidates = await resolveAllowedModels(modelRegistry, settings, modelMatchPreferences);
1931
+ const defaultModel = pickDefaultAvailableModel(fallbackCandidates.filter(hasModelAuth));
1932
+ if (defaultModel) {
1933
+ model = defaultModel;
1934
+ }
1935
+ if (model) {
1936
+ if (modelFallbackMessage) {
1937
+ modelFallbackMessage += `. Using ${model.provider}/${model.id}`;
1938
+ }
1939
+ } else {
1940
+ const patterns = settings.get("enabledModels");
1941
+ modelFallbackMessage =
1942
+ patterns && patterns.length > 0
1943
+ ? `No model available matching enabledModels (${patterns.join(", ")}) with usable credentials. Configure auth for an allowed provider or adjust enabledModels.`
1944
+ : "No models available. Use /login or set an API key environment variable. Then use /model to select a model.";
1945
+ }
1946
+ }
1947
+
1948
+ // Discover custom commands (TypeScript slash commands)
1949
+ const customCommandsResult: CustomCommandsLoadResult = options.disableExtensionDiscovery
1950
+ ? { commands: [], errors: [] }
1951
+ : await logger.time("discoverCustomCommands", loadCustomCommandsInternal, { cwd, agentDir });
1952
+ if (!options.disableExtensionDiscovery) {
1953
+ for (const { path, error } of customCommandsResult.errors) {
1954
+ logger.error("Failed to load custom command", { path, error });
1955
+ }
1956
+ }
1957
+
1958
+ // The runner is created unconditionally — even with zero extensions loaded — because the
1959
+ // `ExtensionToolWrapper` installed below is the only place the per-tool approval gate runs.
1960
+ // A conditional runner means the approval system silently disappears for users with no
1961
+ // extensions, contradicting non-yolo `tools.approvalMode` settings without feedback.
1962
+ // (The builtin autoresearch extension is unconditionally loaded above, so this scenario
1963
+ // is unreachable; unconditional runner construction keeps that invariant explicit and
1964
+ // prevents future optional extensions from silently re-opening the hole.)
1965
+ const extensionRunner: ExtensionRunner = new ExtensionRunner(
1966
+ extensionsResult.extensions,
1967
+ extensionsResult.runtime,
1968
+ cwd,
1969
+ sessionManager,
1970
+ modelRegistry,
1971
+ () => (hasSession ? createSessionMemoryRuntimeContext(session, agentDir, cwd) : undefined),
1972
+ settings,
1973
+ );
1974
+
1975
+ credentialDisabledTarget = extensionRunner;
1976
+ for (const event of startupCredentialDisabledEvents.splice(0)) {
1977
+ // Discard return: any handler error is routed through runner.onError listeners.
1978
+ void extensionRunner.emitCredentialDisabled(event);
1979
+ }
1980
+
1981
+ const getSessionContext = () => ({
1982
+ sessionManager,
1983
+ modelRegistry,
1984
+ model: agent.state.model,
1985
+ isIdle: () => !session.isStreaming,
1986
+ hasQueuedMessages: () => session.queuedMessageCount > 0,
1987
+ abort: () => {
1988
+ session.abort({ reason: USER_INTERRUPT_LABEL });
1989
+ },
1990
+ settings,
1991
+ autoApprove: options.autoApprove ?? false,
1992
+ });
1993
+ const toolContextStore = new ToolContextStore(getSessionContext);
1994
+
1995
+ const registeredTools = extensionRunner.getAllRegisteredTools();
1996
+ const sdkCustomTools = options.customTools?.filter(tool => !isLegacyBuiltinToolDefinition(tool)) ?? [];
1997
+ const allCustomTools = [
1998
+ ...registeredTools,
1999
+ ...sdkCustomTools.map(tool => {
2000
+ const definition = isCustomTool(tool) ? customToolToDefinition(tool) : tool;
2001
+ return { definition, extensionPath: "<sdk>" };
2002
+ }),
2003
+ ];
2004
+ // `wrapToolWithMetaNotice` runs the centralized large-output → artifact spill.
2005
+ // Built-in tools get it in `createTools`; extension, SDK-custom, image-gen,
2006
+ // TTS, and startup (non-deferred) MCP tools all funnel through here, so apply
2007
+ // it once at this adapter boundary (idempotent — a no-op if already wrapped).
2008
+ const wrappedExtensionTools: Tool[] = wrapRegisteredTools(allCustomTools, extensionRunner).map(
2009
+ wrapToolWithMetaNotice,
2010
+ );
2011
+
2012
+ // All built-in tools are active (conditional tools like git/ask return null from factory if disabled)
2013
+ const toolRegistry = new Map<string, Tool>();
2014
+ for (const tool of builtinTools) {
2015
+ toolRegistry.set(tool.name, tool);
2016
+ }
2017
+ if (!toolRegistry.has("goal") && settings.get("goal.enabled")) {
2018
+ const goalTool = await logger.time("createTools:goal:session", HIDDEN_TOOLS.goal, toolSession);
2019
+ if (goalTool) {
2020
+ toolRegistry.set(goalTool.name, wrapToolWithMetaNotice(goalTool));
2021
+ }
2022
+ }
2023
+ for (const tool of wrappedExtensionTools) {
2024
+ toolRegistry.set(tool.name, tool);
2025
+ }
2026
+ if (deferMCPDiscoveryForUI && mcpManager) {
2027
+ for (const name of collectPendingMCPToolNames(options.toolNames, existingSession.selectedMCPToolNames)) {
2028
+ if (!toolRegistry.has(name)) {
2029
+ toolRegistry.set(name, createPendingMCPTool(name));
2030
+ }
2031
+ }
2032
+ }
2033
+
2034
+ // Wrap every tool with `ExtensionToolWrapper` so the per-tool approval gate runs on every
2035
+ // call site, regardless of whether any user extensions are loaded. See the runner-construction
2036
+ // comment above for the safety invariant this enforces.
2037
+ for (const tool of toolRegistry.values()) {
2038
+ toolRegistry.set(tool.name, new ExtensionToolWrapper(tool, extensionRunner));
2039
+ }
2040
+ if (model?.provider === "cursor") {
2041
+ toolRegistry.delete("edit");
2042
+ }
2043
+
2044
+ // `resolve` is hidden but must stay in the registry whenever any code path can invoke it:
2045
+ // either a deferrable tool stages a preview action, or plan mode installs a standing handler
2046
+ // that consumes `resolve { action: "apply" }` to submit the plan for approval (issue #1428).
2047
+ // Dropping it on read-only sessions (e.g. plan-mode toolset `read`, `search`, `find`,
2048
+ // `web_search`) leaves plan mode unable to exit through the intended path.
2049
+ const hasDeferrableTools = Array.from(toolRegistry.values()).some(tool => tool.deferrable === true);
2050
+ const planModeAvailable = settings.get("plan.enabled");
2051
+ const needsResolveTool = hasDeferrableTools || planModeAvailable;
2052
+ if (!needsResolveTool) {
2053
+ toolRegistry.delete("resolve");
2054
+ } else if (!toolRegistry.has("resolve")) {
2055
+ const resolveTool = await logger.time("createTools:resolve:session", HIDDEN_TOOLS.resolve, toolSession);
2056
+ if (resolveTool) {
2057
+ toolRegistry.set(resolveTool.name, wrapToolWithMetaNotice(resolveTool));
2058
+ }
2059
+ }
2060
+
2061
+ // `let`: the deferred MCP discovery closure upgrades these when the real
2062
+ // MCP tool count pushes `auto` past its threshold; `rebuildSystemPrompt`
2063
+ // below reads the live bindings.
2064
+ let effectiveDiscoveryMode = resolveEffectiveToolDiscoveryMode(
2065
+ settings,
2066
+ countToolsForAutoDiscovery(toolRegistry.keys()),
2067
+ );
2068
+ if (effectiveDiscoveryMode !== "off" && !toolRegistry.has("search_tool_bm25")) {
2069
+ const searchTool: Tool = new SearchToolBm25Tool(toolSession);
2070
+ toolRegistry.set(
2071
+ searchTool.name,
2072
+ new ExtensionToolWrapper(wrapToolWithMetaNotice(searchTool), extensionRunner) as Tool,
2073
+ );
2074
+ }
2075
+ let mcpDiscoveryEnabled = effectiveDiscoveryMode !== "off"; // back-compat: true when any discovery active
2076
+
2077
+ const reloadSshTool = async (): Promise<AgentTool | null> => {
2078
+ if (!requestedToolNameSet.has("ssh")) return null;
2079
+ const sshTool = (await loadSshTool({
2080
+ ...toolSession,
2081
+ cwd: sessionManager.getCwd(),
2082
+ })) as unknown as AgentTool | null;
2083
+ if (!sshTool) return null;
2084
+ const wrapped = wrapToolWithMetaNotice(sshTool);
2085
+ return new ExtensionToolWrapper(wrapped, extensionRunner) as AgentTool;
2086
+ };
2087
+
2088
+ let cursorEventEmitter: ((event: AgentEvent) => void) | undefined;
2089
+ const cursorExecHandlers = new CursorExecHandlers({
2090
+ cwd,
2091
+ tools: toolRegistry,
2092
+ getToolContext: () => toolContextStore.getContext(),
2093
+ emitEvent: event => cursorEventEmitter?.(event),
2094
+ });
2095
+
2096
+ const inlineToolDescriptors = settings.get("inlineToolDescriptors");
2097
+ const eagerTasks = settings.get("task.eager") !== "default";
2098
+ const eagerTasksAlways = settings.get("task.eager") === "always";
2099
+ const intentField = $flag("PI_INTENT_TRACING", settings.get("tools.intentTracing")) ? INTENT_FIELD : undefined;
2100
+ const rebuildSystemPrompt = async (
2101
+ toolNames: string[],
2102
+ tools: Map<string, AgentTool>,
2103
+ ): Promise<BuildSystemPromptResult> => {
2104
+ toolContextStore.setToolNames(toolNames);
2105
+ const discoverableMCPTools: DiscoverableTool[] = mcpDiscoveryEnabled
2106
+ ? filterBySource(collectDiscoverableTools(tools.values()), "mcp")
2107
+ : [];
2108
+ const activeToolNames = new Set(toolNames);
2109
+ const discoverableBuiltinTools: DiscoverableTool[] =
2110
+ effectiveDiscoveryMode === "all"
2111
+ ? collectDiscoverableTools(
2112
+ Array.from(tools.values()).filter(
2113
+ tool => tool.loadMode === "discoverable" && !activeToolNames.has(tool.name),
2114
+ ),
2115
+ { source: "builtin" },
2116
+ )
2117
+ : [];
2118
+ const discoverableToolsForDesc: DiscoverableTool[] = [...discoverableBuiltinTools, ...discoverableMCPTools];
2119
+ const discoverableToolSummary = summarizeDiscoverableTools(discoverableToolsForDesc);
2120
+ const hasDiscoverableTools =
2121
+ mcpDiscoveryEnabled && toolNames.includes("search_tool_bm25") && discoverableToolsForDesc.length > 0;
2122
+ const promptTools = buildSystemPromptToolMetadata(tools, {
2123
+ search_tool_bm25: { description: renderSearchToolBm25Description(discoverableToolsForDesc) },
2124
+ });
2125
+ const memoryBackend = await resolveMemoryBackend(settings);
2126
+ const memoryInstructions = await memoryBackend.buildDeveloperInstructions(agentDir, settings, session);
2127
+
2128
+ // Build combined append prompt: memory instructions + auto-learn guidance
2129
+ // + MCP server instructions. For UI sessions MCP discovery is deferred, so
2130
+ // `getServerInstructions()` is empty until the background connect completes;
2131
+ // the rebuild that `refreshMCPTools` triggers post-discovery then picks up
2132
+ // the now-connected servers' instructions, so they join the prompt for the
2133
+ // rest of the session.
2134
+ const serverInstructions = mcpManager?.getServerInstructions();
2135
+ // Drive guidance off the auto-learn BUILTINS that createTools actually built
2136
+ // (provenance, not just an active name): `builtInToolNames` excludes a
2137
+ // custom/extension tool that merely shares the name, and reflects the
2138
+ // session-start build — so a subagent that filtered them out, a mid-session
2139
+ // enable that never built them, or a same-named custom tool while auto-learn
2140
+ // is off all get no guidance.
2141
+ const autoLearnInstructions = buildAutoLearnInstructions({
2142
+ manageSkill: builtInToolNames.includes("manage_skill"),
2143
+ learn: builtInToolNames.includes("learn"),
2144
+ });
2145
+ const appendParts: string[] = [];
2146
+ if (memoryInstructions) appendParts.push(memoryInstructions);
2147
+ if (autoLearnInstructions) appendParts.push(autoLearnInstructions);
2148
+ let appendPrompt: string | undefined = appendParts.length > 0 ? appendParts.join("\n\n") : undefined;
2149
+ if (serverInstructions && serverInstructions.size > 0) {
2150
+ const parts: string[] = [];
2151
+ if (appendPrompt) parts.push(appendPrompt);
2152
+ parts.push(
2153
+ "## MCP Server Instructions\n\nThe following instructions are provided by connected MCP servers. They are server-controlled and may not be verified.",
2154
+ );
2155
+ for (const [srvName, srvInstructions] of serverInstructions) {
2156
+ const truncated =
2157
+ srvInstructions.length > MAX_MCP_INSTRUCTIONS_LENGTH
2158
+ ? `${srvInstructions.slice(0, MAX_MCP_INSTRUCTIONS_LENGTH)}\n[truncated]`
2159
+ : srvInstructions;
2160
+ parts.push(`### ${srvName}\n${truncated}`);
2161
+ }
2162
+ appendPrompt = parts.join("\n\n");
2163
+ }
2164
+ // Owned/in-band tool dialects (non-native) require the catalog as `# Tool:`
2165
+ // sections; native tool calling lets the compact name list suffice.
2166
+ const nativeTools = resolveDialect(settings.get("tools.format"), agent?.state.model ?? model) === undefined;
2167
+ if (options.appendSystemPrompt) {
2168
+ appendPrompt = appendPrompt
2169
+ ? `${appendPrompt}\n\n${options.appendSystemPrompt}`
2170
+ : options.appendSystemPrompt;
2171
+ }
2172
+ const defaultPrompt = await buildSystemPromptInternal({
2173
+ cwd,
2174
+ resolvedCustomPrompt: options.customSystemPrompt,
2175
+ skills,
2176
+ contextFiles,
2177
+ tools: promptTools,
2178
+ toolNames,
2179
+ rules: rulebookRules,
2180
+ alwaysApplyRules,
2181
+ resolvedAppendSystemPrompt: appendPrompt,
2182
+ skillsSettings: settings.getGroup("skills"),
2183
+ inlineToolDescriptors,
2184
+ nativeTools,
2185
+ intentField,
2186
+ mcpDiscoveryMode: hasDiscoverableTools,
2187
+ mcpDiscoveryServerSummaries: discoverableToolSummary.servers.map(formatDiscoverableToolServerSummary),
2188
+ eagerTasks,
2189
+ eagerTasksAlways,
2190
+ taskBatch: settings.get("task.batch"),
2191
+ secretsEnabled,
2192
+ workspaceTree: workspaceTreePromise,
2193
+ memoryRootEnabled: memoryBackend.id === "local",
2194
+ model: settings.get("includeModelInPrompt") ? getActiveModelString() : undefined,
2195
+ personality: agentKind === "sub" ? "none" : settings.get("personality"),
2196
+ });
2197
+
2198
+ if (options.systemPrompt === undefined) {
2199
+ return defaultPrompt;
2200
+ }
2201
+ const customPrompt =
2202
+ typeof options.systemPrompt === "function"
2203
+ ? options.systemPrompt(defaultPrompt.systemPrompt)
2204
+ : options.systemPrompt;
2205
+ return {
2206
+ systemPrompt: typeof customPrompt === "string" ? [customPrompt] : customPrompt,
2207
+ };
2208
+ };
2209
+
2210
+ const toolNamesFromRegistry = Array.from(toolRegistry.keys());
2211
+ const explicitlyRequestedToolNames = options.toolNames
2212
+ ? [...new Set(options.toolNames.map(name => name.toLowerCase()))]
2213
+ : undefined;
2214
+ // When `requireYieldTool` is set, the subagent's prompts and idle-reminders demand a
2215
+ // `yield` call to terminate. The tool registry already includes `yield` (see
2216
+ // `createTools`), but an explicit `toolNames` list would otherwise drop it from the
2217
+ // active set — leaving the model unable to satisfy the contract. Mirror the same
2218
+ // invariant `parseAgentFields` enforces on frontmatter `tools`.
2219
+ if (
2220
+ options.requireYieldTool === true &&
2221
+ explicitlyRequestedToolNames &&
2222
+ !explicitlyRequestedToolNames.includes("yield")
2223
+ ) {
2224
+ explicitlyRequestedToolNames.push("yield");
2225
+ }
2226
+ // Auto-learn builtins are force-included into the registry by `createTools`
2227
+ // for enabled top-level sessions (tools/index.ts), but — like `yield` above —
2228
+ // an explicit `toolNames` list would otherwise drop them from the ACTIVE set,
2229
+ // leaving the nudge/guidance pointing at tools the model cannot call. Activate
2230
+ // exactly the builtins createTools built (`builtInToolNames` — provenance, so a
2231
+ // same-named custom/extension tool is never force-activated when auto-learn is
2232
+ // off) to keep guidance, controller, and the active set consistent.
2233
+ if (explicitlyRequestedToolNames) {
2234
+ for (const name of ["manage_skill", "learn"]) {
2235
+ if (builtInToolNames.includes(name) && !explicitlyRequestedToolNames.includes(name)) {
2236
+ explicitlyRequestedToolNames.push(name);
2237
+ }
2238
+ }
2239
+ }
2240
+ const requestedToolNames = explicitlyRequestedToolNames ?? toolNamesFromRegistry;
2241
+ const normalizedRequested = requestedToolNames.filter(name => toolRegistry.has(name));
2242
+ const requestedToolNameSet = new Set(normalizedRequested);
2243
+ // Effective discovery mode is resolved after the full registry exists so auto mode can count MCP/extension tools.
2244
+ const defaultInactiveToolNames = new Set(
2245
+ registeredTools.filter(tool => tool.definition.defaultInactive).map(tool => tool.definition.name),
2246
+ );
2247
+ const requestedActiveToolNames = normalizedRequested.filter(name => name !== "goal");
2248
+ const initialRequestedActiveToolNames = options.toolNames
2249
+ ? requestedActiveToolNames
2250
+ : requestedActiveToolNames.filter(name => !defaultInactiveToolNames.has(name));
2251
+ const explicitlyRequestedMCPToolNames = options.toolNames
2252
+ ? requestedActiveToolNames.filter(name => name.startsWith("mcp__"))
2253
+ : [];
2254
+ const discoveryDefaultServers = new Set(
2255
+ (settings.get("mcp.discoveryDefaultServers") ?? []).map(serverName => serverName.trim()).filter(Boolean),
2256
+ );
2257
+ const discoveryDefaultServerToolNames = mcpDiscoveryEnabled
2258
+ ? selectDiscoverableToolNamesByServer(
2259
+ filterBySource(collectDiscoverableTools(toolRegistry.values()), "mcp"),
2260
+ discoveryDefaultServers,
2261
+ )
2262
+ : [];
2263
+ let initialSelectedMCPToolNames: string[] = [];
2264
+ let defaultSelectedMCPToolNames: string[] = [];
2265
+ let initialToolNames = [...initialRequestedActiveToolNames];
2266
+ if (mcpDiscoveryEnabled) {
2267
+ const restoredSelectedMCPToolNames = existingSession.selectedMCPToolNames.filter(name =>
2268
+ toolRegistry.has(name),
2269
+ );
2270
+ defaultSelectedMCPToolNames = [
2271
+ ...new Set([...discoveryDefaultServerToolNames, ...explicitlyRequestedMCPToolNames]),
2272
+ ];
2273
+ initialSelectedMCPToolNames = existingSession.hasPersistedMCPToolSelection
2274
+ ? restoredSelectedMCPToolNames
2275
+ : [...new Set([...restoredSelectedMCPToolNames, ...defaultSelectedMCPToolNames])];
2276
+ initialToolNames = [
2277
+ ...new Set([
2278
+ ...initialRequestedActiveToolNames.filter(name => !name.startsWith("mcp__")),
2279
+ ...initialSelectedMCPToolNames,
2280
+ ]),
2281
+ ];
2282
+ }
2283
+
2284
+ // Custom tools and extension-registered tools are always included regardless of toolNames filter
2285
+ const alwaysInclude: string[] = [
2286
+ ...sdkCustomTools.map(t => (isCustomTool(t) ? t.name : t.name)),
2287
+ ...registeredTools.filter(t => !t.definition.defaultInactive).map(t => t.definition.name),
2288
+ ];
2289
+ for (const name of alwaysInclude) {
2290
+ if (mcpDiscoveryEnabled && name.startsWith("mcp__")) {
2291
+ continue;
2292
+ }
2293
+ if (toolRegistry.has(name) && !initialToolNames.includes(name)) {
2294
+ initialToolNames.push(name);
2295
+ }
2296
+ }
2297
+
2298
+ // When tools.discoveryMode === "all", hide non-essential built-in discoverable tools
2299
+ // from the initial set unless they were explicitly requested or restored from persistence.
2300
+ // The model finds them via search_tool_bm25 and activates them on demand.
2301
+ if (effectiveDiscoveryMode === "all") {
2302
+ // Tools a forced tool_choice will target must stay active, or the named
2303
+ // choice references a tool absent from the request (provider 400). Eager
2304
+ // todos force a named `todo` choice on the first turn. `task` is also kept
2305
+ // active under discovery-all when `task.eager` is not `default`, so eager delegation is
2306
+ // possible and the Eager Tasks prompt section renders, even though nothing
2307
+ // forces a `task` tool_choice.
2308
+ const forceActive = new Set<string>();
2309
+ if (settings.get("todo.eager") !== "default" && settings.get("todo.enabled") && toolRegistry.has("todo")) {
2310
+ forceActive.add("todo");
2311
+ }
2312
+ if (settings.get("task.eager") !== "default" && toolRegistry.has("task")) {
2313
+ forceActive.add("task");
2314
+ }
2315
+ initialToolNames = filterInitialToolsForDiscoveryAll(initialToolNames, {
2316
+ loadModeOf: name => toolRegistry.get(name)?.loadMode,
2317
+ essentialNames: new Set(computeEssentialBuiltinNames(settings)),
2318
+ explicitlyRequested: new Set(options.toolNames?.map(name => name.toLowerCase()) ?? []),
2319
+ // Back-compat: persisted activations live under selectedMCPToolNames today (built-in
2320
+ // activation persistence is a follow-up). MCP names won't collide with built-in names.
2321
+ restored: new Set(existingSession.selectedMCPToolNames),
2322
+ forceActive,
2323
+ });
2324
+ }
2325
+
2326
+ // Pre-register in the global agent registry BEFORE building the system prompt,
2327
+ // so that subagents launched in the same parallel batch can see each other in
2328
+ // their initial `# IRC Peers` block (rendered inside `rebuildSystemPrompt`).
2329
+ // The session reference is attached after construction below.
2330
+ agentRegistry.register({
2331
+ id: resolvedAgentId,
2332
+ displayName: resolvedAgentDisplayName,
2333
+ kind: agentKind,
2334
+ parentId: options.parentAgentId,
2335
+ session: null,
2336
+ sessionFile: sessionManager.getSessionFile() ?? null,
2337
+ status: "running",
2338
+ });
2339
+ hasRegistered = true;
2340
+
2341
+ const { systemPrompt } = await logger.time(
2342
+ "buildSystemPrompt",
2343
+ rebuildSystemPrompt,
2344
+ initialToolNames,
2345
+ toolRegistry,
2346
+ );
2347
+
2348
+ const promptTemplates = await promptTemplatesPromise;
2349
+ toolSession.promptTemplates = promptTemplates;
2350
+
2351
+ const slashCommands = await slashCommandsPromise;
2352
+
2353
+ // Create convertToLlm wrapper that filters images if blockImages is enabled (defense-in-depth)
2354
+ const convertToLlmWithBlockImages = (messages: AgentMessage[]): Message[] => {
2355
+ const converted = convertToLlm(messages);
2356
+ // Check setting dynamically so mid-session changes take effect
2357
+ if (!settings.get("images.blockImages")) {
2358
+ return converted;
2359
+ }
2360
+ // Filter out ImageContent from all messages, replacing with text placeholder
2361
+ return converted.map(msg => {
2362
+ if (msg.role === "user" || msg.role === "toolResult") {
2363
+ const content = msg.content;
2364
+ if (Array.isArray(content)) {
2365
+ const hasImages = content.some(c => c.type === "image");
2366
+ if (hasImages) {
2367
+ const filteredContent = content
2368
+ .map(c =>
2369
+ c.type === "image" ? { type: "text" as const, text: "Image reading is disabled." } : c,
2370
+ )
2371
+ .filter((c, i, arr) => {
2372
+ // Dedupe consecutive "Image reading is disabled." texts
2373
+ if (!(c.type === "text" && c.text === "Image reading is disabled." && i > 0)) return true;
2374
+ const prev = arr[i - 1];
2375
+ return !(prev.type === "text" && prev.text === "Image reading is disabled.");
2376
+ });
2377
+ return { ...msg, content: filteredContent };
2378
+ }
2379
+ }
2380
+ }
2381
+ return msg;
2382
+ });
2383
+ };
2384
+
2385
+ // Final convertToLlm: chain block-images filter with secret obfuscation
2386
+ const convertToLlmFinal = (messages: AgentMessage[]): Message[] => {
2387
+ const converted = convertToLlmWithBlockImages(messages);
2388
+ if (!obfuscator?.hasSecrets()) return converted;
2389
+ return obfuscateMessages(obfuscator, converted);
2390
+ };
2391
+
2392
+ const transformContext = async (messages: AgentMessage[], _signal?: AbortSignal) => {
2393
+ const withContext = await extensionRunner.emitContext(messages);
2394
+ return wrapSteeringForModel(withContext);
2395
+ };
2396
+ // Per-request provider-context transforms. Obfuscate FIRST so secrets are
2397
+ // redacted from text before snapcompact rasterizes it into PNG frames.
2398
+ // Both operate on the transient outgoing Context only — never persisted.
2399
+ const snapcompactSystemPromptMode = settings.get("snapcompact.systemPrompt");
2400
+ const snapcompactInline =
2401
+ snapcompactSystemPromptMode !== "none" || settings.get("snapcompact.toolResults")
2402
+ ? new SnapcompactInlineTransformer(
2403
+ {
2404
+ renderSystemPrompt: snapcompactSystemPromptMode,
2405
+ renderToolResults: settings.get("snapcompact.toolResults"),
2406
+ shape: settings.get("snapcompact.shape"),
2407
+ },
2408
+ // Journal the tokens each imaged tool result keeps off the wire
2409
+ // (frames never reach session.jsonl, so this is their only trace).
2410
+ createSnapcompactSavingsRecorder(() => sessionManager.getSessionFile() ?? null),
2411
+ )
2412
+ : undefined;
2413
+ const transformProviderContext =
2414
+ obfuscator || snapcompactInline
2415
+ ? (context: Context, transformModel: Model): Context => {
2416
+ let transformed = obfuscator ? obfuscateProviderContext(obfuscator, context) : context;
2417
+ if (snapcompactInline) transformed = snapcompactInline.transform(transformed, transformModel);
2418
+ return transformed;
2419
+ }
2420
+ : undefined;
2421
+ const onPayload = async (payload: unknown, _model?: Model) => {
2422
+ return await extensionRunner.emitBeforeProviderRequest(payload);
2423
+ };
2424
+ const onResponse: SimpleStreamOptions["onResponse"] = async (response, model) => {
2425
+ await extensionRunner.emitAfterProviderResponse(response, model);
2426
+ };
2427
+
2428
+ const setToolUIContext = (uiContext: ExtensionUIContext, hasUI: boolean) => {
2429
+ toolContextStore.setUIContext(uiContext, hasUI);
2430
+ };
2431
+
2432
+ const initialTools = initialToolNames
2433
+ .map(name => toolRegistry.get(name))
2434
+ .filter((tool): tool is AgentTool => tool !== undefined);
2435
+
2436
+ const openaiWebsocketSetting = settings.get("providers.openaiWebsockets") ?? "off";
2437
+ const preferOpenAICodexWebsockets =
2438
+ openaiWebsocketSetting === "on" ? true : openaiWebsocketSetting === "off" ? false : undefined;
2439
+ const serviceTierSetting = settings.get("serviceTier");
2440
+
2441
+ const initialServiceTier = hasServiceTierEntry
2442
+ ? existingSession.serviceTier
2443
+ : serviceTierSetting === "none"
2444
+ ? undefined
2445
+ : serviceTierSetting;
2446
+
2447
+ // One-shot launch-latency marker: fired the first time the loop dispatches
2448
+ // a chat request to the provider transport. See onFirstChatDispatch.
2449
+ let notifyFirstChatDispatch = options.onFirstChatDispatch;
2450
+ agent = new Agent({
2451
+ initialState: {
2452
+ systemPrompt,
2453
+ model,
2454
+ thinkingLevel: toReasoningEffort(effectiveThinkingLevel),
2455
+ disableReasoning: shouldDisableReasoning(effectiveThinkingLevel),
2456
+ tools: initialTools,
2457
+ },
2458
+ convertToLlm: convertToLlmFinal,
2459
+ onPayload,
2460
+ onResponse,
2461
+ sessionId: providerSessionId,
2462
+ promptCacheKey: options.providerPromptCacheKey,
2463
+ deadline: options.deadline,
2464
+ transformContext,
2465
+ transformProviderContext,
2466
+ steeringMode: settings.get("steeringMode") ?? "one-at-a-time",
2467
+ followUpMode: settings.get("followUpMode") ?? "one-at-a-time",
2468
+ interruptMode: settings.get("interruptMode") ?? "immediate",
2469
+ thinkingBudgets: settings.getGroup("thinkingBudgets"),
2470
+ temperature: settings.get("temperature") >= 0 ? settings.get("temperature") : undefined,
2471
+ topP: settings.get("topP") >= 0 ? settings.get("topP") : undefined,
2472
+ topK: settings.get("topK") >= 0 ? settings.get("topK") : undefined,
2473
+ minP: settings.get("minP") >= 0 ? settings.get("minP") : undefined,
2474
+ presencePenalty: settings.get("presencePenalty") >= 0 ? settings.get("presencePenalty") : undefined,
2475
+ repetitionPenalty: settings.get("repetitionPenalty") >= 0 ? settings.get("repetitionPenalty") : undefined,
2476
+ serviceTier: initialServiceTier,
2477
+ hideThinkingSummary: settings.get("hideThinkingBlock"),
2478
+ kimiApiFormat: settings.get("providers.kimiApiFormat") ?? "anthropic",
2479
+ preferWebsockets: preferOpenAICodexWebsockets,
2480
+ getToolContext: tc => toolContextStore.getContext(tc),
2481
+ getApiKey: requestModel => modelRegistry.resolver(requestModel, agent.sessionId),
2482
+ streamFn: (streamModel, context, streamOptions) => {
2483
+ if (notifyFirstChatDispatch) {
2484
+ const cb = notifyFirstChatDispatch;
2485
+ notifyFirstChatDispatch = undefined;
2486
+ try {
2487
+ cb();
2488
+ } catch (err) {
2489
+ logger.warn("onFirstChatDispatch hook threw", {
2490
+ error: err instanceof Error ? err.message : String(err),
2491
+ });
2492
+ }
2493
+ }
2494
+ const openrouterRoutingPreset = settings.get("providers.openrouterVariant");
2495
+ const openrouterVariant =
2496
+ openrouterRoutingPreset && openrouterRoutingPreset !== "default" ? openrouterRoutingPreset : undefined;
2497
+ const antigravityEndpointMode = settings.get("providers.antigravityEndpoint");
2498
+ return streamSimple(streamModel, context, {
2499
+ ...streamOptions,
2500
+ openrouterVariant: streamOptions?.openrouterVariant ?? openrouterVariant,
2501
+ antigravityEndpointMode: streamOptions?.antigravityEndpointMode ?? antigravityEndpointMode,
2502
+ loopGuard: {
2503
+ enabled: settings.get("model.loopGuard.enabled"),
2504
+ checkAssistantContent: settings.get("model.loopGuard.checkAssistantContent"),
2505
+ ...streamOptions?.loopGuard,
2506
+ },
2507
+ });
2508
+ },
2509
+ cursorExecHandlers,
2510
+ transformToolCallArguments: (args, _toolName) => {
2511
+ let result = args;
2512
+ const maxTimeout = settings.get("tools.maxTimeout");
2513
+ if (maxTimeout > 0 && typeof result.timeout === "number") {
2514
+ result = { ...result, timeout: Math.min(result.timeout, maxTimeout) };
2515
+ }
2516
+ if (obfuscator?.hasSecrets()) {
2517
+ result = obfuscator.deobfuscateObject(result);
2518
+ }
2519
+ return result;
2520
+ },
2521
+ intentTracing: !!intentField,
2522
+ pruneToolDescriptions: inlineToolDescriptors,
2523
+ dialect: resolveDialect(settings.get("tools.format"), model),
2524
+ abortOnFabricatedToolResult: settings.get("tools.abortOnFabricatedResult"),
2525
+ getToolChoice: () => session?.nextToolChoiceDirective(),
2526
+ telemetry: options.telemetry,
2527
+ appendOnlyContext: model
2528
+ ? shouldEnableAppendOnlyContext(settings.get("provider.appendOnlyContext"), model)
2529
+ ? new AppendOnlyContextManager()
2530
+ : undefined
2531
+ : undefined,
2532
+ });
2533
+
2534
+ cursorEventEmitter = event => agent.emitExternalEvent(event);
2535
+
2536
+ // Restore messages if session has existing data
2537
+ if (hasExistingSession) {
2538
+ agent.replaceMessages(existingSession.messages);
2539
+ } else {
2540
+ // Save initial model, thinking level, and service tier for new sessions so they can be restored on resume.
2541
+ if (model) {
2542
+ sessionManager.appendModelChange(`${model.provider}/${model.id}`);
2543
+ }
2544
+ if (!autoThinking) {
2545
+ // Do not write the `auto` selector before the first turn resolves; auto
2546
+ // classification persists its concrete effort once a real user turn runs.
2547
+ sessionManager.appendThinkingLevelChange(effectiveThinkingLevel);
2548
+ }
2549
+ if (initialServiceTier) {
2550
+ sessionManager.appendServiceTierChange(initialServiceTier);
2551
+ }
2552
+ }
2553
+
2554
+ // Hard-isolated read-only toolset for the advisor (built unconditionally so
2555
+ // it can be toggled at runtime). Fresh ReadTool/SearchTool/FindTool bound to a
2556
+ // DISTINCT ToolSession so the advisor's investigative reads never touch the
2557
+ // primary's snapshot, seen-lines, conflict, or summary caches (all keyed on
2558
+ // session identity). `cwd` stays dynamic; edit/yield capabilities are off.
2559
+ const advisorToolSession: ToolSession = {
2560
+ ...toolSession,
2561
+ get cwd() {
2562
+ return sessionManager.getCwd();
2563
+ },
2564
+ hasEditTool: false,
2565
+ requireYieldTool: false,
2566
+ conflictHistory: undefined,
2567
+ fileSnapshotStore: undefined,
2568
+ getSessionId: () => {
2569
+ const id = sessionManager.getSessionId?.();
2570
+ return id ? `${id}-advisor` : null;
2571
+ },
2572
+ getAgentId: () => "advisor",
2573
+ };
2574
+ const built = await Promise.all(
2575
+ [...ADVISOR_READONLY_TOOL_NAMES].map(name =>
2576
+ BUILTIN_TOOLS[name as keyof typeof BUILTIN_TOOLS](advisorToolSession),
2577
+ ),
2578
+ );
2579
+ const advisorReadOnlyTools: Tool[] = built
2580
+ .filter((tool): tool is Tool => tool != null)
2581
+ .map(wrapToolWithMetaNotice);
2582
+
2583
+ let advisorWatchdogPrompt: string | undefined;
2584
+ if (watchdogFiles && watchdogFiles.length > 0) {
2585
+ advisorWatchdogPrompt = watchdogFiles.join("\n\n");
2586
+ }
2587
+ // Owned only when this session created the manager; subagents receive a
2588
+ // parent's manager via `options.mcpManager` and MUST NOT disconnect it.
2589
+ const ownedMcpManager = options.mcpManager ? undefined : mcpManager;
2590
+ session = new AgentSession({
2591
+ advisorWatchdogPrompt,
2592
+ agent,
2593
+ pruneToolDescriptions: inlineToolDescriptors,
2594
+ thinkingLevel: autoThinking ? AUTO_THINKING : effectiveThinkingLevel,
2595
+ sessionManager,
2596
+ settings,
2597
+ autoApprove: options.autoApprove,
2598
+ evalKernelOwnerId,
2599
+ // Defined only for top-level sessions (creation is gated above).
2600
+ // AgentSession uses this to decide whether it may dispose the global
2601
+ // AsyncJobManager on teardown; subagents inherit the parent's and
2602
+ // **MUST NOT** tear it down.
2603
+ ownedAsyncJobManager: asyncJobManager,
2604
+ asyncJobManager: scopedAsyncJobManager,
2605
+ scopedModels: options.scopedModels,
2606
+ promptTemplates,
2607
+ slashCommands,
2608
+ extensionRunner,
2609
+ customCommands: customCommandsResult.commands,
2610
+ skills,
2611
+ skillWarnings,
2612
+ skillsSettings: settings.getGroup("skills"),
2613
+ modelRegistry,
2614
+ toolRegistry,
2615
+ transformContext,
2616
+ onPayload,
2617
+ onResponse,
2618
+ convertToLlm: convertToLlmFinal,
2619
+ rebuildSystemPrompt,
2620
+ reloadSshTool,
2621
+ requestedToolNames: requestedToolNameSet,
2622
+ getMcpServerInstructions: mcpManager
2623
+ ? () => {
2624
+ const raw = mcpManager.getServerInstructions();
2625
+ if (!raw || raw.size === 0) return raw;
2626
+ const out = new Map<string, string>();
2627
+ for (const [name, text] of raw) {
2628
+ out.set(
2629
+ name,
2630
+ text.length > MAX_MCP_INSTRUCTIONS_LENGTH ? text.slice(0, MAX_MCP_INSTRUCTIONS_LENGTH) : text,
2631
+ );
2632
+ }
2633
+ return out;
2634
+ }
2635
+ : undefined,
2636
+ disconnectOwnedMcpManager: ownedMcpManager ? () => ownedMcpManager.disconnectAll() : undefined,
2637
+ mcpDiscoveryEnabled,
2638
+ initialSelectedMCPToolNames,
2639
+ defaultSelectedMCPToolNames,
2640
+ persistInitialMCPToolSelection: !hasExistingSession,
2641
+ defaultSelectedMCPServerNames: [...discoveryDefaultServers],
2642
+ ttsrManager,
2643
+ obfuscator,
2644
+ agentId: resolvedAgentId,
2645
+ agentKind,
2646
+ providerSessionId: options.providerSessionId,
2647
+ parentEvalSessionId: options.parentEvalSessionId,
2648
+ advisorReadOnlyTools,
2649
+ });
2650
+ hasSession = true;
2651
+ if (asyncJobManager) {
2652
+ session.yieldQueue.register<AsyncResultEntry>("async-result", {
2653
+ isStale: entry => asyncJobManager.isDeliverySuppressed(entry.jobId),
2654
+ build: buildAsyncResultBatchMessage,
2655
+ });
2656
+ }
2657
+ session.yieldQueue.register<McpNotificationEntry>("mcp-notification", {
2658
+ build: buildMcpNotificationBatchMessage,
2659
+ });
2660
+ session.yieldQueue.register<DeferredDiagnosticsEntry>(LSP_LATE_DIAGNOSTIC_MESSAGE_TYPE, {
2661
+ isStale: entry => entry.isStale(),
2662
+ build: buildLateDiagnosticsBatchMessage,
2663
+ });
2664
+
2665
+ // Attach the live session to the pre-registered ref so peers can route IRC
2666
+ // messages here. Refresh sessionFile in case it was unavailable at pre-register
2667
+ // time. The dispose wrapper below unregisters on teardown (unless parked).
2668
+ agentRegistry.attachSession(resolvedAgentId, session, sessionManager.getSessionFile() ?? null);
2669
+ {
2670
+ const originalDispose = session.dispose.bind(session);
2671
+ session.dispose = async () => {
2672
+ try {
2673
+ // Reject new session work (Python/eval starts) the moment disposal
2674
+ // begins — the lifecycle await below opens an async gap before
2675
+ // AgentSession.dispose() would otherwise set its guards.
2676
+ session.beginDispose();
2677
+ if (agentKind === "main") {
2678
+ // Top-level teardown owns the global agent lifecycle: park timers,
2679
+ // adopted subagent sessions, revivers. Tear it down while shared
2680
+ // resources (kernels, MCP, LSP) are still live. Subagent disposal
2681
+ // must NOT touch the global lifecycle.
2682
+ await AgentLifecycleManager.global().dispose();
2683
+ }
2684
+ await originalDispose();
2685
+ } finally {
2686
+ unregisterUnlessParked();
2687
+ unsubscribeCredentialDisabled?.();
2688
+ }
2689
+ };
2690
+ }
2691
+
2692
+ if (model?.api === "openai-codex-responses") {
2693
+ // `.api` equality doesn't narrow the generic; the guard makes this cast sound.
2694
+ const codexModel = model as Model<"openai-codex-responses">;
2695
+ const codexTransport = getOpenAICodexTransportDetails(codexModel, {
2696
+ sessionId: providerSessionId,
2697
+ baseUrl: codexModel.baseUrl,
2698
+ preferWebsockets: preferOpenAICodexWebsockets,
2699
+ providerSessionState: session.providerSessionState,
2700
+ });
2701
+ if (codexTransport.websocketPreferred) {
2702
+ void (async () => {
2703
+ try {
2704
+ const codexPrewarmApiKey = await modelRegistry.getApiKey(codexModel, providerSessionId);
2705
+ if (!codexPrewarmApiKey) return;
2706
+ await logger.time("prewarmOpenAICodexResponses", prewarmOpenAICodexResponses, codexModel, {
2707
+ apiKey: codexPrewarmApiKey,
2708
+ sessionId: providerSessionId,
2709
+ preferWebsockets: preferOpenAICodexWebsockets,
2710
+ providerSessionState: session.providerSessionState,
2711
+ });
2712
+ } catch (error) {
2713
+ const errorMessage = error instanceof Error ? error.message : String(error);
2714
+ logger.debug("Codex websocket prewarm failed", {
2715
+ error: errorMessage,
2716
+ provider: codexModel.provider,
2717
+ model: codexModel.id,
2718
+ });
2719
+ }
2720
+ })();
2721
+ }
2722
+ }
2723
+
2724
+ // Start LSP warmup in the background so startup does not block on language server initialization.
2725
+ // With `lsp.lazy` (the default) the warmup is skipped: recognized servers are still discovered and
2726
+ // surfaced in the UI as "available", but cold-start on first use — the lsp tool or an edit/write
2727
+ // touching a matching file type — through `getOrCreateClient`.
2728
+ // Print/script invocations (`hasUI=false`) skip it regardless: they don't render the warmup status
2729
+ // indicator AND typically finish before LSP servers would have stabilized — warming them just spends
2730
+ // CPU parsing big `initialize` responses concurrently with the LLM stream consumer, jittering
2731
+ // perceived latency.
2732
+ let lspServers: CreateAgentSessionResult["lspServers"];
2733
+ if (enableLsp && options.hasUI && settings.get("lsp.lazy")) {
2734
+ lspServers = discoverStartupLspServers(cwd, "available");
2735
+ } else if (enableLsp && options.hasUI) {
2736
+ lspServers = discoverStartupLspServers(cwd);
2737
+ if (lspServers.length > 0) {
2738
+ void (async () => {
2739
+ try {
2740
+ const result = await logger.time("warmupLspServers", warmupLspServers, cwd);
2741
+ const serversByName = new Map(result.servers.map(server => [server.name, server] as const));
2742
+ for (const server of lspServers ?? []) {
2743
+ const next = serversByName.get(server.name);
2744
+ if (!next) continue;
2745
+ server.status = next.status;
2746
+ server.fileTypes = next.fileTypes;
2747
+ server.error = next.error;
2748
+ }
2749
+ const event: LspStartupEvent = {
2750
+ type: "completed",
2751
+ servers: result.servers,
2752
+ };
2753
+ if (!startupQuiet) eventBus.emit(LSP_STARTUP_EVENT_CHANNEL, event);
2754
+ } catch (error) {
2755
+ const errorMessage = error instanceof Error ? error.message : String(error);
2756
+ logger.warn("LSP server warmup failed", { cwd, error: errorMessage });
2757
+ for (const server of lspServers ?? []) {
2758
+ server.status = "error";
2759
+ server.error = errorMessage;
2760
+ }
2761
+ const event: LspStartupEvent = {
2762
+ type: "failed",
2763
+ error: errorMessage,
2764
+ };
2765
+ if (!startupQuiet) eventBus.emit(LSP_STARTUP_EVENT_CHANNEL, event);
2766
+ }
2767
+ })();
2768
+ }
2769
+ }
2770
+
2771
+ const startMemoryBackend = async () => {
2772
+ const memoryBackend = await resolveMemoryBackend(settings);
2773
+ await memoryBackend.start({
2774
+ session,
2775
+ settings,
2776
+ modelRegistry,
2777
+ agentDir,
2778
+ taskDepth,
2779
+ parentHindsightSessionState: options.parentHindsightSessionState,
2780
+ parentMnemopiSessionState: options.parentMnemopiSessionState,
2781
+ });
2782
+ };
2783
+
2784
+ // Auto-learn can immediately trigger a synthetic capture turn after the
2785
+ // first real stop. When a memory backend is selected, install that backend's
2786
+ // per-session state first so the capture turn's `learn` tool observes the
2787
+ // same initialized state as normal memory tools. Other sessions keep memory
2788
+ // startup in the background to preserve the existing startup profile.
2789
+ //
2790
+ // Gated on `autolearn.enabled` to match the tools: `createTools` builds the
2791
+ // `learn`/`manage_skill` registry ONCE at session start and no settings
2792
+ // change rebuilds it, so installing the controller while disabled would let a
2793
+ // mid-session enable fire a nudge pointing at tools the session never built.
2794
+ // Activation is therefore a session-start decision for BOTH the controller
2795
+ // and the tools; the fire-time re-check in `#onAgentEnd` still handles a
2796
+ // mid-session DISABLE. The subscription lives for the session's lifetime; the
2797
+ // reference is intentionally discarded (the listener retains it).
2798
+ if (settings.get("autolearn.enabled") && taskDepth === 0) {
2799
+ await logger.time("startMemoryStartupTask", startMemoryBackend);
2800
+ new AutoLearnController({ session, settings });
2801
+ } else {
2802
+ void logger.time("startMemoryStartupTask", startMemoryBackend);
2803
+ }
2804
+
2805
+ // Wire MCP manager callbacks to session for reactive tool updates.
2806
+ // Skip when reusing a parent's manager — the parent owns the callbacks.
2807
+ if (mcpManager && !options.mcpManager) {
2808
+ mcpManager.setOnToolsChanged(tools => {
2809
+ void (async () => {
2810
+ try {
2811
+ await session.refreshMCPTools(
2812
+ tools,
2813
+ deferMCPDiscoveryForUI && !mcpDiscoveryEnabled && options.toolNames === undefined
2814
+ ? { activateAll: true }
2815
+ : undefined,
2816
+ );
2817
+ if (deferMCPDiscoveryForUI && !mcpDiscoveryEnabled && explicitlyRequestedMCPToolNames.length > 0) {
2818
+ await session.setActiveToolsByName([
2819
+ ...session.getActiveToolNames(),
2820
+ ...explicitlyRequestedMCPToolNames,
2821
+ ]);
2822
+ }
2823
+ } catch (error) {
2824
+ logger.warn("MCP tool refresh failed", {
2825
+ error: error instanceof Error ? error.message : String(error),
2826
+ });
2827
+ }
2828
+ })();
2829
+ });
2830
+ // Wire prompt refresh → rebuild MCP prompt slash commands
2831
+ mcpManager.setOnPromptsChanged(serverName => {
2832
+ const promptCommands = buildMCPPromptCommands(mcpManager);
2833
+ session.setMCPPromptCommands(promptCommands);
2834
+ logger.debug("MCP prompt commands refreshed", { path: `mcp:${serverName}` });
2835
+ });
2836
+ const notificationDebounceTimers = new Map<string, Timer>();
2837
+ const clearDebounceTimers = () => {
2838
+ for (const timer of notificationDebounceTimers.values()) clearTimeout(timer);
2839
+ notificationDebounceTimers.clear();
2840
+ };
2841
+ postmortem.register("mcp-notification-cleanup", clearDebounceTimers);
2842
+ mcpManager.setOnResourcesChanged((serverName, uri) => {
2843
+ logger.debug("MCP resources changed", { path: `mcp:${serverName}`, uri });
2844
+ if (!settings.get("mcp.notifications")) return;
2845
+ const debounceMs = settings.get("mcp.notificationDebounceMs");
2846
+ const key = `${serverName}:${uri}`;
2847
+ const existing = notificationDebounceTimers.get(key);
2848
+ if (existing) clearTimeout(existing);
2849
+ notificationDebounceTimers.set(
2850
+ key,
2851
+ setTimeout(() => {
2852
+ notificationDebounceTimers.delete(key);
2853
+ // Re-check: user may have disabled notifications during the debounce window
2854
+ if (!settings.get("mcp.notifications")) return;
2855
+ session.yieldQueue.enqueue<McpNotificationEntry>("mcp-notification", { serverName, uri });
2856
+ }, debounceMs),
2857
+ );
2858
+ });
2859
+ }
2860
+
2861
+ startDeferredMCPDiscovery?.(session, {
2862
+ mcpDiscoveryEnabled,
2863
+ explicitlyRequestedMCPToolNames,
2864
+ activateAllMCPTools: !mcpDiscoveryEnabled && options.toolNames === undefined,
2865
+ });
2866
+
2867
+ return {
2868
+ session,
2869
+ extensionsResult,
2870
+ setToolUIContext,
2871
+ mcpManager,
2872
+ modelFallbackMessage,
2873
+ lspServers,
2874
+ eventBus,
2875
+ };
2876
+ } catch (error) {
2877
+ // Release the subscription if the throw happened after install but before the
2878
+ // dispose-wrap took ownership. Idempotent with dispose() — Set.delete is a no-op
2879
+ // for already-removed listeners.
2880
+ unsubscribeCredentialDisabled?.();
2881
+ try {
2882
+ if (hasSession) {
2883
+ await session.dispose();
2884
+ } else {
2885
+ if (hasRegistered) unregisterUnlessParked();
2886
+ if (asyncJobManager) {
2887
+ if (AsyncJobManager.instance() === asyncJobManager) {
2888
+ AsyncJobManager.setInstance(undefined);
2889
+ }
2890
+ await asyncJobManager.dispose({ timeoutMs: 3_000 });
2891
+ }
2892
+ await disposeKernelSessionsByOwner(evalKernelOwnerId);
2893
+ if (ownsAuthStorage) authStorage.close();
2894
+ }
2895
+ } catch (cleanupError) {
2896
+ logger.warn("Failed to clean up createAgentSession resources after startup error", {
2897
+ error: cleanupError instanceof Error ? cleanupError.message : String(cleanupError),
2898
+ });
2899
+ }
2900
+ throw error;
2901
+ }
2902
+ }
2903
+
2904
+ /**
2905
+ * Best-effort preconnect to the model's API host. Bun's `fetch.preconnect`
2906
+ * primes DNS + TCP + TLS + H2 so the first real request reuses the warm
2907
+ * connection. Errors are swallowed: preconnect is an optimization, never a
2908
+ * hard dependency.
2909
+ */
2910
+ function preconnectModelHost(baseUrl: string | undefined): void {
2911
+ if (!baseUrl) return;
2912
+ const preconnect = (globalThis.fetch as typeof fetch & { preconnect?: (url: string) => void }).preconnect;
2913
+ if (typeof preconnect !== "function") return;
2914
+ try {
2915
+ preconnect(baseUrl);
2916
+ } catch {
2917
+ // Best effort.
2918
+ }
2919
+ }