@xelauvas/xela-cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1920) hide show
  1. package/README.md +200 -0
  2. package/bin/xela +100 -0
  3. package/package.json +88 -0
  4. package/src/QueryEngine.ts +1295 -0
  5. package/src/Task.ts +125 -0
  6. package/src/Tool.ts +792 -0
  7. package/src/_shims/_generated_stubs/_universal_stub.mjs +168 -0
  8. package/src/_shims/_generated_stubs/text_stub.mjs +1 -0
  9. package/src/_shims/bun_bundle.js +9 -0
  10. package/src/_shims/cjs_stub.cjs +23 -0
  11. package/src/_shims/empty_stub.js +33 -0
  12. package/src/_shims/loader.js +352 -0
  13. package/src/_shims/openai_adapter.ts +486 -0
  14. package/src/_shims/react_compiler_runtime.js +17 -0
  15. package/src/_shims/register.js +148 -0
  16. package/src/assistant/sessionHistory.ts +87 -0
  17. package/src/bootstrap/state.ts +1759 -0
  18. package/src/bridge/bridgeApi.ts +539 -0
  19. package/src/bridge/bridgeConfig.ts +48 -0
  20. package/src/bridge/bridgeDebug.ts +135 -0
  21. package/src/bridge/bridgeEnabled.ts +202 -0
  22. package/src/bridge/bridgeMain.ts +2999 -0
  23. package/src/bridge/bridgeMessaging.ts +461 -0
  24. package/src/bridge/bridgePermissionCallbacks.ts +43 -0
  25. package/src/bridge/bridgePointer.ts +210 -0
  26. package/src/bridge/bridgeStatusUtil.ts +163 -0
  27. package/src/bridge/bridgeUI.ts +530 -0
  28. package/src/bridge/capacityWake.ts +56 -0
  29. package/src/bridge/codeSessionApi.ts +168 -0
  30. package/src/bridge/createSession.ts +384 -0
  31. package/src/bridge/debugUtils.ts +141 -0
  32. package/src/bridge/envLessBridgeConfig.ts +165 -0
  33. package/src/bridge/flushGate.ts +71 -0
  34. package/src/bridge/inboundAttachments.ts +175 -0
  35. package/src/bridge/inboundMessages.ts +80 -0
  36. package/src/bridge/initReplBridge.ts +569 -0
  37. package/src/bridge/jwtUtils.ts +256 -0
  38. package/src/bridge/pollConfig.ts +110 -0
  39. package/src/bridge/pollConfigDefaults.ts +82 -0
  40. package/src/bridge/remoteBridgeCore.ts +1008 -0
  41. package/src/bridge/replBridge.ts +2406 -0
  42. package/src/bridge/replBridgeHandle.ts +36 -0
  43. package/src/bridge/replBridgeTransport.ts +370 -0
  44. package/src/bridge/sessionIdCompat.ts +57 -0
  45. package/src/bridge/sessionRunner.ts +550 -0
  46. package/src/bridge/trustedDevice.ts +210 -0
  47. package/src/bridge/types.ts +262 -0
  48. package/src/bridge/workSecret.ts +127 -0
  49. package/src/buddy/CompanionSprite.tsx +371 -0
  50. package/src/buddy/companion.ts +133 -0
  51. package/src/buddy/prompt.ts +36 -0
  52. package/src/buddy/sprites.ts +514 -0
  53. package/src/buddy/types.ts +148 -0
  54. package/src/buddy/useBuddyNotification.tsx +98 -0
  55. package/src/cli/exit.ts +31 -0
  56. package/src/cli/handlers/agents.ts +70 -0
  57. package/src/cli/handlers/auth.ts +330 -0
  58. package/src/cli/handlers/autoMode.ts +170 -0
  59. package/src/cli/handlers/mcp.tsx +362 -0
  60. package/src/cli/handlers/plugins.ts +878 -0
  61. package/src/cli/handlers/util.tsx +110 -0
  62. package/src/cli/ndjsonSafeStringify.ts +32 -0
  63. package/src/cli/print.ts +5594 -0
  64. package/src/cli/remoteIO.ts +255 -0
  65. package/src/cli/structuredIO.ts +859 -0
  66. package/src/cli/transports/HybridTransport.ts +282 -0
  67. package/src/cli/transports/SSETransport.ts +711 -0
  68. package/src/cli/transports/SerialBatchEventUploader.ts +275 -0
  69. package/src/cli/transports/WebSocketTransport.ts +800 -0
  70. package/src/cli/transports/WorkerStateUploader.ts +131 -0
  71. package/src/cli/transports/ccrClient.ts +998 -0
  72. package/src/cli/transports/transportUtils.ts +45 -0
  73. package/src/cli/update.ts +422 -0
  74. package/src/commands/add-dir/add-dir.tsx +126 -0
  75. package/src/commands/add-dir/index.ts +11 -0
  76. package/src/commands/add-dir/validation.ts +110 -0
  77. package/src/commands/advisor.ts +109 -0
  78. package/src/commands/agents/agents.tsx +12 -0
  79. package/src/commands/agents/index.ts +10 -0
  80. package/src/commands/ant-trace/index.js +1 -0
  81. package/src/commands/autofix-pr/index.js +1 -0
  82. package/src/commands/backfill-sessions/index.js +1 -0
  83. package/src/commands/branch/branch.ts +296 -0
  84. package/src/commands/branch/index.ts +14 -0
  85. package/src/commands/break-cache/index.js +1 -0
  86. package/src/commands/bridge/bridge.tsx +509 -0
  87. package/src/commands/bridge/index.ts +26 -0
  88. package/src/commands/bridge-kick.ts +200 -0
  89. package/src/commands/brief.ts +130 -0
  90. package/src/commands/btw/btw.tsx +243 -0
  91. package/src/commands/btw/index.ts +13 -0
  92. package/src/commands/bughunter/index.js +1 -0
  93. package/src/commands/chrome/chrome.tsx +285 -0
  94. package/src/commands/chrome/index.ts +13 -0
  95. package/src/commands/clear/caches.ts +144 -0
  96. package/src/commands/clear/clear.ts +7 -0
  97. package/src/commands/clear/conversation.ts +251 -0
  98. package/src/commands/clear/index.ts +19 -0
  99. package/src/commands/color/color.ts +93 -0
  100. package/src/commands/color/index.ts +16 -0
  101. package/src/commands/commit-push-pr.ts +158 -0
  102. package/src/commands/commit.ts +92 -0
  103. package/src/commands/compact/compact.ts +287 -0
  104. package/src/commands/compact/index.ts +15 -0
  105. package/src/commands/config/config.tsx +7 -0
  106. package/src/commands/config/index.ts +11 -0
  107. package/src/commands/context/context-noninteractive.ts +325 -0
  108. package/src/commands/context/context.tsx +64 -0
  109. package/src/commands/context/index.ts +24 -0
  110. package/src/commands/copy/copy.tsx +371 -0
  111. package/src/commands/copy/index.ts +15 -0
  112. package/src/commands/cost/cost.ts +24 -0
  113. package/src/commands/cost/index.ts +23 -0
  114. package/src/commands/createMovedToPluginCommand.ts +65 -0
  115. package/src/commands/ctx_viz/index.js +1 -0
  116. package/src/commands/debug-tool-call/index.js +1 -0
  117. package/src/commands/desktop/desktop.tsx +9 -0
  118. package/src/commands/desktop/index.ts +26 -0
  119. package/src/commands/diff/diff.tsx +9 -0
  120. package/src/commands/diff/index.ts +8 -0
  121. package/src/commands/doctor/doctor.tsx +7 -0
  122. package/src/commands/doctor/index.ts +12 -0
  123. package/src/commands/effort/effort.tsx +183 -0
  124. package/src/commands/effort/index.ts +13 -0
  125. package/src/commands/env/index.js +1 -0
  126. package/src/commands/exit/exit.tsx +33 -0
  127. package/src/commands/exit/index.ts +12 -0
  128. package/src/commands/export/export.tsx +91 -0
  129. package/src/commands/export/index.ts +11 -0
  130. package/src/commands/extra-usage/extra-usage-core.ts +118 -0
  131. package/src/commands/extra-usage/extra-usage-noninteractive.ts +16 -0
  132. package/src/commands/extra-usage/extra-usage.tsx +17 -0
  133. package/src/commands/extra-usage/index.ts +31 -0
  134. package/src/commands/fast/fast.tsx +269 -0
  135. package/src/commands/fast/index.ts +26 -0
  136. package/src/commands/feedback/feedback.tsx +25 -0
  137. package/src/commands/feedback/index.ts +26 -0
  138. package/src/commands/files/files.ts +19 -0
  139. package/src/commands/files/index.ts +12 -0
  140. package/src/commands/good-claude/index.js +1 -0
  141. package/src/commands/heapdump/heapdump.ts +17 -0
  142. package/src/commands/heapdump/index.ts +12 -0
  143. package/src/commands/help/help.tsx +11 -0
  144. package/src/commands/help/index.ts +10 -0
  145. package/src/commands/hooks/hooks.tsx +13 -0
  146. package/src/commands/hooks/index.ts +11 -0
  147. package/src/commands/ide/ide.tsx +646 -0
  148. package/src/commands/ide/index.ts +11 -0
  149. package/src/commands/init-verifiers.ts +262 -0
  150. package/src/commands/init.ts +256 -0
  151. package/src/commands/insights.ts +3200 -0
  152. package/src/commands/install-github-app/ApiKeyStep.tsx +231 -0
  153. package/src/commands/install-github-app/CheckExistingSecretStep.tsx +190 -0
  154. package/src/commands/install-github-app/CheckGitHubStep.tsx +15 -0
  155. package/src/commands/install-github-app/ChooseRepoStep.tsx +211 -0
  156. package/src/commands/install-github-app/CreatingStep.tsx +65 -0
  157. package/src/commands/install-github-app/ErrorStep.tsx +85 -0
  158. package/src/commands/install-github-app/ExistingWorkflowStep.tsx +103 -0
  159. package/src/commands/install-github-app/InstallAppStep.tsx +94 -0
  160. package/src/commands/install-github-app/OAuthFlowStep.tsx +276 -0
  161. package/src/commands/install-github-app/SuccessStep.tsx +96 -0
  162. package/src/commands/install-github-app/WarningsStep.tsx +73 -0
  163. package/src/commands/install-github-app/index.ts +13 -0
  164. package/src/commands/install-github-app/install-github-app.tsx +587 -0
  165. package/src/commands/install-github-app/setupGitHubActions.ts +325 -0
  166. package/src/commands/install-slack-app/index.ts +12 -0
  167. package/src/commands/install-slack-app/install-slack-app.ts +30 -0
  168. package/src/commands/install.tsx +300 -0
  169. package/src/commands/issue/index.js +1 -0
  170. package/src/commands/keybindings/index.ts +13 -0
  171. package/src/commands/keybindings/keybindings.ts +53 -0
  172. package/src/commands/login/index.ts +14 -0
  173. package/src/commands/login/login.tsx +104 -0
  174. package/src/commands/logout/index.ts +10 -0
  175. package/src/commands/logout/logout.tsx +82 -0
  176. package/src/commands/mcp/addCommand.ts +280 -0
  177. package/src/commands/mcp/index.ts +12 -0
  178. package/src/commands/mcp/mcp.tsx +85 -0
  179. package/src/commands/mcp/xaaIdpCommand.ts +266 -0
  180. package/src/commands/memory/index.ts +10 -0
  181. package/src/commands/memory/memory.tsx +90 -0
  182. package/src/commands/mobile/index.ts +11 -0
  183. package/src/commands/mobile/mobile.tsx +274 -0
  184. package/src/commands/mock-limits/index.js +1 -0
  185. package/src/commands/model/index.ts +16 -0
  186. package/src/commands/model/model.tsx +297 -0
  187. package/src/commands/oauth-refresh/index.js +1 -0
  188. package/src/commands/onboarding/index.js +1 -0
  189. package/src/commands/output-style/index.ts +11 -0
  190. package/src/commands/output-style/output-style.tsx +7 -0
  191. package/src/commands/passes/index.ts +22 -0
  192. package/src/commands/passes/passes.tsx +24 -0
  193. package/src/commands/perf-issue/index.js +1 -0
  194. package/src/commands/permissions/index.ts +11 -0
  195. package/src/commands/permissions/permissions.tsx +10 -0
  196. package/src/commands/plan/index.ts +11 -0
  197. package/src/commands/plan/plan.tsx +122 -0
  198. package/src/commands/plugin/AddMarketplace.tsx +162 -0
  199. package/src/commands/plugin/BrowseMarketplace.tsx +802 -0
  200. package/src/commands/plugin/DiscoverPlugins.tsx +781 -0
  201. package/src/commands/plugin/ManageMarketplaces.tsx +838 -0
  202. package/src/commands/plugin/ManagePlugins.tsx +2215 -0
  203. package/src/commands/plugin/PluginErrors.tsx +124 -0
  204. package/src/commands/plugin/PluginOptionsDialog.tsx +357 -0
  205. package/src/commands/plugin/PluginOptionsFlow.tsx +135 -0
  206. package/src/commands/plugin/PluginSettings.tsx +1072 -0
  207. package/src/commands/plugin/PluginTrustWarning.tsx +32 -0
  208. package/src/commands/plugin/UnifiedInstalledCell.tsx +565 -0
  209. package/src/commands/plugin/ValidatePlugin.tsx +98 -0
  210. package/src/commands/plugin/index.tsx +11 -0
  211. package/src/commands/plugin/parseArgs.ts +103 -0
  212. package/src/commands/plugin/plugin.tsx +7 -0
  213. package/src/commands/plugin/pluginDetailsHelpers.tsx +117 -0
  214. package/src/commands/plugin/usePagination.ts +171 -0
  215. package/src/commands/pr_comments/index.ts +50 -0
  216. package/src/commands/privacy-settings/index.ts +14 -0
  217. package/src/commands/privacy-settings/privacy-settings.tsx +58 -0
  218. package/src/commands/rate-limit-options/index.ts +19 -0
  219. package/src/commands/rate-limit-options/rate-limit-options.tsx +210 -0
  220. package/src/commands/release-notes/index.ts +11 -0
  221. package/src/commands/release-notes/release-notes.ts +50 -0
  222. package/src/commands/reload-plugins/index.ts +18 -0
  223. package/src/commands/reload-plugins/reload-plugins.ts +61 -0
  224. package/src/commands/remote-env/index.ts +15 -0
  225. package/src/commands/remote-env/remote-env.tsx +7 -0
  226. package/src/commands/remote-setup/api.ts +182 -0
  227. package/src/commands/remote-setup/index.ts +20 -0
  228. package/src/commands/remote-setup/remote-setup.tsx +187 -0
  229. package/src/commands/rename/generateSessionName.ts +67 -0
  230. package/src/commands/rename/index.ts +12 -0
  231. package/src/commands/rename/rename.ts +87 -0
  232. package/src/commands/reset-limits/index.js +4 -0
  233. package/src/commands/resume/index.ts +12 -0
  234. package/src/commands/resume/resume.tsx +275 -0
  235. package/src/commands/review/UltrareviewOverageDialog.tsx +96 -0
  236. package/src/commands/review/reviewRemote.ts +316 -0
  237. package/src/commands/review/ultrareviewCommand.tsx +58 -0
  238. package/src/commands/review/ultrareviewEnabled.ts +14 -0
  239. package/src/commands/review.ts +57 -0
  240. package/src/commands/rewind/index.ts +13 -0
  241. package/src/commands/rewind/rewind.ts +13 -0
  242. package/src/commands/sandbox-toggle/index.ts +50 -0
  243. package/src/commands/sandbox-toggle/sandbox-toggle.tsx +83 -0
  244. package/src/commands/security-review.ts +243 -0
  245. package/src/commands/session/index.ts +16 -0
  246. package/src/commands/session/session.tsx +140 -0
  247. package/src/commands/share/index.js +1 -0
  248. package/src/commands/skills/index.ts +10 -0
  249. package/src/commands/skills/skills.tsx +8 -0
  250. package/src/commands/stats/index.ts +10 -0
  251. package/src/commands/stats/stats.tsx +7 -0
  252. package/src/commands/status/index.ts +12 -0
  253. package/src/commands/status/status.tsx +8 -0
  254. package/src/commands/statusline.tsx +24 -0
  255. package/src/commands/stickers/index.ts +11 -0
  256. package/src/commands/stickers/stickers.ts +16 -0
  257. package/src/commands/summary/index.js +1 -0
  258. package/src/commands/tag/index.ts +12 -0
  259. package/src/commands/tag/tag.tsx +215 -0
  260. package/src/commands/tasks/index.ts +11 -0
  261. package/src/commands/tasks/tasks.tsx +8 -0
  262. package/src/commands/teleport/index.js +1 -0
  263. package/src/commands/terminalSetup/index.ts +23 -0
  264. package/src/commands/terminalSetup/terminalSetup.tsx +531 -0
  265. package/src/commands/theme/index.ts +10 -0
  266. package/src/commands/theme/theme.tsx +57 -0
  267. package/src/commands/thinkback/index.ts +13 -0
  268. package/src/commands/thinkback/thinkback.tsx +554 -0
  269. package/src/commands/thinkback-play/index.ts +17 -0
  270. package/src/commands/thinkback-play/thinkback-play.ts +43 -0
  271. package/src/commands/ultraplan.tsx +471 -0
  272. package/src/commands/upgrade/index.ts +16 -0
  273. package/src/commands/upgrade/upgrade.tsx +38 -0
  274. package/src/commands/usage/index.ts +9 -0
  275. package/src/commands/usage/usage.tsx +7 -0
  276. package/src/commands/version.ts +22 -0
  277. package/src/commands/vim/index.ts +11 -0
  278. package/src/commands/vim/vim.ts +38 -0
  279. package/src/commands/voice/index.ts +20 -0
  280. package/src/commands/voice/voice.ts +150 -0
  281. package/src/commands.ts +754 -0
  282. package/src/components/AgentProgressLine.tsx +136 -0
  283. package/src/components/App.tsx +56 -0
  284. package/src/components/ApproveApiKey.tsx +123 -0
  285. package/src/components/AutoModeOptInDialog.tsx +142 -0
  286. package/src/components/AutoUpdater.tsx +198 -0
  287. package/src/components/AutoUpdaterWrapper.tsx +91 -0
  288. package/src/components/AwsAuthStatusBox.tsx +82 -0
  289. package/src/components/BaseTextInput.tsx +136 -0
  290. package/src/components/BashModeProgress.tsx +56 -0
  291. package/src/components/BridgeDialog.tsx +401 -0
  292. package/src/components/BypassPermissionsModeDialog.tsx +87 -0
  293. package/src/components/ChannelDowngradeDialog.tsx +102 -0
  294. package/src/components/ClaudeCodeHint/PluginHintMenu.tsx +78 -0
  295. package/src/components/ClaudeInChromeOnboarding.tsx +121 -0
  296. package/src/components/ClaudeMdExternalIncludesDialog.tsx +137 -0
  297. package/src/components/ClickableImageRef.tsx +73 -0
  298. package/src/components/CompactSummary.tsx +118 -0
  299. package/src/components/ConfigurableShortcutHint.tsx +57 -0
  300. package/src/components/ConsoleOAuthFlow.tsx +631 -0
  301. package/src/components/ContextSuggestions.tsx +47 -0
  302. package/src/components/ContextVisualization.tsx +489 -0
  303. package/src/components/CoordinatorAgentStatus.tsx +273 -0
  304. package/src/components/CostThresholdDialog.tsx +50 -0
  305. package/src/components/CtrlOToExpand.tsx +51 -0
  306. package/src/components/CustomSelect/SelectMulti.tsx +213 -0
  307. package/src/components/CustomSelect/index.ts +3 -0
  308. package/src/components/CustomSelect/option-map.ts +50 -0
  309. package/src/components/CustomSelect/select-input-option.tsx +488 -0
  310. package/src/components/CustomSelect/select-option.tsx +68 -0
  311. package/src/components/CustomSelect/select.tsx +690 -0
  312. package/src/components/CustomSelect/use-multi-select-state.ts +414 -0
  313. package/src/components/CustomSelect/use-select-input.ts +287 -0
  314. package/src/components/CustomSelect/use-select-navigation.ts +653 -0
  315. package/src/components/CustomSelect/use-select-state.ts +157 -0
  316. package/src/components/DesktopHandoff.tsx +193 -0
  317. package/src/components/DesktopUpsell/DesktopUpsellStartup.tsx +171 -0
  318. package/src/components/DevBar.tsx +49 -0
  319. package/src/components/DevChannelsDialog.tsx +105 -0
  320. package/src/components/DiagnosticsDisplay.tsx +95 -0
  321. package/src/components/EffortCallout.tsx +265 -0
  322. package/src/components/EffortIndicator.ts +42 -0
  323. package/src/components/ExitFlow.tsx +48 -0
  324. package/src/components/ExportDialog.tsx +128 -0
  325. package/src/components/FallbackToolUseErrorMessage.tsx +116 -0
  326. package/src/components/FallbackToolUseRejectedMessage.tsx +16 -0
  327. package/src/components/FastIcon.tsx +46 -0
  328. package/src/components/Feedback.tsx +592 -0
  329. package/src/components/FeedbackSurvey/FeedbackSurvey.tsx +174 -0
  330. package/src/components/FeedbackSurvey/FeedbackSurveyView.tsx +108 -0
  331. package/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx +88 -0
  332. package/src/components/FeedbackSurvey/submitTranscriptShare.ts +112 -0
  333. package/src/components/FeedbackSurvey/useDebouncedDigitInput.ts +82 -0
  334. package/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +296 -0
  335. package/src/components/FeedbackSurvey/useMemorySurvey.tsx +213 -0
  336. package/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +206 -0
  337. package/src/components/FeedbackSurvey/useSurveyState.tsx +100 -0
  338. package/src/components/FileEditToolDiff.tsx +181 -0
  339. package/src/components/FileEditToolUpdatedMessage.tsx +124 -0
  340. package/src/components/FileEditToolUseRejectedMessage.tsx +170 -0
  341. package/src/components/FilePathLink.tsx +43 -0
  342. package/src/components/FullscreenLayout.tsx +637 -0
  343. package/src/components/GlobalSearchDialog.tsx +343 -0
  344. package/src/components/HelpV2/Commands.tsx +82 -0
  345. package/src/components/HelpV2/General.tsx +23 -0
  346. package/src/components/HelpV2/HelpV2.tsx +184 -0
  347. package/src/components/HighlightedCode/Fallback.tsx +193 -0
  348. package/src/components/HighlightedCode.tsx +190 -0
  349. package/src/components/HistorySearchDialog.tsx +118 -0
  350. package/src/components/IdeAutoConnectDialog.tsx +154 -0
  351. package/src/components/IdeOnboardingDialog.tsx +167 -0
  352. package/src/components/IdeStatusIndicator.tsx +58 -0
  353. package/src/components/IdleReturnDialog.tsx +118 -0
  354. package/src/components/InterruptedByUser.tsx +15 -0
  355. package/src/components/InvalidConfigDialog.tsx +156 -0
  356. package/src/components/InvalidSettingsDialog.tsx +89 -0
  357. package/src/components/KeybindingWarnings.tsx +55 -0
  358. package/src/components/LanguagePicker.tsx +86 -0
  359. package/src/components/LogSelector.tsx +1575 -0
  360. package/src/components/LogoV2/AnimatedAsterisk.tsx +50 -0
  361. package/src/components/LogoV2/AnimatedClawd.tsx +124 -0
  362. package/src/components/LogoV2/ChannelsNotice.tsx +266 -0
  363. package/src/components/LogoV2/Clawd.tsx +240 -0
  364. package/src/components/LogoV2/CondensedLogo.tsx +161 -0
  365. package/src/components/LogoV2/EmergencyTip.tsx +58 -0
  366. package/src/components/LogoV2/Feed.tsx +112 -0
  367. package/src/components/LogoV2/FeedColumn.tsx +59 -0
  368. package/src/components/LogoV2/GuestPassesUpsell.tsx +70 -0
  369. package/src/components/LogoV2/LogoV2.tsx +543 -0
  370. package/src/components/LogoV2/Opus1mMergeNotice.tsx +55 -0
  371. package/src/components/LogoV2/OverageCreditUpsell.tsx +166 -0
  372. package/src/components/LogoV2/VoiceModeNotice.tsx +68 -0
  373. package/src/components/LogoV2/WelcomeV2.tsx +433 -0
  374. package/src/components/LogoV2/feedConfigs.tsx +92 -0
  375. package/src/components/LspRecommendation/LspRecommendationMenu.tsx +88 -0
  376. package/src/components/MCPServerApprovalDialog.tsx +115 -0
  377. package/src/components/MCPServerDesktopImportDialog.tsx +203 -0
  378. package/src/components/MCPServerDialogCopy.tsx +15 -0
  379. package/src/components/MCPServerMultiselectDialog.tsx +133 -0
  380. package/src/components/ManagedSettingsSecurityDialog/ManagedSettingsSecurityDialog.tsx +149 -0
  381. package/src/components/ManagedSettingsSecurityDialog/utils.ts +144 -0
  382. package/src/components/Markdown.tsx +236 -0
  383. package/src/components/MarkdownTable.tsx +322 -0
  384. package/src/components/MemoryUsageIndicator.tsx +37 -0
  385. package/src/components/Message.tsx +627 -0
  386. package/src/components/MessageModel.tsx +43 -0
  387. package/src/components/MessageResponse.tsx +78 -0
  388. package/src/components/MessageRow.tsx +383 -0
  389. package/src/components/MessageSelector.tsx +831 -0
  390. package/src/components/MessageTimestamp.tsx +63 -0
  391. package/src/components/Messages.tsx +834 -0
  392. package/src/components/ModelPicker.tsx +448 -0
  393. package/src/components/NativeAutoUpdater.tsx +193 -0
  394. package/src/components/NotebookEditToolUseRejectedMessage.tsx +92 -0
  395. package/src/components/OffscreenFreeze.tsx +44 -0
  396. package/src/components/Onboarding.tsx +244 -0
  397. package/src/components/OutputStylePicker.tsx +112 -0
  398. package/src/components/PackageManagerAutoUpdater.tsx +104 -0
  399. package/src/components/Passes/Passes.tsx +184 -0
  400. package/src/components/PrBadge.tsx +97 -0
  401. package/src/components/PressEnterToContinue.tsx +15 -0
  402. package/src/components/PromptInput/HistorySearchInput.tsx +51 -0
  403. package/src/components/PromptInput/IssueFlagBanner.tsx +12 -0
  404. package/src/components/PromptInput/Notifications.tsx +332 -0
  405. package/src/components/PromptInput/PromptInput.tsx +2339 -0
  406. package/src/components/PromptInput/PromptInputFooter.tsx +191 -0
  407. package/src/components/PromptInput/PromptInputFooterLeftSide.tsx +517 -0
  408. package/src/components/PromptInput/PromptInputFooterSuggestions.tsx +293 -0
  409. package/src/components/PromptInput/PromptInputHelpMenu.tsx +358 -0
  410. package/src/components/PromptInput/PromptInputModeIndicator.tsx +93 -0
  411. package/src/components/PromptInput/PromptInputQueuedCommands.tsx +117 -0
  412. package/src/components/PromptInput/PromptInputStashNotice.tsx +25 -0
  413. package/src/components/PromptInput/SandboxPromptFooterHint.tsx +64 -0
  414. package/src/components/PromptInput/ShimmeredInput.tsx +143 -0
  415. package/src/components/PromptInput/VoiceIndicator.tsx +137 -0
  416. package/src/components/PromptInput/inputModes.ts +33 -0
  417. package/src/components/PromptInput/inputPaste.ts +90 -0
  418. package/src/components/PromptInput/useMaybeTruncateInput.ts +58 -0
  419. package/src/components/PromptInput/usePromptInputPlaceholder.ts +76 -0
  420. package/src/components/PromptInput/useShowFastIconHint.ts +31 -0
  421. package/src/components/PromptInput/useSwarmBanner.ts +155 -0
  422. package/src/components/PromptInput/utils.ts +60 -0
  423. package/src/components/QuickOpenDialog.tsx +244 -0
  424. package/src/components/RemoteCallout.tsx +76 -0
  425. package/src/components/RemoteEnvironmentDialog.tsx +340 -0
  426. package/src/components/ResumeTask.tsx +268 -0
  427. package/src/components/SandboxViolationExpandedView.tsx +99 -0
  428. package/src/components/ScrollKeybindingHandler.tsx +1012 -0
  429. package/src/components/SearchBox.tsx +72 -0
  430. package/src/components/SentryErrorBoundary.ts +28 -0
  431. package/src/components/SessionBackgroundHint.tsx +108 -0
  432. package/src/components/SessionPreview.tsx +194 -0
  433. package/src/components/Settings/Config.tsx +1822 -0
  434. package/src/components/Settings/Settings.tsx +137 -0
  435. package/src/components/Settings/Status.tsx +241 -0
  436. package/src/components/Settings/Usage.tsx +377 -0
  437. package/src/components/ShowInIDEPrompt.tsx +170 -0
  438. package/src/components/SkillImprovementSurvey.tsx +152 -0
  439. package/src/components/Spinner/FlashingChar.tsx +61 -0
  440. package/src/components/Spinner/GlimmerMessage.tsx +328 -0
  441. package/src/components/Spinner/ShimmerChar.tsx +36 -0
  442. package/src/components/Spinner/SpinnerAnimationRow.tsx +265 -0
  443. package/src/components/Spinner/SpinnerGlyph.tsx +80 -0
  444. package/src/components/Spinner/TeammateSpinnerLine.tsx +233 -0
  445. package/src/components/Spinner/TeammateSpinnerTree.tsx +272 -0
  446. package/src/components/Spinner/index.ts +10 -0
  447. package/src/components/Spinner/teammateSelectHint.ts +1 -0
  448. package/src/components/Spinner/useShimmerAnimation.ts +31 -0
  449. package/src/components/Spinner/useStalledAnimation.ts +75 -0
  450. package/src/components/Spinner/utils.ts +84 -0
  451. package/src/components/Spinner.tsx +562 -0
  452. package/src/components/Stats.tsx +1228 -0
  453. package/src/components/StatusLine.tsx +324 -0
  454. package/src/components/StatusNotices.tsx +55 -0
  455. package/src/components/StructuredDiff/Fallback.tsx +487 -0
  456. package/src/components/StructuredDiff/colorDiff.ts +37 -0
  457. package/src/components/StructuredDiff.tsx +190 -0
  458. package/src/components/StructuredDiffList.tsx +30 -0
  459. package/src/components/TagTabs.tsx +139 -0
  460. package/src/components/TaskListV2.tsx +378 -0
  461. package/src/components/TeammateViewHeader.tsx +82 -0
  462. package/src/components/TeleportError.tsx +189 -0
  463. package/src/components/TeleportProgress.tsx +140 -0
  464. package/src/components/TeleportRepoMismatchDialog.tsx +104 -0
  465. package/src/components/TeleportResumeWrapper.tsx +167 -0
  466. package/src/components/TeleportStash.tsx +116 -0
  467. package/src/components/TextInput.tsx +124 -0
  468. package/src/components/ThemePicker.tsx +333 -0
  469. package/src/components/ThinkingToggle.tsx +153 -0
  470. package/src/components/TokenWarning.tsx +179 -0
  471. package/src/components/ToolUseLoader.tsx +42 -0
  472. package/src/components/TrustDialog/TrustDialog.tsx +290 -0
  473. package/src/components/TrustDialog/utils.ts +245 -0
  474. package/src/components/ValidationErrorsList.tsx +148 -0
  475. package/src/components/VimTextInput.tsx +140 -0
  476. package/src/components/VirtualMessageList.tsx +1082 -0
  477. package/src/components/WorkflowMultiselectDialog.tsx +128 -0
  478. package/src/components/WorktreeExitDialog.tsx +231 -0
  479. package/src/components/agents/AgentDetail.tsx +220 -0
  480. package/src/components/agents/AgentEditor.tsx +178 -0
  481. package/src/components/agents/AgentNavigationFooter.tsx +26 -0
  482. package/src/components/agents/AgentsList.tsx +440 -0
  483. package/src/components/agents/AgentsMenu.tsx +800 -0
  484. package/src/components/agents/ColorPicker.tsx +112 -0
  485. package/src/components/agents/ModelSelector.tsx +68 -0
  486. package/src/components/agents/ToolSelector.tsx +562 -0
  487. package/src/components/agents/agentFileUtils.ts +272 -0
  488. package/src/components/agents/generateAgent.ts +197 -0
  489. package/src/components/agents/new-agent-creation/CreateAgentWizard.tsx +97 -0
  490. package/src/components/agents/new-agent-creation/wizard-steps/ColorStep.tsx +84 -0
  491. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStep.tsx +378 -0
  492. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStepWrapper.tsx +74 -0
  493. package/src/components/agents/new-agent-creation/wizard-steps/DescriptionStep.tsx +123 -0
  494. package/src/components/agents/new-agent-creation/wizard-steps/GenerateStep.tsx +143 -0
  495. package/src/components/agents/new-agent-creation/wizard-steps/LocationStep.tsx +80 -0
  496. package/src/components/agents/new-agent-creation/wizard-steps/MemoryStep.tsx +113 -0
  497. package/src/components/agents/new-agent-creation/wizard-steps/MethodStep.tsx +80 -0
  498. package/src/components/agents/new-agent-creation/wizard-steps/ModelStep.tsx +52 -0
  499. package/src/components/agents/new-agent-creation/wizard-steps/PromptStep.tsx +128 -0
  500. package/src/components/agents/new-agent-creation/wizard-steps/ToolsStep.tsx +61 -0
  501. package/src/components/agents/new-agent-creation/wizard-steps/TypeStep.tsx +103 -0
  502. package/src/components/agents/types.ts +27 -0
  503. package/src/components/agents/utils.ts +18 -0
  504. package/src/components/agents/validateAgent.ts +109 -0
  505. package/src/components/design-system/Byline.tsx +77 -0
  506. package/src/components/design-system/Dialog.tsx +138 -0
  507. package/src/components/design-system/Divider.tsx +149 -0
  508. package/src/components/design-system/FuzzyPicker.tsx +312 -0
  509. package/src/components/design-system/KeyboardShortcutHint.tsx +81 -0
  510. package/src/components/design-system/ListItem.tsx +244 -0
  511. package/src/components/design-system/LoadingState.tsx +94 -0
  512. package/src/components/design-system/Pane.tsx +77 -0
  513. package/src/components/design-system/ProgressBar.tsx +86 -0
  514. package/src/components/design-system/Ratchet.tsx +80 -0
  515. package/src/components/design-system/StatusIcon.tsx +95 -0
  516. package/src/components/design-system/Tabs.tsx +340 -0
  517. package/src/components/design-system/ThemeProvider.tsx +174 -0
  518. package/src/components/design-system/ThemedBox.tsx +156 -0
  519. package/src/components/design-system/ThemedText.tsx +124 -0
  520. package/src/components/design-system/color.ts +30 -0
  521. package/src/components/diff/DiffDetailView.tsx +281 -0
  522. package/src/components/diff/DiffDialog.tsx +383 -0
  523. package/src/components/diff/DiffFileList.tsx +292 -0
  524. package/src/components/grove/Grove.tsx +463 -0
  525. package/src/components/hooks/HooksConfigMenu.tsx +578 -0
  526. package/src/components/hooks/PromptDialog.tsx +90 -0
  527. package/src/components/hooks/SelectEventMode.tsx +127 -0
  528. package/src/components/hooks/SelectHookMode.tsx +112 -0
  529. package/src/components/hooks/SelectMatcherMode.tsx +144 -0
  530. package/src/components/hooks/ViewHookMode.tsx +199 -0
  531. package/src/components/mcp/CapabilitiesSection.tsx +61 -0
  532. package/src/components/mcp/ElicitationDialog.tsx +1169 -0
  533. package/src/components/mcp/MCPAgentServerMenu.tsx +183 -0
  534. package/src/components/mcp/MCPListPanel.tsx +504 -0
  535. package/src/components/mcp/MCPReconnect.tsx +167 -0
  536. package/src/components/mcp/MCPRemoteServerMenu.tsx +649 -0
  537. package/src/components/mcp/MCPSettings.tsx +398 -0
  538. package/src/components/mcp/MCPStdioServerMenu.tsx +177 -0
  539. package/src/components/mcp/MCPToolDetailView.tsx +212 -0
  540. package/src/components/mcp/MCPToolListView.tsx +141 -0
  541. package/src/components/mcp/McpParsingWarnings.tsx +213 -0
  542. package/src/components/mcp/index.ts +9 -0
  543. package/src/components/mcp/utils/reconnectHelpers.tsx +49 -0
  544. package/src/components/memory/MemoryFileSelector.tsx +438 -0
  545. package/src/components/memory/MemoryUpdateNotification.tsx +45 -0
  546. package/src/components/messageActions.tsx +450 -0
  547. package/src/components/messages/AdvisorMessage.tsx +158 -0
  548. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +31 -0
  549. package/src/components/messages/AssistantTextMessage.tsx +270 -0
  550. package/src/components/messages/AssistantThinkingMessage.tsx +86 -0
  551. package/src/components/messages/AssistantToolUseMessage.tsx +368 -0
  552. package/src/components/messages/AttachmentMessage.tsx +536 -0
  553. package/src/components/messages/CollapsedReadSearchContent.tsx +484 -0
  554. package/src/components/messages/CompactBoundaryMessage.tsx +18 -0
  555. package/src/components/messages/GroupedToolUseContent.tsx +58 -0
  556. package/src/components/messages/HighlightedThinkingText.tsx +162 -0
  557. package/src/components/messages/HookProgressMessage.tsx +116 -0
  558. package/src/components/messages/PlanApprovalMessage.tsx +222 -0
  559. package/src/components/messages/RateLimitMessage.tsx +161 -0
  560. package/src/components/messages/ShutdownMessage.tsx +132 -0
  561. package/src/components/messages/SystemAPIErrorMessage.tsx +141 -0
  562. package/src/components/messages/SystemTextMessage.tsx +827 -0
  563. package/src/components/messages/TaskAssignmentMessage.tsx +76 -0
  564. package/src/components/messages/UserAgentNotificationMessage.tsx +83 -0
  565. package/src/components/messages/UserBashInputMessage.tsx +58 -0
  566. package/src/components/messages/UserBashOutputMessage.tsx +54 -0
  567. package/src/components/messages/UserChannelMessage.tsx +137 -0
  568. package/src/components/messages/UserCommandMessage.tsx +108 -0
  569. package/src/components/messages/UserImageMessage.tsx +59 -0
  570. package/src/components/messages/UserLocalCommandOutputMessage.tsx +167 -0
  571. package/src/components/messages/UserMemoryInputMessage.tsx +75 -0
  572. package/src/components/messages/UserPlanMessage.tsx +42 -0
  573. package/src/components/messages/UserPromptMessage.tsx +80 -0
  574. package/src/components/messages/UserResourceUpdateMessage.tsx +121 -0
  575. package/src/components/messages/UserTeammateMessage.tsx +206 -0
  576. package/src/components/messages/UserTextMessage.tsx +275 -0
  577. package/src/components/messages/UserToolResultMessage/RejectedPlanMessage.tsx +31 -0
  578. package/src/components/messages/UserToolResultMessage/RejectedToolUseMessage.tsx +16 -0
  579. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +16 -0
  580. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +103 -0
  581. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +95 -0
  582. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +106 -0
  583. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +104 -0
  584. package/src/components/messages/UserToolResultMessage/utils.tsx +44 -0
  585. package/src/components/messages/nullRenderingAttachments.ts +70 -0
  586. package/src/components/messages/teamMemCollapsed.tsx +140 -0
  587. package/src/components/messages/teamMemSaved.ts +19 -0
  588. package/src/components/permissions/AskUserQuestionPermissionRequest/AskUserQuestionPermissionRequest.tsx +645 -0
  589. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewBox.tsx +229 -0
  590. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewQuestionView.tsx +328 -0
  591. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionNavigationBar.tsx +178 -0
  592. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionView.tsx +465 -0
  593. package/src/components/permissions/AskUserQuestionPermissionRequest/SubmitQuestionsView.tsx +144 -0
  594. package/src/components/permissions/AskUserQuestionPermissionRequest/use-multiple-choice-state.ts +179 -0
  595. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +482 -0
  596. package/src/components/permissions/BashPermissionRequest/bashToolUseOptions.tsx +147 -0
  597. package/src/components/permissions/ComputerUseApproval/ComputerUseApproval.tsx +441 -0
  598. package/src/components/permissions/EnterPlanModePermissionRequest/EnterPlanModePermissionRequest.tsx +122 -0
  599. package/src/components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.tsx +768 -0
  600. package/src/components/permissions/FallbackPermissionRequest.tsx +333 -0
  601. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  602. package/src/components/permissions/FilePermissionDialog/FilePermissionDialog.tsx +204 -0
  603. package/src/components/permissions/FilePermissionDialog/ideDiffConfig.ts +42 -0
  604. package/src/components/permissions/FilePermissionDialog/permissionOptions.tsx +177 -0
  605. package/src/components/permissions/FilePermissionDialog/useFilePermissionDialog.ts +212 -0
  606. package/src/components/permissions/FilePermissionDialog/usePermissionHandler.ts +185 -0
  607. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +161 -0
  608. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +89 -0
  609. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +115 -0
  610. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditPermissionRequest.tsx +166 -0
  611. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditToolDiff.tsx +235 -0
  612. package/src/components/permissions/PermissionDecisionDebugInfo.tsx +460 -0
  613. package/src/components/permissions/PermissionDialog.tsx +72 -0
  614. package/src/components/permissions/PermissionExplanation.tsx +272 -0
  615. package/src/components/permissions/PermissionPrompt.tsx +336 -0
  616. package/src/components/permissions/PermissionRequest.tsx +217 -0
  617. package/src/components/permissions/PermissionRequestTitle.tsx +66 -0
  618. package/src/components/permissions/PermissionRuleExplanation.tsx +121 -0
  619. package/src/components/permissions/PowerShellPermissionRequest/PowerShellPermissionRequest.tsx +235 -0
  620. package/src/components/permissions/PowerShellPermissionRequest/powershellToolUseOptions.tsx +91 -0
  621. package/src/components/permissions/SandboxPermissionRequest.tsx +163 -0
  622. package/src/components/permissions/SedEditPermissionRequest/SedEditPermissionRequest.tsx +230 -0
  623. package/src/components/permissions/SkillPermissionRequest/SkillPermissionRequest.tsx +369 -0
  624. package/src/components/permissions/WebFetchPermissionRequest/WebFetchPermissionRequest.tsx +258 -0
  625. package/src/components/permissions/WorkerBadge.tsx +49 -0
  626. package/src/components/permissions/WorkerPendingPermission.tsx +105 -0
  627. package/src/components/permissions/hooks.ts +209 -0
  628. package/src/components/permissions/rules/AddPermissionRules.tsx +180 -0
  629. package/src/components/permissions/rules/AddWorkspaceDirectory.tsx +340 -0
  630. package/src/components/permissions/rules/PermissionRuleDescription.tsx +76 -0
  631. package/src/components/permissions/rules/PermissionRuleInput.tsx +138 -0
  632. package/src/components/permissions/rules/PermissionRuleList.tsx +1179 -0
  633. package/src/components/permissions/rules/RecentDenialsTab.tsx +207 -0
  634. package/src/components/permissions/rules/RemoveWorkspaceDirectory.tsx +110 -0
  635. package/src/components/permissions/rules/WorkspaceTab.tsx +150 -0
  636. package/src/components/permissions/shellPermissionHelpers.tsx +164 -0
  637. package/src/components/permissions/useShellPermissionFeedback.ts +148 -0
  638. package/src/components/permissions/utils.ts +25 -0
  639. package/src/components/sandbox/SandboxConfigTab.tsx +45 -0
  640. package/src/components/sandbox/SandboxDependenciesTab.tsx +120 -0
  641. package/src/components/sandbox/SandboxDoctorSection.tsx +46 -0
  642. package/src/components/sandbox/SandboxOverridesTab.tsx +193 -0
  643. package/src/components/sandbox/SandboxSettings.tsx +296 -0
  644. package/src/components/shell/ExpandShellOutputContext.tsx +36 -0
  645. package/src/components/shell/OutputLine.tsx +118 -0
  646. package/src/components/shell/ShellProgressMessage.tsx +150 -0
  647. package/src/components/shell/ShellTimeDisplay.tsx +74 -0
  648. package/src/components/skills/SkillsMenu.tsx +237 -0
  649. package/src/components/tasks/AsyncAgentDetailDialog.tsx +229 -0
  650. package/src/components/tasks/BackgroundTask.tsx +345 -0
  651. package/src/components/tasks/BackgroundTaskStatus.tsx +429 -0
  652. package/src/components/tasks/BackgroundTasksDialog.tsx +652 -0
  653. package/src/components/tasks/DreamDetailDialog.tsx +251 -0
  654. package/src/components/tasks/InProcessTeammateDetailDialog.tsx +266 -0
  655. package/src/components/tasks/RemoteSessionDetailDialog.tsx +904 -0
  656. package/src/components/tasks/RemoteSessionProgress.tsx +243 -0
  657. package/src/components/tasks/ShellDetailDialog.tsx +404 -0
  658. package/src/components/tasks/ShellProgress.tsx +87 -0
  659. package/src/components/tasks/renderToolActivity.tsx +33 -0
  660. package/src/components/tasks/taskStatusUtils.tsx +107 -0
  661. package/src/components/teams/TeamStatus.tsx +80 -0
  662. package/src/components/teams/TeamsDialog.tsx +715 -0
  663. package/src/components/ui/OrderedList.tsx +71 -0
  664. package/src/components/ui/OrderedListItem.tsx +45 -0
  665. package/src/components/ui/TreeSelect.tsx +397 -0
  666. package/src/components/wizard/WizardDialogLayout.tsx +65 -0
  667. package/src/components/wizard/WizardNavigationFooter.tsx +24 -0
  668. package/src/components/wizard/WizardProvider.tsx +213 -0
  669. package/src/components/wizard/index.ts +9 -0
  670. package/src/components/wizard/useWizard.ts +13 -0
  671. package/src/constants/apiLimits.ts +94 -0
  672. package/src/constants/betas.ts +52 -0
  673. package/src/constants/common.ts +33 -0
  674. package/src/constants/cyberRiskInstruction.ts +24 -0
  675. package/src/constants/errorIds.ts +15 -0
  676. package/src/constants/figures.ts +45 -0
  677. package/src/constants/files.ts +156 -0
  678. package/src/constants/github-app.ts +144 -0
  679. package/src/constants/keys.ts +11 -0
  680. package/src/constants/messages.ts +1 -0
  681. package/src/constants/oauth.ts +234 -0
  682. package/src/constants/outputStyles.ts +216 -0
  683. package/src/constants/product.ts +76 -0
  684. package/src/constants/prompts.ts +914 -0
  685. package/src/constants/spinnerVerbs.ts +204 -0
  686. package/src/constants/system.ts +95 -0
  687. package/src/constants/systemPromptSections.ts +68 -0
  688. package/src/constants/toolLimits.ts +56 -0
  689. package/src/constants/tools.ts +112 -0
  690. package/src/constants/turnCompletionVerbs.ts +12 -0
  691. package/src/constants/xml.ts +86 -0
  692. package/src/context/QueuedMessageContext.tsx +63 -0
  693. package/src/context/fpsMetrics.tsx +30 -0
  694. package/src/context/mailbox.tsx +38 -0
  695. package/src/context/modalContext.tsx +58 -0
  696. package/src/context/notifications.tsx +240 -0
  697. package/src/context/overlayContext.tsx +151 -0
  698. package/src/context/promptOverlayContext.tsx +125 -0
  699. package/src/context/stats.tsx +220 -0
  700. package/src/context/voice.tsx +88 -0
  701. package/src/context.ts +189 -0
  702. package/src/coordinator/coordinatorMode.ts +369 -0
  703. package/src/cost-tracker.ts +323 -0
  704. package/src/costHook.ts +22 -0
  705. package/src/dialogLaunchers.tsx +133 -0
  706. package/src/entrypoints/agentSdkTypes.ts +443 -0
  707. package/src/entrypoints/cli.tsx +303 -0
  708. package/src/entrypoints/init.ts +340 -0
  709. package/src/entrypoints/mcp.ts +196 -0
  710. package/src/entrypoints/sandboxTypes.ts +156 -0
  711. package/src/entrypoints/sdk/controlSchemas.ts +663 -0
  712. package/src/entrypoints/sdk/coreSchemas.ts +1889 -0
  713. package/src/entrypoints/sdk/coreTypes.ts +62 -0
  714. package/src/history.ts +464 -0
  715. package/src/hooks/fileSuggestions.ts +811 -0
  716. package/src/hooks/notifs/useAutoModeUnavailableNotification.ts +56 -0
  717. package/src/hooks/notifs/useCanSwitchToExistingSubscription.tsx +60 -0
  718. package/src/hooks/notifs/useDeprecationWarningNotification.tsx +44 -0
  719. package/src/hooks/notifs/useFastModeNotification.tsx +162 -0
  720. package/src/hooks/notifs/useIDEStatusIndicator.tsx +186 -0
  721. package/src/hooks/notifs/useInstallMessages.tsx +26 -0
  722. package/src/hooks/notifs/useLspInitializationNotification.tsx +143 -0
  723. package/src/hooks/notifs/useMcpConnectivityStatus.tsx +88 -0
  724. package/src/hooks/notifs/useModelMigrationNotifications.tsx +52 -0
  725. package/src/hooks/notifs/useNpmDeprecationNotification.tsx +25 -0
  726. package/src/hooks/notifs/usePluginAutoupdateNotification.tsx +83 -0
  727. package/src/hooks/notifs/usePluginInstallationStatus.tsx +128 -0
  728. package/src/hooks/notifs/useRateLimitWarningNotification.tsx +114 -0
  729. package/src/hooks/notifs/useSettingsErrors.tsx +69 -0
  730. package/src/hooks/notifs/useStartupNotification.ts +41 -0
  731. package/src/hooks/notifs/useTeammateShutdownNotification.ts +78 -0
  732. package/src/hooks/renderPlaceholder.ts +51 -0
  733. package/src/hooks/toolPermission/PermissionContext.ts +388 -0
  734. package/src/hooks/toolPermission/handlers/coordinatorHandler.ts +65 -0
  735. package/src/hooks/toolPermission/handlers/interactiveHandler.ts +536 -0
  736. package/src/hooks/toolPermission/handlers/swarmWorkerHandler.ts +159 -0
  737. package/src/hooks/toolPermission/permissionLogging.ts +238 -0
  738. package/src/hooks/unifiedSuggestions.ts +202 -0
  739. package/src/hooks/useAfterFirstRender.ts +17 -0
  740. package/src/hooks/useApiKeyVerification.ts +84 -0
  741. package/src/hooks/useArrowKeyHistory.tsx +229 -0
  742. package/src/hooks/useAssistantHistory.ts +250 -0
  743. package/src/hooks/useAwaySummary.ts +125 -0
  744. package/src/hooks/useBackgroundTaskNavigation.ts +251 -0
  745. package/src/hooks/useBlink.ts +34 -0
  746. package/src/hooks/useCanUseTool.tsx +204 -0
  747. package/src/hooks/useCancelRequest.ts +276 -0
  748. package/src/hooks/useChromeExtensionNotification.tsx +50 -0
  749. package/src/hooks/useClaudeCodeHintRecommendation.tsx +129 -0
  750. package/src/hooks/useClipboardImageHint.ts +77 -0
  751. package/src/hooks/useCommandKeybindings.tsx +108 -0
  752. package/src/hooks/useCommandQueue.ts +15 -0
  753. package/src/hooks/useCopyOnSelect.ts +98 -0
  754. package/src/hooks/useDeferredHookMessages.ts +46 -0
  755. package/src/hooks/useDiffData.ts +110 -0
  756. package/src/hooks/useDiffInIDE.ts +379 -0
  757. package/src/hooks/useDirectConnect.ts +229 -0
  758. package/src/hooks/useDoublePress.ts +62 -0
  759. package/src/hooks/useDynamicConfig.ts +22 -0
  760. package/src/hooks/useElapsedTime.ts +37 -0
  761. package/src/hooks/useExitOnCtrlCD.ts +95 -0
  762. package/src/hooks/useExitOnCtrlCDWithKeybindings.ts +24 -0
  763. package/src/hooks/useFileHistorySnapshotInit.ts +25 -0
  764. package/src/hooks/useGlobalKeybindings.tsx +249 -0
  765. package/src/hooks/useHistorySearch.ts +303 -0
  766. package/src/hooks/useIDEIntegration.tsx +70 -0
  767. package/src/hooks/useIdeAtMentioned.ts +76 -0
  768. package/src/hooks/useIdeConnectionStatus.ts +33 -0
  769. package/src/hooks/useIdeLogging.ts +41 -0
  770. package/src/hooks/useIdeSelection.ts +150 -0
  771. package/src/hooks/useInboxPoller.ts +969 -0
  772. package/src/hooks/useInputBuffer.ts +132 -0
  773. package/src/hooks/useIssueFlagBanner.ts +133 -0
  774. package/src/hooks/useLogMessages.ts +119 -0
  775. package/src/hooks/useLspPluginRecommendation.tsx +194 -0
  776. package/src/hooks/useMailboxBridge.ts +21 -0
  777. package/src/hooks/useMainLoopModel.ts +34 -0
  778. package/src/hooks/useManagePlugins.ts +304 -0
  779. package/src/hooks/useMemoryUsage.ts +39 -0
  780. package/src/hooks/useMergedClients.ts +23 -0
  781. package/src/hooks/useMergedCommands.ts +15 -0
  782. package/src/hooks/useMergedTools.ts +44 -0
  783. package/src/hooks/useMinDisplayTime.ts +35 -0
  784. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  785. package/src/hooks/useOfficialMarketplaceNotification.tsx +48 -0
  786. package/src/hooks/usePasteHandler.ts +285 -0
  787. package/src/hooks/usePluginRecommendationBase.tsx +105 -0
  788. package/src/hooks/usePrStatus.ts +106 -0
  789. package/src/hooks/usePromptSuggestion.ts +177 -0
  790. package/src/hooks/usePromptsFromClaudeInChrome.tsx +71 -0
  791. package/src/hooks/useQueueProcessor.ts +68 -0
  792. package/src/hooks/useRemoteSession.ts +605 -0
  793. package/src/hooks/useReplBridge.tsx +723 -0
  794. package/src/hooks/useSSHSession.ts +241 -0
  795. package/src/hooks/useScheduledTasks.ts +139 -0
  796. package/src/hooks/useSearchInput.ts +364 -0
  797. package/src/hooks/useSessionBackgrounding.ts +158 -0
  798. package/src/hooks/useSettings.ts +17 -0
  799. package/src/hooks/useSettingsChange.ts +25 -0
  800. package/src/hooks/useSkillImprovementSurvey.ts +105 -0
  801. package/src/hooks/useSkillsChange.ts +62 -0
  802. package/src/hooks/useSwarmInitialization.ts +81 -0
  803. package/src/hooks/useSwarmPermissionPoller.ts +330 -0
  804. package/src/hooks/useTaskListWatcher.ts +221 -0
  805. package/src/hooks/useTasksV2.ts +250 -0
  806. package/src/hooks/useTeammateViewAutoExit.ts +63 -0
  807. package/src/hooks/useTeleportResume.tsx +85 -0
  808. package/src/hooks/useTerminalSize.ts +15 -0
  809. package/src/hooks/useTextInput.ts +529 -0
  810. package/src/hooks/useTimeout.ts +14 -0
  811. package/src/hooks/useTurnDiffs.ts +213 -0
  812. package/src/hooks/useTypeahead.tsx +1385 -0
  813. package/src/hooks/useUpdateNotification.ts +34 -0
  814. package/src/hooks/useVimInput.ts +316 -0
  815. package/src/hooks/useVirtualScroll.ts +721 -0
  816. package/src/hooks/useVoice.ts +1144 -0
  817. package/src/hooks/useVoiceEnabled.ts +25 -0
  818. package/src/hooks/useVoiceIntegration.tsx +677 -0
  819. package/src/ink/Ansi.tsx +292 -0
  820. package/src/ink/bidi.ts +139 -0
  821. package/src/ink/clearTerminal.ts +74 -0
  822. package/src/ink/colorize.ts +231 -0
  823. package/src/ink/components/AlternateScreen.tsx +80 -0
  824. package/src/ink/components/App.tsx +659 -0
  825. package/src/ink/components/AppContext.ts +21 -0
  826. package/src/ink/components/Box.tsx +214 -0
  827. package/src/ink/components/Button.tsx +192 -0
  828. package/src/ink/components/ClockContext.tsx +112 -0
  829. package/src/ink/components/CursorDeclarationContext.ts +32 -0
  830. package/src/ink/components/ErrorOverview.tsx +109 -0
  831. package/src/ink/components/Link.tsx +42 -0
  832. package/src/ink/components/Newline.tsx +39 -0
  833. package/src/ink/components/NoSelect.tsx +68 -0
  834. package/src/ink/components/RawAnsi.tsx +57 -0
  835. package/src/ink/components/ScrollBox.tsx +237 -0
  836. package/src/ink/components/Spacer.tsx +20 -0
  837. package/src/ink/components/StdinContext.ts +49 -0
  838. package/src/ink/components/TerminalFocusContext.tsx +52 -0
  839. package/src/ink/components/TerminalSizeContext.tsx +7 -0
  840. package/src/ink/components/Text.tsx +254 -0
  841. package/src/ink/constants.ts +2 -0
  842. package/src/ink/dom.ts +484 -0
  843. package/src/ink/events/click-event.ts +38 -0
  844. package/src/ink/events/dispatcher.ts +233 -0
  845. package/src/ink/events/emitter.ts +39 -0
  846. package/src/ink/events/event-handlers.ts +73 -0
  847. package/src/ink/events/event.ts +11 -0
  848. package/src/ink/events/focus-event.ts +21 -0
  849. package/src/ink/events/input-event.ts +205 -0
  850. package/src/ink/events/keyboard-event.ts +51 -0
  851. package/src/ink/events/terminal-event.ts +107 -0
  852. package/src/ink/events/terminal-focus-event.ts +19 -0
  853. package/src/ink/focus.ts +181 -0
  854. package/src/ink/frame.ts +124 -0
  855. package/src/ink/get-max-width.ts +27 -0
  856. package/src/ink/hit-test.ts +130 -0
  857. package/src/ink/hooks/use-animation-frame.ts +57 -0
  858. package/src/ink/hooks/use-app.ts +8 -0
  859. package/src/ink/hooks/use-declared-cursor.ts +73 -0
  860. package/src/ink/hooks/use-input.ts +92 -0
  861. package/src/ink/hooks/use-interval.ts +67 -0
  862. package/src/ink/hooks/use-search-highlight.ts +53 -0
  863. package/src/ink/hooks/use-selection.ts +104 -0
  864. package/src/ink/hooks/use-stdin.ts +8 -0
  865. package/src/ink/hooks/use-tab-status.ts +72 -0
  866. package/src/ink/hooks/use-terminal-focus.ts +16 -0
  867. package/src/ink/hooks/use-terminal-title.ts +31 -0
  868. package/src/ink/hooks/use-terminal-viewport.ts +96 -0
  869. package/src/ink/ink.tsx +1728 -0
  870. package/src/ink/instances.ts +10 -0
  871. package/src/ink/layout/engine.ts +6 -0
  872. package/src/ink/layout/geometry.ts +97 -0
  873. package/src/ink/layout/node.ts +152 -0
  874. package/src/ink/layout/yoga.ts +308 -0
  875. package/src/ink/line-width-cache.ts +24 -0
  876. package/src/ink/log-update.ts +773 -0
  877. package/src/ink/measure-element.ts +23 -0
  878. package/src/ink/measure-text.ts +47 -0
  879. package/src/ink/node-cache.ts +54 -0
  880. package/src/ink/optimizer.ts +93 -0
  881. package/src/ink/output.ts +797 -0
  882. package/src/ink/parse-keypress.ts +801 -0
  883. package/src/ink/reconciler.ts +512 -0
  884. package/src/ink/render-border.ts +231 -0
  885. package/src/ink/render-node-to-output.ts +1462 -0
  886. package/src/ink/render-to-screen.ts +231 -0
  887. package/src/ink/renderer.ts +178 -0
  888. package/src/ink/root.ts +184 -0
  889. package/src/ink/screen.ts +1486 -0
  890. package/src/ink/searchHighlight.ts +93 -0
  891. package/src/ink/selection.ts +917 -0
  892. package/src/ink/squash-text-nodes.ts +92 -0
  893. package/src/ink/stringWidth.ts +222 -0
  894. package/src/ink/styles.ts +771 -0
  895. package/src/ink/supports-hyperlinks.ts +57 -0
  896. package/src/ink/tabstops.ts +46 -0
  897. package/src/ink/terminal-focus-state.ts +47 -0
  898. package/src/ink/terminal-querier.ts +212 -0
  899. package/src/ink/terminal.ts +248 -0
  900. package/src/ink/termio/ansi.ts +75 -0
  901. package/src/ink/termio/csi.ts +319 -0
  902. package/src/ink/termio/dec.ts +60 -0
  903. package/src/ink/termio/esc.ts +67 -0
  904. package/src/ink/termio/osc.ts +493 -0
  905. package/src/ink/termio/parser.ts +394 -0
  906. package/src/ink/termio/sgr.ts +308 -0
  907. package/src/ink/termio/tokenize.ts +319 -0
  908. package/src/ink/termio/types.ts +236 -0
  909. package/src/ink/termio.ts +42 -0
  910. package/src/ink/useTerminalNotification.ts +126 -0
  911. package/src/ink/warn.ts +9 -0
  912. package/src/ink/widest-line.ts +19 -0
  913. package/src/ink/wrap-text.ts +74 -0
  914. package/src/ink/wrapAnsi.ts +20 -0
  915. package/src/ink.ts +85 -0
  916. package/src/interactiveHelpers.tsx +367 -0
  917. package/src/keybindings/KeybindingContext.tsx +243 -0
  918. package/src/keybindings/KeybindingProviderSetup.tsx +308 -0
  919. package/src/keybindings/defaultBindings.ts +340 -0
  920. package/src/keybindings/loadUserBindings.ts +472 -0
  921. package/src/keybindings/match.ts +120 -0
  922. package/src/keybindings/parser.ts +203 -0
  923. package/src/keybindings/reservedShortcuts.ts +127 -0
  924. package/src/keybindings/resolver.ts +244 -0
  925. package/src/keybindings/schema.ts +236 -0
  926. package/src/keybindings/shortcutFormat.ts +63 -0
  927. package/src/keybindings/template.ts +52 -0
  928. package/src/keybindings/useKeybinding.ts +196 -0
  929. package/src/keybindings/useShortcutDisplay.ts +59 -0
  930. package/src/keybindings/validate.ts +498 -0
  931. package/src/main.tsx +4684 -0
  932. package/src/memdir/findRelevantMemories.ts +141 -0
  933. package/src/memdir/memdir.ts +507 -0
  934. package/src/memdir/memoryAge.ts +53 -0
  935. package/src/memdir/memoryScan.ts +94 -0
  936. package/src/memdir/memoryTypes.ts +271 -0
  937. package/src/memdir/paths.ts +278 -0
  938. package/src/memdir/teamMemPaths.ts +292 -0
  939. package/src/memdir/teamMemPrompts.ts +100 -0
  940. package/src/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  941. package/src/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  942. package/src/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  943. package/src/migrations/migrateFennecToOpus.ts +45 -0
  944. package/src/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  945. package/src/migrations/migrateOpusToOpus1m.ts +43 -0
  946. package/src/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  947. package/src/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  948. package/src/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  949. package/src/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  950. package/src/migrations/resetProToOpusDefault.ts +51 -0
  951. package/src/moreright/useMoreRight.tsx +26 -0
  952. package/src/native-ts/color-diff/index.ts +999 -0
  953. package/src/native-ts/file-index/index.ts +370 -0
  954. package/src/native-ts/yoga-layout/enums.ts +134 -0
  955. package/src/native-ts/yoga-layout/index.ts +2578 -0
  956. package/src/outputStyles/loadOutputStylesDir.ts +98 -0
  957. package/src/plugins/builtinPlugins.ts +159 -0
  958. package/src/plugins/bundled/index.ts +23 -0
  959. package/src/projectOnboardingState.ts +83 -0
  960. package/src/query/config.ts +46 -0
  961. package/src/query/deps.ts +40 -0
  962. package/src/query/stopHooks.ts +473 -0
  963. package/src/query/tokenBudget.ts +93 -0
  964. package/src/query.ts +1729 -0
  965. package/src/remote/RemoteSessionManager.ts +343 -0
  966. package/src/remote/SessionsWebSocket.ts +404 -0
  967. package/src/remote/remotePermissionBridge.ts +78 -0
  968. package/src/remote/sdkMessageAdapter.ts +302 -0
  969. package/src/replLauncher.tsx +23 -0
  970. package/src/schemas/hooks.ts +222 -0
  971. package/src/screens/Doctor.tsx +575 -0
  972. package/src/screens/REPL.tsx +5006 -0
  973. package/src/screens/ResumeConversation.tsx +399 -0
  974. package/src/server/createDirectConnectSession.ts +88 -0
  975. package/src/server/directConnectManager.ts +213 -0
  976. package/src/server/types.ts +57 -0
  977. package/src/services/AgentSummary/agentSummary.ts +179 -0
  978. package/src/services/MagicDocs/magicDocs.ts +254 -0
  979. package/src/services/MagicDocs/prompts.ts +127 -0
  980. package/src/services/PromptSuggestion/promptSuggestion.ts +523 -0
  981. package/src/services/PromptSuggestion/speculation.ts +991 -0
  982. package/src/services/SessionMemory/prompts.ts +324 -0
  983. package/src/services/SessionMemory/sessionMemory.ts +495 -0
  984. package/src/services/SessionMemory/sessionMemoryUtils.ts +207 -0
  985. package/src/services/analytics/config.ts +38 -0
  986. package/src/services/analytics/datadog.ts +307 -0
  987. package/src/services/analytics/firstPartyEventLogger.ts +449 -0
  988. package/src/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
  989. package/src/services/analytics/growthbook.ts +1157 -0
  990. package/src/services/analytics/index.ts +173 -0
  991. package/src/services/analytics/metadata.ts +973 -0
  992. package/src/services/analytics/sink.ts +114 -0
  993. package/src/services/analytics/sinkKillswitch.ts +25 -0
  994. package/src/services/api/adminRequests.ts +119 -0
  995. package/src/services/api/bootstrap.ts +142 -0
  996. package/src/services/api/claude.ts +3433 -0
  997. package/src/services/api/client.ts +395 -0
  998. package/src/services/api/dumpPrompts.ts +226 -0
  999. package/src/services/api/emptyUsage.ts +22 -0
  1000. package/src/services/api/errorUtils.ts +260 -0
  1001. package/src/services/api/errors.ts +1207 -0
  1002. package/src/services/api/filesApi.ts +748 -0
  1003. package/src/services/api/firstTokenDate.ts +60 -0
  1004. package/src/services/api/grove.ts +357 -0
  1005. package/src/services/api/logging.ts +788 -0
  1006. package/src/services/api/metricsOptOut.ts +159 -0
  1007. package/src/services/api/overageCreditGrant.ts +137 -0
  1008. package/src/services/api/promptCacheBreakDetection.ts +727 -0
  1009. package/src/services/api/referral.ts +281 -0
  1010. package/src/services/api/sessionIngress.ts +514 -0
  1011. package/src/services/api/ultrareviewQuota.ts +38 -0
  1012. package/src/services/api/usage.ts +63 -0
  1013. package/src/services/api/withRetry.ts +826 -0
  1014. package/src/services/autoDream/autoDream.ts +324 -0
  1015. package/src/services/autoDream/config.ts +21 -0
  1016. package/src/services/autoDream/consolidationLock.ts +140 -0
  1017. package/src/services/autoDream/consolidationPrompt.ts +65 -0
  1018. package/src/services/awaySummary.ts +74 -0
  1019. package/src/services/claudeAiLimits.ts +515 -0
  1020. package/src/services/claudeAiLimitsHook.ts +23 -0
  1021. package/src/services/compact/apiMicrocompact.ts +153 -0
  1022. package/src/services/compact/autoCompact.ts +351 -0
  1023. package/src/services/compact/compact.ts +1705 -0
  1024. package/src/services/compact/compactWarningHook.ts +16 -0
  1025. package/src/services/compact/compactWarningState.ts +18 -0
  1026. package/src/services/compact/grouping.ts +63 -0
  1027. package/src/services/compact/microCompact.ts +530 -0
  1028. package/src/services/compact/postCompactCleanup.ts +77 -0
  1029. package/src/services/compact/prompt.ts +374 -0
  1030. package/src/services/compact/sessionMemoryCompact.ts +630 -0
  1031. package/src/services/compact/timeBasedMCConfig.ts +43 -0
  1032. package/src/services/diagnosticTracking.ts +397 -0
  1033. package/src/services/extractMemories/extractMemories.ts +615 -0
  1034. package/src/services/extractMemories/prompts.ts +154 -0
  1035. package/src/services/internalLogging.ts +90 -0
  1036. package/src/services/lsp/LSPClient.ts +447 -0
  1037. package/src/services/lsp/LSPDiagnosticRegistry.ts +386 -0
  1038. package/src/services/lsp/LSPServerInstance.ts +511 -0
  1039. package/src/services/lsp/LSPServerManager.ts +420 -0
  1040. package/src/services/lsp/config.ts +79 -0
  1041. package/src/services/lsp/manager.ts +289 -0
  1042. package/src/services/lsp/passiveFeedback.ts +328 -0
  1043. package/src/services/mcp/InProcessTransport.ts +63 -0
  1044. package/src/services/mcp/MCPConnectionManager.tsx +73 -0
  1045. package/src/services/mcp/SdkControlTransport.ts +136 -0
  1046. package/src/services/mcp/auth.ts +2465 -0
  1047. package/src/services/mcp/channelAllowlist.ts +76 -0
  1048. package/src/services/mcp/channelNotification.ts +316 -0
  1049. package/src/services/mcp/channelPermissions.ts +240 -0
  1050. package/src/services/mcp/claudeai.ts +164 -0
  1051. package/src/services/mcp/client.ts +3348 -0
  1052. package/src/services/mcp/config.ts +1578 -0
  1053. package/src/services/mcp/elicitationHandler.ts +313 -0
  1054. package/src/services/mcp/envExpansion.ts +38 -0
  1055. package/src/services/mcp/headersHelper.ts +138 -0
  1056. package/src/services/mcp/mcpStringUtils.ts +106 -0
  1057. package/src/services/mcp/normalization.ts +23 -0
  1058. package/src/services/mcp/oauthPort.ts +78 -0
  1059. package/src/services/mcp/officialRegistry.ts +72 -0
  1060. package/src/services/mcp/types.ts +258 -0
  1061. package/src/services/mcp/useManageMCPConnections.ts +1141 -0
  1062. package/src/services/mcp/utils.ts +575 -0
  1063. package/src/services/mcp/vscodeSdkMcp.ts +112 -0
  1064. package/src/services/mcp/xaa.ts +511 -0
  1065. package/src/services/mcp/xaaIdpLogin.ts +487 -0
  1066. package/src/services/mcpServerApproval.tsx +41 -0
  1067. package/src/services/mockRateLimits.ts +882 -0
  1068. package/src/services/notifier.ts +156 -0
  1069. package/src/services/oauth/auth-code-listener.ts +211 -0
  1070. package/src/services/oauth/client.ts +566 -0
  1071. package/src/services/oauth/crypto.ts +23 -0
  1072. package/src/services/oauth/getOauthProfile.ts +53 -0
  1073. package/src/services/oauth/index.ts +198 -0
  1074. package/src/services/plugins/PluginInstallationManager.ts +184 -0
  1075. package/src/services/plugins/pluginCliCommands.ts +344 -0
  1076. package/src/services/plugins/pluginOperations.ts +1088 -0
  1077. package/src/services/policyLimits/index.ts +664 -0
  1078. package/src/services/policyLimits/types.ts +27 -0
  1079. package/src/services/preventSleep.ts +165 -0
  1080. package/src/services/rateLimitMessages.ts +344 -0
  1081. package/src/services/rateLimitMocking.ts +144 -0
  1082. package/src/services/remoteManagedSettings/index.ts +639 -0
  1083. package/src/services/remoteManagedSettings/securityCheck.tsx +74 -0
  1084. package/src/services/remoteManagedSettings/syncCache.ts +112 -0
  1085. package/src/services/remoteManagedSettings/syncCacheState.ts +96 -0
  1086. package/src/services/remoteManagedSettings/types.ts +31 -0
  1087. package/src/services/settingsSync/index.ts +581 -0
  1088. package/src/services/settingsSync/types.ts +67 -0
  1089. package/src/services/teamMemorySync/index.ts +1256 -0
  1090. package/src/services/teamMemorySync/secretScanner.ts +324 -0
  1091. package/src/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
  1092. package/src/services/teamMemorySync/types.ts +156 -0
  1093. package/src/services/teamMemorySync/watcher.ts +387 -0
  1094. package/src/services/tips/tipHistory.ts +17 -0
  1095. package/src/services/tips/tipRegistry.ts +686 -0
  1096. package/src/services/tips/tipScheduler.ts +58 -0
  1097. package/src/services/tokenEstimation.ts +495 -0
  1098. package/src/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
  1099. package/src/services/tools/StreamingToolExecutor.ts +530 -0
  1100. package/src/services/tools/toolExecution.ts +1745 -0
  1101. package/src/services/tools/toolHooks.ts +650 -0
  1102. package/src/services/tools/toolOrchestration.ts +188 -0
  1103. package/src/services/vcr.ts +406 -0
  1104. package/src/services/voice.ts +525 -0
  1105. package/src/services/voiceKeyterms.ts +106 -0
  1106. package/src/services/voiceStreamSTT.ts +544 -0
  1107. package/src/setup.ts +477 -0
  1108. package/src/skills/bundled/batch.ts +124 -0
  1109. package/src/skills/bundled/claudeApi.ts +196 -0
  1110. package/src/skills/bundled/claudeApiContent.ts +75 -0
  1111. package/src/skills/bundled/claudeInChrome.ts +34 -0
  1112. package/src/skills/bundled/debug.ts +103 -0
  1113. package/src/skills/bundled/index.ts +79 -0
  1114. package/src/skills/bundled/keybindings.ts +339 -0
  1115. package/src/skills/bundled/loop.ts +92 -0
  1116. package/src/skills/bundled/loremIpsum.ts +282 -0
  1117. package/src/skills/bundled/remember.ts +82 -0
  1118. package/src/skills/bundled/scheduleRemoteAgents.ts +447 -0
  1119. package/src/skills/bundled/simplify.ts +69 -0
  1120. package/src/skills/bundled/skillify.ts +197 -0
  1121. package/src/skills/bundled/stuck.ts +79 -0
  1122. package/src/skills/bundled/updateConfig.ts +475 -0
  1123. package/src/skills/bundled/verify.ts +30 -0
  1124. package/src/skills/bundled/verifyContent.ts +13 -0
  1125. package/src/skills/bundledSkills.ts +220 -0
  1126. package/src/skills/loadSkillsDir.ts +1086 -0
  1127. package/src/skills/mcpSkillBuilders.ts +44 -0
  1128. package/src/state/AppState.tsx +200 -0
  1129. package/src/state/AppStateStore.ts +569 -0
  1130. package/src/state/onChangeAppState.ts +171 -0
  1131. package/src/state/selectors.ts +76 -0
  1132. package/src/state/store.ts +34 -0
  1133. package/src/state/teammateViewHelpers.ts +141 -0
  1134. package/src/tasks/DreamTask/DreamTask.ts +157 -0
  1135. package/src/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  1136. package/src/tasks/InProcessTeammateTask/types.ts +121 -0
  1137. package/src/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  1138. package/src/tasks/LocalMainSessionTask.ts +479 -0
  1139. package/src/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  1140. package/src/tasks/LocalShellTask/guards.ts +41 -0
  1141. package/src/tasks/LocalShellTask/killShellTasks.ts +76 -0
  1142. package/src/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  1143. package/src/tasks/pillLabel.ts +82 -0
  1144. package/src/tasks/stopTask.ts +100 -0
  1145. package/src/tasks/types.ts +46 -0
  1146. package/src/tasks.ts +39 -0
  1147. package/src/tools/AgentTool/AgentTool.tsx +1398 -0
  1148. package/src/tools/AgentTool/UI.tsx +872 -0
  1149. package/src/tools/AgentTool/agentColorManager.ts +66 -0
  1150. package/src/tools/AgentTool/agentDisplay.ts +104 -0
  1151. package/src/tools/AgentTool/agentMemory.ts +177 -0
  1152. package/src/tools/AgentTool/agentMemorySnapshot.ts +197 -0
  1153. package/src/tools/AgentTool/agentToolUtils.ts +686 -0
  1154. package/src/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
  1155. package/src/tools/AgentTool/built-in/exploreAgent.ts +83 -0
  1156. package/src/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
  1157. package/src/tools/AgentTool/built-in/planAgent.ts +92 -0
  1158. package/src/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
  1159. package/src/tools/AgentTool/built-in/verificationAgent.ts +152 -0
  1160. package/src/tools/AgentTool/builtInAgents.ts +72 -0
  1161. package/src/tools/AgentTool/constants.ts +12 -0
  1162. package/src/tools/AgentTool/forkSubagent.ts +210 -0
  1163. package/src/tools/AgentTool/loadAgentsDir.ts +755 -0
  1164. package/src/tools/AgentTool/prompt.ts +287 -0
  1165. package/src/tools/AgentTool/resumeAgent.ts +265 -0
  1166. package/src/tools/AgentTool/runAgent.ts +973 -0
  1167. package/src/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
  1168. package/src/tools/AskUserQuestionTool/prompt.ts +44 -0
  1169. package/src/tools/BashTool/BashTool.tsx +1144 -0
  1170. package/src/tools/BashTool/BashToolResultMessage.tsx +191 -0
  1171. package/src/tools/BashTool/UI.tsx +185 -0
  1172. package/src/tools/BashTool/bashCommandHelpers.ts +265 -0
  1173. package/src/tools/BashTool/bashPermissions.ts +2621 -0
  1174. package/src/tools/BashTool/bashSecurity.ts +2592 -0
  1175. package/src/tools/BashTool/commandSemantics.ts +140 -0
  1176. package/src/tools/BashTool/commentLabel.ts +13 -0
  1177. package/src/tools/BashTool/destructiveCommandWarning.ts +102 -0
  1178. package/src/tools/BashTool/modeValidation.ts +115 -0
  1179. package/src/tools/BashTool/pathValidation.ts +1303 -0
  1180. package/src/tools/BashTool/prompt.ts +369 -0
  1181. package/src/tools/BashTool/readOnlyValidation.ts +1990 -0
  1182. package/src/tools/BashTool/sedEditParser.ts +322 -0
  1183. package/src/tools/BashTool/sedValidation.ts +684 -0
  1184. package/src/tools/BashTool/shouldUseSandbox.ts +153 -0
  1185. package/src/tools/BashTool/toolName.ts +2 -0
  1186. package/src/tools/BashTool/utils.ts +223 -0
  1187. package/src/tools/BriefTool/BriefTool.ts +204 -0
  1188. package/src/tools/BriefTool/UI.tsx +101 -0
  1189. package/src/tools/BriefTool/attachments.ts +110 -0
  1190. package/src/tools/BriefTool/prompt.ts +22 -0
  1191. package/src/tools/BriefTool/upload.ts +174 -0
  1192. package/src/tools/ConfigTool/ConfigTool.ts +467 -0
  1193. package/src/tools/ConfigTool/UI.tsx +38 -0
  1194. package/src/tools/ConfigTool/constants.ts +1 -0
  1195. package/src/tools/ConfigTool/prompt.ts +93 -0
  1196. package/src/tools/ConfigTool/supportedSettings.ts +211 -0
  1197. package/src/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
  1198. package/src/tools/EnterPlanModeTool/UI.tsx +33 -0
  1199. package/src/tools/EnterPlanModeTool/constants.ts +1 -0
  1200. package/src/tools/EnterPlanModeTool/prompt.ts +170 -0
  1201. package/src/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
  1202. package/src/tools/EnterWorktreeTool/UI.tsx +20 -0
  1203. package/src/tools/EnterWorktreeTool/constants.ts +1 -0
  1204. package/src/tools/EnterWorktreeTool/prompt.ts +30 -0
  1205. package/src/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
  1206. package/src/tools/ExitPlanModeTool/UI.tsx +82 -0
  1207. package/src/tools/ExitPlanModeTool/constants.ts +2 -0
  1208. package/src/tools/ExitPlanModeTool/prompt.ts +29 -0
  1209. package/src/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
  1210. package/src/tools/ExitWorktreeTool/UI.tsx +25 -0
  1211. package/src/tools/ExitWorktreeTool/constants.ts +1 -0
  1212. package/src/tools/ExitWorktreeTool/prompt.ts +32 -0
  1213. package/src/tools/FileEditTool/FileEditTool.ts +625 -0
  1214. package/src/tools/FileEditTool/UI.tsx +289 -0
  1215. package/src/tools/FileEditTool/constants.ts +11 -0
  1216. package/src/tools/FileEditTool/prompt.ts +28 -0
  1217. package/src/tools/FileEditTool/types.ts +85 -0
  1218. package/src/tools/FileEditTool/utils.ts +775 -0
  1219. package/src/tools/FileReadTool/FileReadTool.ts +1183 -0
  1220. package/src/tools/FileReadTool/UI.tsx +185 -0
  1221. package/src/tools/FileReadTool/imageProcessor.ts +94 -0
  1222. package/src/tools/FileReadTool/limits.ts +92 -0
  1223. package/src/tools/FileReadTool/prompt.ts +49 -0
  1224. package/src/tools/FileWriteTool/FileWriteTool.ts +434 -0
  1225. package/src/tools/FileWriteTool/UI.tsx +405 -0
  1226. package/src/tools/FileWriteTool/prompt.ts +18 -0
  1227. package/src/tools/GlobTool/GlobTool.ts +198 -0
  1228. package/src/tools/GlobTool/UI.tsx +63 -0
  1229. package/src/tools/GlobTool/prompt.ts +7 -0
  1230. package/src/tools/GrepTool/GrepTool.ts +577 -0
  1231. package/src/tools/GrepTool/UI.tsx +201 -0
  1232. package/src/tools/GrepTool/prompt.ts +18 -0
  1233. package/src/tools/LSPTool/LSPTool.ts +860 -0
  1234. package/src/tools/LSPTool/UI.tsx +228 -0
  1235. package/src/tools/LSPTool/formatters.ts +592 -0
  1236. package/src/tools/LSPTool/prompt.ts +21 -0
  1237. package/src/tools/LSPTool/schemas.ts +215 -0
  1238. package/src/tools/LSPTool/symbolContext.ts +90 -0
  1239. package/src/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
  1240. package/src/tools/ListMcpResourcesTool/UI.tsx +29 -0
  1241. package/src/tools/ListMcpResourcesTool/prompt.ts +20 -0
  1242. package/src/tools/MCPTool/MCPTool.ts +77 -0
  1243. package/src/tools/MCPTool/UI.tsx +403 -0
  1244. package/src/tools/MCPTool/classifyForCollapse.ts +604 -0
  1245. package/src/tools/MCPTool/prompt.ts +3 -0
  1246. package/src/tools/McpAuthTool/McpAuthTool.ts +215 -0
  1247. package/src/tools/NotebookEditTool/NotebookEditTool.ts +490 -0
  1248. package/src/tools/NotebookEditTool/UI.tsx +93 -0
  1249. package/src/tools/NotebookEditTool/constants.ts +2 -0
  1250. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  1251. package/src/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
  1252. package/src/tools/PowerShellTool/UI.tsx +131 -0
  1253. package/src/tools/PowerShellTool/clmTypes.ts +211 -0
  1254. package/src/tools/PowerShellTool/commandSemantics.ts +142 -0
  1255. package/src/tools/PowerShellTool/commonParameters.ts +30 -0
  1256. package/src/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
  1257. package/src/tools/PowerShellTool/gitSafety.ts +176 -0
  1258. package/src/tools/PowerShellTool/modeValidation.ts +404 -0
  1259. package/src/tools/PowerShellTool/pathValidation.ts +2049 -0
  1260. package/src/tools/PowerShellTool/powershellPermissions.ts +1648 -0
  1261. package/src/tools/PowerShellTool/powershellSecurity.ts +1090 -0
  1262. package/src/tools/PowerShellTool/prompt.ts +145 -0
  1263. package/src/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
  1264. package/src/tools/PowerShellTool/toolName.ts +2 -0
  1265. package/src/tools/REPLTool/constants.ts +46 -0
  1266. package/src/tools/REPLTool/primitiveTools.ts +39 -0
  1267. package/src/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
  1268. package/src/tools/ReadMcpResourceTool/UI.tsx +37 -0
  1269. package/src/tools/ReadMcpResourceTool/prompt.ts +16 -0
  1270. package/src/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
  1271. package/src/tools/RemoteTriggerTool/UI.tsx +17 -0
  1272. package/src/tools/RemoteTriggerTool/prompt.ts +15 -0
  1273. package/src/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
  1274. package/src/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
  1275. package/src/tools/ScheduleCronTool/CronListTool.ts +97 -0
  1276. package/src/tools/ScheduleCronTool/UI.tsx +60 -0
  1277. package/src/tools/ScheduleCronTool/prompt.ts +135 -0
  1278. package/src/tools/SendMessageTool/SendMessageTool.ts +917 -0
  1279. package/src/tools/SendMessageTool/UI.tsx +31 -0
  1280. package/src/tools/SendMessageTool/constants.ts +1 -0
  1281. package/src/tools/SendMessageTool/prompt.ts +49 -0
  1282. package/src/tools/SkillTool/SkillTool.ts +1108 -0
  1283. package/src/tools/SkillTool/UI.tsx +128 -0
  1284. package/src/tools/SkillTool/constants.ts +1 -0
  1285. package/src/tools/SkillTool/prompt.ts +241 -0
  1286. package/src/tools/SleepTool/prompt.ts +17 -0
  1287. package/src/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
  1288. package/src/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
  1289. package/src/tools/TaskCreateTool/constants.ts +1 -0
  1290. package/src/tools/TaskCreateTool/prompt.ts +56 -0
  1291. package/src/tools/TaskGetTool/TaskGetTool.ts +128 -0
  1292. package/src/tools/TaskGetTool/constants.ts +1 -0
  1293. package/src/tools/TaskGetTool/prompt.ts +24 -0
  1294. package/src/tools/TaskListTool/TaskListTool.ts +116 -0
  1295. package/src/tools/TaskListTool/constants.ts +1 -0
  1296. package/src/tools/TaskListTool/prompt.ts +49 -0
  1297. package/src/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
  1298. package/src/tools/TaskOutputTool/constants.ts +1 -0
  1299. package/src/tools/TaskStopTool/TaskStopTool.ts +131 -0
  1300. package/src/tools/TaskStopTool/UI.tsx +41 -0
  1301. package/src/tools/TaskStopTool/prompt.ts +8 -0
  1302. package/src/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
  1303. package/src/tools/TaskUpdateTool/constants.ts +1 -0
  1304. package/src/tools/TaskUpdateTool/prompt.ts +77 -0
  1305. package/src/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
  1306. package/src/tools/TeamCreateTool/UI.tsx +6 -0
  1307. package/src/tools/TeamCreateTool/constants.ts +1 -0
  1308. package/src/tools/TeamCreateTool/prompt.ts +113 -0
  1309. package/src/tools/TeamDeleteTool/TeamDeleteTool.ts +139 -0
  1310. package/src/tools/TeamDeleteTool/UI.tsx +20 -0
  1311. package/src/tools/TeamDeleteTool/constants.ts +1 -0
  1312. package/src/tools/TeamDeleteTool/prompt.ts +16 -0
  1313. package/src/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
  1314. package/src/tools/TodoWriteTool/constants.ts +1 -0
  1315. package/src/tools/TodoWriteTool/prompt.ts +184 -0
  1316. package/src/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
  1317. package/src/tools/ToolSearchTool/constants.ts +1 -0
  1318. package/src/tools/ToolSearchTool/prompt.ts +121 -0
  1319. package/src/tools/TungstenTool/TungstenTool.js +2 -0
  1320. package/src/tools/TungstenTool/TungstenTool.ts +1 -0
  1321. package/src/tools/WebFetchTool/UI.tsx +72 -0
  1322. package/src/tools/WebFetchTool/WebFetchTool.ts +318 -0
  1323. package/src/tools/WebFetchTool/preapproved.ts +166 -0
  1324. package/src/tools/WebFetchTool/prompt.ts +46 -0
  1325. package/src/tools/WebFetchTool/utils.ts +530 -0
  1326. package/src/tools/WebSearchTool/UI.tsx +101 -0
  1327. package/src/tools/WebSearchTool/WebSearchTool.ts +435 -0
  1328. package/src/tools/WebSearchTool/prompt.ts +34 -0
  1329. package/src/tools/shared/gitOperationTracking.ts +277 -0
  1330. package/src/tools/shared/spawnMultiAgent.ts +1093 -0
  1331. package/src/tools/testing/TestingPermissionTool.tsx +74 -0
  1332. package/src/tools/utils.ts +40 -0
  1333. package/src/tools.ts +389 -0
  1334. package/src/types/command.ts +216 -0
  1335. package/src/types/connectorText.js +5 -0
  1336. package/src/types/connectorText.ts +1 -0
  1337. package/src/types/generated/events_mono/claude_code/v1/claude_code_internal_event.ts +865 -0
  1338. package/src/types/generated/events_mono/common/v1/auth.ts +100 -0
  1339. package/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +223 -0
  1340. package/src/types/generated/google/protobuf/timestamp.ts +187 -0
  1341. package/src/types/hooks.ts +290 -0
  1342. package/src/types/ids.ts +44 -0
  1343. package/src/types/logs.ts +330 -0
  1344. package/src/types/permissions.ts +441 -0
  1345. package/src/types/plugin.ts +363 -0
  1346. package/src/types/textInputTypes.ts +387 -0
  1347. package/src/upstreamproxy/relay.ts +455 -0
  1348. package/src/upstreamproxy/upstreamproxy.ts +285 -0
  1349. package/src/utils/CircularBuffer.ts +84 -0
  1350. package/src/utils/Cursor.ts +1530 -0
  1351. package/src/utils/QueryGuard.ts +121 -0
  1352. package/src/utils/Shell.ts +474 -0
  1353. package/src/utils/ShellCommand.ts +465 -0
  1354. package/src/utils/abortController.ts +99 -0
  1355. package/src/utils/activityManager.ts +164 -0
  1356. package/src/utils/advisor.ts +145 -0
  1357. package/src/utils/agentContext.ts +178 -0
  1358. package/src/utils/agentId.ts +99 -0
  1359. package/src/utils/agentSwarmsEnabled.ts +44 -0
  1360. package/src/utils/agenticSessionSearch.ts +307 -0
  1361. package/src/utils/analyzeContext.ts +1382 -0
  1362. package/src/utils/ansiToPng.ts +334 -0
  1363. package/src/utils/ansiToSvg.ts +272 -0
  1364. package/src/utils/api.ts +718 -0
  1365. package/src/utils/apiPreconnect.ts +72 -0
  1366. package/src/utils/appleTerminalBackup.ts +124 -0
  1367. package/src/utils/argumentSubstitution.ts +145 -0
  1368. package/src/utils/array.ts +13 -0
  1369. package/src/utils/asciicast.ts +239 -0
  1370. package/src/utils/attachments.ts +3997 -0
  1371. package/src/utils/attribution.ts +393 -0
  1372. package/src/utils/auth.ts +2007 -0
  1373. package/src/utils/authFileDescriptor.ts +196 -0
  1374. package/src/utils/authPortable.ts +19 -0
  1375. package/src/utils/autoModeDenials.ts +26 -0
  1376. package/src/utils/autoRunIssue.tsx +122 -0
  1377. package/src/utils/autoUpdater.ts +562 -0
  1378. package/src/utils/aws.ts +74 -0
  1379. package/src/utils/awsAuthStatusManager.ts +81 -0
  1380. package/src/utils/background/remote/preconditions.ts +235 -0
  1381. package/src/utils/background/remote/remoteSession.ts +98 -0
  1382. package/src/utils/backgroundHousekeeping.ts +94 -0
  1383. package/src/utils/bash/ParsedCommand.ts +318 -0
  1384. package/src/utils/bash/ShellSnapshot.ts +582 -0
  1385. package/src/utils/bash/ast.ts +2679 -0
  1386. package/src/utils/bash/bashParser.ts +4436 -0
  1387. package/src/utils/bash/bashPipeCommand.ts +294 -0
  1388. package/src/utils/bash/commands.ts +1339 -0
  1389. package/src/utils/bash/heredoc.ts +733 -0
  1390. package/src/utils/bash/parser.ts +230 -0
  1391. package/src/utils/bash/prefix.ts +204 -0
  1392. package/src/utils/bash/registry.ts +53 -0
  1393. package/src/utils/bash/shellCompletion.ts +259 -0
  1394. package/src/utils/bash/shellPrefix.ts +28 -0
  1395. package/src/utils/bash/shellQuote.ts +304 -0
  1396. package/src/utils/bash/shellQuoting.ts +128 -0
  1397. package/src/utils/bash/specs/alias.ts +14 -0
  1398. package/src/utils/bash/specs/index.ts +18 -0
  1399. package/src/utils/bash/specs/nohup.ts +13 -0
  1400. package/src/utils/bash/specs/pyright.ts +91 -0
  1401. package/src/utils/bash/specs/sleep.ts +13 -0
  1402. package/src/utils/bash/specs/srun.ts +31 -0
  1403. package/src/utils/bash/specs/time.ts +13 -0
  1404. package/src/utils/bash/specs/timeout.ts +20 -0
  1405. package/src/utils/bash/treeSitterAnalysis.ts +506 -0
  1406. package/src/utils/betas.ts +438 -0
  1407. package/src/utils/billing.ts +78 -0
  1408. package/src/utils/binaryCheck.ts +53 -0
  1409. package/src/utils/browser.ts +68 -0
  1410. package/src/utils/bufferedWriter.ts +100 -0
  1411. package/src/utils/bundledMode.ts +22 -0
  1412. package/src/utils/caCerts.ts +115 -0
  1413. package/src/utils/caCertsConfig.ts +88 -0
  1414. package/src/utils/cachePaths.ts +38 -0
  1415. package/src/utils/classifierApprovals.ts +88 -0
  1416. package/src/utils/classifierApprovalsHook.ts +17 -0
  1417. package/src/utils/claudeCodeHints.ts +193 -0
  1418. package/src/utils/claudeDesktop.ts +152 -0
  1419. package/src/utils/claudeInChrome/chromeNativeHost.ts +527 -0
  1420. package/src/utils/claudeInChrome/common.ts +540 -0
  1421. package/src/utils/claudeInChrome/mcpServer.ts +293 -0
  1422. package/src/utils/claudeInChrome/prompt.ts +83 -0
  1423. package/src/utils/claudeInChrome/setup.ts +400 -0
  1424. package/src/utils/claudeInChrome/setupPortable.ts +233 -0
  1425. package/src/utils/claudeInChrome/toolRendering.tsx +262 -0
  1426. package/src/utils/claudemd.ts +1479 -0
  1427. package/src/utils/cleanup.ts +602 -0
  1428. package/src/utils/cleanupRegistry.ts +25 -0
  1429. package/src/utils/cliArgs.ts +60 -0
  1430. package/src/utils/cliHighlight.ts +54 -0
  1431. package/src/utils/codeIndexing.ts +206 -0
  1432. package/src/utils/collapseBackgroundBashNotifications.ts +84 -0
  1433. package/src/utils/collapseHookSummaries.ts +59 -0
  1434. package/src/utils/collapseReadSearch.ts +1109 -0
  1435. package/src/utils/collapseTeammateShutdowns.ts +55 -0
  1436. package/src/utils/combinedAbortSignal.ts +47 -0
  1437. package/src/utils/commandLifecycle.ts +21 -0
  1438. package/src/utils/commitAttribution.ts +961 -0
  1439. package/src/utils/completionCache.ts +166 -0
  1440. package/src/utils/computerUse/appNames.ts +196 -0
  1441. package/src/utils/computerUse/cleanup.ts +86 -0
  1442. package/src/utils/computerUse/common.ts +61 -0
  1443. package/src/utils/computerUse/computerUseLock.ts +215 -0
  1444. package/src/utils/computerUse/drainRunLoop.ts +79 -0
  1445. package/src/utils/computerUse/escHotkey.ts +54 -0
  1446. package/src/utils/computerUse/executor.ts +658 -0
  1447. package/src/utils/computerUse/gates.ts +72 -0
  1448. package/src/utils/computerUse/hostAdapter.ts +69 -0
  1449. package/src/utils/computerUse/inputLoader.ts +30 -0
  1450. package/src/utils/computerUse/mcpServer.ts +106 -0
  1451. package/src/utils/computerUse/setup.ts +53 -0
  1452. package/src/utils/computerUse/swiftLoader.ts +23 -0
  1453. package/src/utils/computerUse/toolRendering.tsx +125 -0
  1454. package/src/utils/computerUse/wrapper.tsx +336 -0
  1455. package/src/utils/concurrentSessions.ts +204 -0
  1456. package/src/utils/config.ts +1817 -0
  1457. package/src/utils/configConstants.ts +21 -0
  1458. package/src/utils/contentArray.ts +51 -0
  1459. package/src/utils/context.ts +221 -0
  1460. package/src/utils/contextAnalysis.ts +272 -0
  1461. package/src/utils/contextSuggestions.ts +235 -0
  1462. package/src/utils/controlMessageCompat.ts +32 -0
  1463. package/src/utils/conversationRecovery.ts +597 -0
  1464. package/src/utils/cron.ts +308 -0
  1465. package/src/utils/cronJitterConfig.ts +75 -0
  1466. package/src/utils/cronScheduler.ts +565 -0
  1467. package/src/utils/cronTasks.ts +458 -0
  1468. package/src/utils/cronTasksLock.ts +195 -0
  1469. package/src/utils/crossProjectResume.ts +75 -0
  1470. package/src/utils/crypto.ts +13 -0
  1471. package/src/utils/cwd.ts +32 -0
  1472. package/src/utils/debug.ts +268 -0
  1473. package/src/utils/debugFilter.ts +157 -0
  1474. package/src/utils/deepLink/banner.ts +123 -0
  1475. package/src/utils/deepLink/parseDeepLink.ts +170 -0
  1476. package/src/utils/deepLink/protocolHandler.ts +136 -0
  1477. package/src/utils/deepLink/registerProtocol.ts +348 -0
  1478. package/src/utils/deepLink/terminalLauncher.ts +557 -0
  1479. package/src/utils/deepLink/terminalPreference.ts +54 -0
  1480. package/src/utils/desktopDeepLink.ts +236 -0
  1481. package/src/utils/detectRepository.ts +178 -0
  1482. package/src/utils/diagLogs.ts +94 -0
  1483. package/src/utils/diff.ts +177 -0
  1484. package/src/utils/directMemberMessage.ts +69 -0
  1485. package/src/utils/displayTags.ts +51 -0
  1486. package/src/utils/doctorContextWarnings.ts +265 -0
  1487. package/src/utils/doctorDiagnostic.ts +625 -0
  1488. package/src/utils/dxt/helpers.ts +88 -0
  1489. package/src/utils/dxt/zip.ts +226 -0
  1490. package/src/utils/earlyInput.ts +191 -0
  1491. package/src/utils/editor.ts +183 -0
  1492. package/src/utils/effort.ts +329 -0
  1493. package/src/utils/embeddedTools.ts +29 -0
  1494. package/src/utils/env.ts +347 -0
  1495. package/src/utils/envDynamic.ts +151 -0
  1496. package/src/utils/envUtils.ts +183 -0
  1497. package/src/utils/envValidation.ts +38 -0
  1498. package/src/utils/errorLogSink.ts +235 -0
  1499. package/src/utils/errors.ts +238 -0
  1500. package/src/utils/exampleCommands.ts +184 -0
  1501. package/src/utils/execFileNoThrow.ts +150 -0
  1502. package/src/utils/execFileNoThrowPortable.ts +89 -0
  1503. package/src/utils/execSyncWrapper.ts +38 -0
  1504. package/src/utils/exportRenderer.tsx +98 -0
  1505. package/src/utils/extraUsage.ts +23 -0
  1506. package/src/utils/fastMode.ts +532 -0
  1507. package/src/utils/file.ts +584 -0
  1508. package/src/utils/fileHistory.ts +1115 -0
  1509. package/src/utils/fileOperationAnalytics.ts +71 -0
  1510. package/src/utils/filePersistence/filePersistence.ts +287 -0
  1511. package/src/utils/filePersistence/outputsScanner.ts +126 -0
  1512. package/src/utils/fileRead.ts +102 -0
  1513. package/src/utils/fileReadCache.ts +96 -0
  1514. package/src/utils/fileStateCache.ts +142 -0
  1515. package/src/utils/findExecutable.ts +17 -0
  1516. package/src/utils/fingerprint.ts +76 -0
  1517. package/src/utils/forkedAgent.ts +689 -0
  1518. package/src/utils/format.ts +308 -0
  1519. package/src/utils/formatBriefTimestamp.ts +81 -0
  1520. package/src/utils/fpsTracker.ts +47 -0
  1521. package/src/utils/frontmatterParser.ts +370 -0
  1522. package/src/utils/fsOperations.ts +770 -0
  1523. package/src/utils/fullscreen.ts +202 -0
  1524. package/src/utils/generatedFiles.ts +136 -0
  1525. package/src/utils/generators.ts +88 -0
  1526. package/src/utils/genericProcessUtils.ts +184 -0
  1527. package/src/utils/getWorktreePaths.ts +70 -0
  1528. package/src/utils/getWorktreePathsPortable.ts +27 -0
  1529. package/src/utils/ghPrStatus.ts +106 -0
  1530. package/src/utils/git/gitConfigParser.ts +277 -0
  1531. package/src/utils/git/gitFilesystem.ts +699 -0
  1532. package/src/utils/git/gitignore.ts +99 -0
  1533. package/src/utils/git.ts +926 -0
  1534. package/src/utils/gitDiff.ts +532 -0
  1535. package/src/utils/gitSettings.ts +18 -0
  1536. package/src/utils/github/ghAuthStatus.ts +29 -0
  1537. package/src/utils/githubRepoPathMapping.ts +162 -0
  1538. package/src/utils/glob.ts +130 -0
  1539. package/src/utils/gracefulShutdown.ts +529 -0
  1540. package/src/utils/groupToolUses.ts +182 -0
  1541. package/src/utils/handlePromptSubmit.ts +610 -0
  1542. package/src/utils/hash.ts +46 -0
  1543. package/src/utils/headlessProfiler.ts +178 -0
  1544. package/src/utils/heapDumpService.ts +303 -0
  1545. package/src/utils/heatmap.ts +198 -0
  1546. package/src/utils/highlightMatch.tsx +28 -0
  1547. package/src/utils/hooks/AsyncHookRegistry.ts +309 -0
  1548. package/src/utils/hooks/apiQueryHookHelper.ts +141 -0
  1549. package/src/utils/hooks/execAgentHook.ts +339 -0
  1550. package/src/utils/hooks/execHttpHook.ts +242 -0
  1551. package/src/utils/hooks/execPromptHook.ts +211 -0
  1552. package/src/utils/hooks/fileChangedWatcher.ts +191 -0
  1553. package/src/utils/hooks/hookEvents.ts +192 -0
  1554. package/src/utils/hooks/hookHelpers.ts +83 -0
  1555. package/src/utils/hooks/hooksConfigManager.ts +400 -0
  1556. package/src/utils/hooks/hooksConfigSnapshot.ts +133 -0
  1557. package/src/utils/hooks/hooksSettings.ts +271 -0
  1558. package/src/utils/hooks/postSamplingHooks.ts +70 -0
  1559. package/src/utils/hooks/registerFrontmatterHooks.ts +67 -0
  1560. package/src/utils/hooks/registerSkillHooks.ts +64 -0
  1561. package/src/utils/hooks/sessionHooks.ts +447 -0
  1562. package/src/utils/hooks/skillImprovement.ts +267 -0
  1563. package/src/utils/hooks/ssrfGuard.ts +294 -0
  1564. package/src/utils/hooks.ts +5022 -0
  1565. package/src/utils/horizontalScroll.ts +137 -0
  1566. package/src/utils/http.ts +136 -0
  1567. package/src/utils/hyperlink.ts +39 -0
  1568. package/src/utils/iTermBackup.ts +73 -0
  1569. package/src/utils/ide.ts +1494 -0
  1570. package/src/utils/idePathConversion.ts +90 -0
  1571. package/src/utils/idleTimeout.ts +53 -0
  1572. package/src/utils/imagePaste.ts +416 -0
  1573. package/src/utils/imageResizer.ts +880 -0
  1574. package/src/utils/imageStore.ts +167 -0
  1575. package/src/utils/imageValidation.ts +104 -0
  1576. package/src/utils/immediateCommand.ts +15 -0
  1577. package/src/utils/inProcessTeammateHelpers.ts +102 -0
  1578. package/src/utils/ink.ts +26 -0
  1579. package/src/utils/intl.ts +94 -0
  1580. package/src/utils/jetbrains.ts +191 -0
  1581. package/src/utils/json.ts +277 -0
  1582. package/src/utils/jsonRead.ts +16 -0
  1583. package/src/utils/keyboardShortcuts.ts +14 -0
  1584. package/src/utils/lazySchema.ts +8 -0
  1585. package/src/utils/listSessionsImpl.ts +454 -0
  1586. package/src/utils/localInstaller.ts +162 -0
  1587. package/src/utils/lockfile.ts +43 -0
  1588. package/src/utils/log.ts +362 -0
  1589. package/src/utils/logoV2Utils.ts +350 -0
  1590. package/src/utils/mailbox.ts +73 -0
  1591. package/src/utils/managedEnv.ts +199 -0
  1592. package/src/utils/managedEnvConstants.ts +191 -0
  1593. package/src/utils/markdown.ts +381 -0
  1594. package/src/utils/markdownConfigLoader.ts +600 -0
  1595. package/src/utils/mcp/dateTimeParser.ts +121 -0
  1596. package/src/utils/mcp/elicitationValidation.ts +336 -0
  1597. package/src/utils/mcpInstructionsDelta.ts +130 -0
  1598. package/src/utils/mcpOutputStorage.ts +189 -0
  1599. package/src/utils/mcpValidation.ts +208 -0
  1600. package/src/utils/mcpWebSocketTransport.ts +200 -0
  1601. package/src/utils/memoize.ts +269 -0
  1602. package/src/utils/memory/types.ts +12 -0
  1603. package/src/utils/memory/versions.ts +8 -0
  1604. package/src/utils/memoryFileDetection.ts +289 -0
  1605. package/src/utils/messagePredicates.ts +8 -0
  1606. package/src/utils/messageQueueManager.ts +547 -0
  1607. package/src/utils/messages/mappers.ts +290 -0
  1608. package/src/utils/messages/systemInit.ts +96 -0
  1609. package/src/utils/messages.ts +5512 -0
  1610. package/src/utils/model/agent.ts +157 -0
  1611. package/src/utils/model/aliases.ts +25 -0
  1612. package/src/utils/model/antModels.ts +64 -0
  1613. package/src/utils/model/bedrock.ts +265 -0
  1614. package/src/utils/model/check1mAccess.ts +72 -0
  1615. package/src/utils/model/configs.ts +118 -0
  1616. package/src/utils/model/contextWindowUpgradeCheck.ts +47 -0
  1617. package/src/utils/model/deprecation.ts +101 -0
  1618. package/src/utils/model/model.ts +634 -0
  1619. package/src/utils/model/modelAllowlist.ts +170 -0
  1620. package/src/utils/model/modelCapabilities.ts +118 -0
  1621. package/src/utils/model/modelOptions.ts +540 -0
  1622. package/src/utils/model/modelStrings.ts +166 -0
  1623. package/src/utils/model/modelSupportOverrides.ts +50 -0
  1624. package/src/utils/model/providers.ts +46 -0
  1625. package/src/utils/model/validateModel.ts +159 -0
  1626. package/src/utils/modelCost.ts +235 -0
  1627. package/src/utils/modifiers.ts +36 -0
  1628. package/src/utils/mtls.ts +179 -0
  1629. package/src/utils/nativeInstaller/download.ts +523 -0
  1630. package/src/utils/nativeInstaller/index.ts +18 -0
  1631. package/src/utils/nativeInstaller/installer.ts +1708 -0
  1632. package/src/utils/nativeInstaller/packageManagers.ts +336 -0
  1633. package/src/utils/nativeInstaller/pidLock.ts +433 -0
  1634. package/src/utils/notebook.ts +224 -0
  1635. package/src/utils/objectGroupBy.ts +18 -0
  1636. package/src/utils/pasteStore.ts +104 -0
  1637. package/src/utils/path.ts +155 -0
  1638. package/src/utils/pdf.ts +300 -0
  1639. package/src/utils/pdfUtils.ts +70 -0
  1640. package/src/utils/peerAddress.ts +21 -0
  1641. package/src/utils/permissions/PermissionMode.ts +141 -0
  1642. package/src/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
  1643. package/src/utils/permissions/PermissionResult.ts +35 -0
  1644. package/src/utils/permissions/PermissionRule.ts +40 -0
  1645. package/src/utils/permissions/PermissionUpdate.ts +389 -0
  1646. package/src/utils/permissions/PermissionUpdateSchema.ts +78 -0
  1647. package/src/utils/permissions/autoModeState.ts +39 -0
  1648. package/src/utils/permissions/bashClassifier.ts +61 -0
  1649. package/src/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
  1650. package/src/utils/permissions/classifierDecision.ts +98 -0
  1651. package/src/utils/permissions/classifierShared.ts +39 -0
  1652. package/src/utils/permissions/dangerousPatterns.ts +80 -0
  1653. package/src/utils/permissions/denialTracking.ts +45 -0
  1654. package/src/utils/permissions/filesystem.ts +1777 -0
  1655. package/src/utils/permissions/getNextPermissionMode.ts +101 -0
  1656. package/src/utils/permissions/pathValidation.ts +485 -0
  1657. package/src/utils/permissions/permissionExplainer.ts +250 -0
  1658. package/src/utils/permissions/permissionRuleParser.ts +198 -0
  1659. package/src/utils/permissions/permissionSetup.ts +1532 -0
  1660. package/src/utils/permissions/permissions.ts +1486 -0
  1661. package/src/utils/permissions/permissionsLoader.ts +296 -0
  1662. package/src/utils/permissions/shadowedRuleDetection.ts +234 -0
  1663. package/src/utils/permissions/shellRuleMatching.ts +228 -0
  1664. package/src/utils/permissions/yoloClassifier.ts +1495 -0
  1665. package/src/utils/planModeV2.ts +95 -0
  1666. package/src/utils/plans.ts +397 -0
  1667. package/src/utils/platform.ts +150 -0
  1668. package/src/utils/plugins/addDirPluginSettings.ts +71 -0
  1669. package/src/utils/plugins/cacheUtils.ts +196 -0
  1670. package/src/utils/plugins/dependencyResolver.ts +305 -0
  1671. package/src/utils/plugins/fetchTelemetry.ts +135 -0
  1672. package/src/utils/plugins/gitAvailability.ts +69 -0
  1673. package/src/utils/plugins/headlessPluginInstall.ts +174 -0
  1674. package/src/utils/plugins/hintRecommendation.ts +164 -0
  1675. package/src/utils/plugins/installCounts.ts +292 -0
  1676. package/src/utils/plugins/installedPluginsManager.ts +1268 -0
  1677. package/src/utils/plugins/loadPluginAgents.ts +348 -0
  1678. package/src/utils/plugins/loadPluginCommands.ts +946 -0
  1679. package/src/utils/plugins/loadPluginHooks.ts +287 -0
  1680. package/src/utils/plugins/loadPluginOutputStyles.ts +178 -0
  1681. package/src/utils/plugins/lspPluginIntegration.ts +387 -0
  1682. package/src/utils/plugins/lspRecommendation.ts +374 -0
  1683. package/src/utils/plugins/managedPlugins.ts +27 -0
  1684. package/src/utils/plugins/marketplaceHelpers.ts +592 -0
  1685. package/src/utils/plugins/marketplaceManager.ts +2643 -0
  1686. package/src/utils/plugins/mcpPluginIntegration.ts +634 -0
  1687. package/src/utils/plugins/mcpbHandler.ts +968 -0
  1688. package/src/utils/plugins/officialMarketplace.ts +25 -0
  1689. package/src/utils/plugins/officialMarketplaceGcs.ts +216 -0
  1690. package/src/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
  1691. package/src/utils/plugins/orphanedPluginFilter.ts +114 -0
  1692. package/src/utils/plugins/parseMarketplaceInput.ts +162 -0
  1693. package/src/utils/plugins/performStartupChecks.tsx +70 -0
  1694. package/src/utils/plugins/pluginAutoupdate.ts +284 -0
  1695. package/src/utils/plugins/pluginBlocklist.ts +127 -0
  1696. package/src/utils/plugins/pluginDirectories.ts +178 -0
  1697. package/src/utils/plugins/pluginFlagging.ts +208 -0
  1698. package/src/utils/plugins/pluginIdentifier.ts +123 -0
  1699. package/src/utils/plugins/pluginInstallationHelpers.ts +595 -0
  1700. package/src/utils/plugins/pluginLoader.ts +3302 -0
  1701. package/src/utils/plugins/pluginOptionsStorage.ts +400 -0
  1702. package/src/utils/plugins/pluginPolicy.ts +20 -0
  1703. package/src/utils/plugins/pluginStartupCheck.ts +341 -0
  1704. package/src/utils/plugins/pluginVersioning.ts +157 -0
  1705. package/src/utils/plugins/reconciler.ts +265 -0
  1706. package/src/utils/plugins/refresh.ts +215 -0
  1707. package/src/utils/plugins/schemas.ts +1681 -0
  1708. package/src/utils/plugins/validatePlugin.ts +903 -0
  1709. package/src/utils/plugins/walkPluginMarkdown.ts +69 -0
  1710. package/src/utils/plugins/zipCache.ts +406 -0
  1711. package/src/utils/plugins/zipCacheAdapters.ts +164 -0
  1712. package/src/utils/powershell/dangerousCmdlets.ts +185 -0
  1713. package/src/utils/powershell/parser.ts +1804 -0
  1714. package/src/utils/powershell/staticPrefix.ts +316 -0
  1715. package/src/utils/preflightChecks.tsx +151 -0
  1716. package/src/utils/privacyLevel.ts +55 -0
  1717. package/src/utils/process.ts +68 -0
  1718. package/src/utils/processUserInput/processBashCommand.tsx +140 -0
  1719. package/src/utils/processUserInput/processSlashCommand.tsx +922 -0
  1720. package/src/utils/processUserInput/processTextPrompt.ts +100 -0
  1721. package/src/utils/processUserInput/processUserInput.ts +605 -0
  1722. package/src/utils/profilerBase.ts +46 -0
  1723. package/src/utils/promptCategory.ts +49 -0
  1724. package/src/utils/promptEditor.ts +188 -0
  1725. package/src/utils/promptShellExecution.ts +183 -0
  1726. package/src/utils/proxy.ts +426 -0
  1727. package/src/utils/queryContext.ts +179 -0
  1728. package/src/utils/queryHelpers.ts +552 -0
  1729. package/src/utils/queryProfiler.ts +301 -0
  1730. package/src/utils/queueProcessor.ts +95 -0
  1731. package/src/utils/readEditContext.ts +227 -0
  1732. package/src/utils/readFileInRange.ts +383 -0
  1733. package/src/utils/releaseNotes.ts +360 -0
  1734. package/src/utils/renderOptions.ts +77 -0
  1735. package/src/utils/ripgrep.ts +679 -0
  1736. package/src/utils/sandbox/sandbox-adapter.ts +985 -0
  1737. package/src/utils/sandbox/sandbox-ui-utils.ts +12 -0
  1738. package/src/utils/sanitization.ts +91 -0
  1739. package/src/utils/screenshotClipboard.ts +121 -0
  1740. package/src/utils/sdkEventQueue.ts +134 -0
  1741. package/src/utils/secureStorage/fallbackStorage.ts +70 -0
  1742. package/src/utils/secureStorage/index.ts +17 -0
  1743. package/src/utils/secureStorage/keychainPrefetch.ts +116 -0
  1744. package/src/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
  1745. package/src/utils/secureStorage/macOsKeychainStorage.ts +231 -0
  1746. package/src/utils/secureStorage/plainTextStorage.ts +84 -0
  1747. package/src/utils/semanticBoolean.ts +29 -0
  1748. package/src/utils/semanticNumber.ts +36 -0
  1749. package/src/utils/semver.ts +59 -0
  1750. package/src/utils/sequential.ts +56 -0
  1751. package/src/utils/sessionActivity.ts +133 -0
  1752. package/src/utils/sessionEnvVars.ts +22 -0
  1753. package/src/utils/sessionEnvironment.ts +166 -0
  1754. package/src/utils/sessionFileAccessHooks.ts +250 -0
  1755. package/src/utils/sessionIngressAuth.ts +140 -0
  1756. package/src/utils/sessionRestore.ts +551 -0
  1757. package/src/utils/sessionStart.ts +232 -0
  1758. package/src/utils/sessionState.ts +150 -0
  1759. package/src/utils/sessionStorage.ts +5105 -0
  1760. package/src/utils/sessionStoragePortable.ts +793 -0
  1761. package/src/utils/sessionTitle.ts +129 -0
  1762. package/src/utils/sessionUrl.ts +64 -0
  1763. package/src/utils/set.ts +53 -0
  1764. package/src/utils/settings/allErrors.ts +32 -0
  1765. package/src/utils/settings/applySettingsChange.ts +92 -0
  1766. package/src/utils/settings/changeDetector.ts +488 -0
  1767. package/src/utils/settings/constants.ts +202 -0
  1768. package/src/utils/settings/internalWrites.ts +37 -0
  1769. package/src/utils/settings/managedPath.ts +34 -0
  1770. package/src/utils/settings/mdm/constants.ts +81 -0
  1771. package/src/utils/settings/mdm/rawRead.ts +130 -0
  1772. package/src/utils/settings/mdm/settings.ts +316 -0
  1773. package/src/utils/settings/permissionValidation.ts +262 -0
  1774. package/src/utils/settings/pluginOnlyPolicy.ts +60 -0
  1775. package/src/utils/settings/schemaOutput.ts +8 -0
  1776. package/src/utils/settings/settings.ts +1015 -0
  1777. package/src/utils/settings/settingsCache.ts +80 -0
  1778. package/src/utils/settings/toolValidationConfig.ts +103 -0
  1779. package/src/utils/settings/types.ts +1148 -0
  1780. package/src/utils/settings/validateEditTool.ts +45 -0
  1781. package/src/utils/settings/validation.ts +265 -0
  1782. package/src/utils/settings/validationTips.ts +164 -0
  1783. package/src/utils/shell/bashProvider.ts +255 -0
  1784. package/src/utils/shell/outputLimits.ts +14 -0
  1785. package/src/utils/shell/powershellDetection.ts +107 -0
  1786. package/src/utils/shell/powershellProvider.ts +123 -0
  1787. package/src/utils/shell/prefix.ts +367 -0
  1788. package/src/utils/shell/readOnlyCommandValidation.ts +1893 -0
  1789. package/src/utils/shell/resolveDefaultShell.ts +14 -0
  1790. package/src/utils/shell/shellProvider.ts +33 -0
  1791. package/src/utils/shell/shellToolUtils.ts +22 -0
  1792. package/src/utils/shell/specPrefix.ts +241 -0
  1793. package/src/utils/shellConfig.ts +167 -0
  1794. package/src/utils/sideQuery.ts +222 -0
  1795. package/src/utils/sideQuestion.ts +155 -0
  1796. package/src/utils/signal.ts +43 -0
  1797. package/src/utils/sinks.ts +16 -0
  1798. package/src/utils/skills/skillChangeDetector.ts +311 -0
  1799. package/src/utils/slashCommandParsing.ts +60 -0
  1800. package/src/utils/sleep.ts +84 -0
  1801. package/src/utils/sliceAnsi.ts +91 -0
  1802. package/src/utils/slowOperations.ts +286 -0
  1803. package/src/utils/standaloneAgent.ts +23 -0
  1804. package/src/utils/startupProfiler.ts +194 -0
  1805. package/src/utils/staticRender.tsx +116 -0
  1806. package/src/utils/stats.ts +1061 -0
  1807. package/src/utils/statsCache.ts +434 -0
  1808. package/src/utils/status.tsx +362 -0
  1809. package/src/utils/statusNoticeDefinitions.tsx +198 -0
  1810. package/src/utils/statusNoticeHelpers.ts +20 -0
  1811. package/src/utils/stream.ts +76 -0
  1812. package/src/utils/streamJsonStdoutGuard.ts +123 -0
  1813. package/src/utils/streamlinedTransform.ts +201 -0
  1814. package/src/utils/stringUtils.ts +235 -0
  1815. package/src/utils/subprocessEnv.ts +99 -0
  1816. package/src/utils/suggestions/commandSuggestions.ts +567 -0
  1817. package/src/utils/suggestions/directoryCompletion.ts +263 -0
  1818. package/src/utils/suggestions/shellHistoryCompletion.ts +119 -0
  1819. package/src/utils/suggestions/skillUsageTracking.ts +55 -0
  1820. package/src/utils/suggestions/slackChannelSuggestions.ts +209 -0
  1821. package/src/utils/swarm/It2SetupPrompt.tsx +380 -0
  1822. package/src/utils/swarm/backends/ITermBackend.ts +370 -0
  1823. package/src/utils/swarm/backends/InProcessBackend.ts +339 -0
  1824. package/src/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
  1825. package/src/utils/swarm/backends/TmuxBackend.ts +764 -0
  1826. package/src/utils/swarm/backends/detection.ts +128 -0
  1827. package/src/utils/swarm/backends/it2Setup.ts +245 -0
  1828. package/src/utils/swarm/backends/registry.ts +464 -0
  1829. package/src/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
  1830. package/src/utils/swarm/backends/types.ts +311 -0
  1831. package/src/utils/swarm/constants.ts +33 -0
  1832. package/src/utils/swarm/inProcessRunner.ts +1552 -0
  1833. package/src/utils/swarm/leaderPermissionBridge.ts +54 -0
  1834. package/src/utils/swarm/permissionSync.ts +928 -0
  1835. package/src/utils/swarm/reconnection.ts +119 -0
  1836. package/src/utils/swarm/spawnInProcess.ts +328 -0
  1837. package/src/utils/swarm/spawnUtils.ts +146 -0
  1838. package/src/utils/swarm/teamHelpers.ts +683 -0
  1839. package/src/utils/swarm/teammateInit.ts +129 -0
  1840. package/src/utils/swarm/teammateLayoutManager.ts +107 -0
  1841. package/src/utils/swarm/teammateModel.ts +10 -0
  1842. package/src/utils/swarm/teammatePromptAddendum.ts +18 -0
  1843. package/src/utils/systemDirectories.ts +74 -0
  1844. package/src/utils/systemPrompt.ts +123 -0
  1845. package/src/utils/systemPromptType.ts +14 -0
  1846. package/src/utils/systemTheme.ts +119 -0
  1847. package/src/utils/taggedId.ts +54 -0
  1848. package/src/utils/task/TaskOutput.ts +390 -0
  1849. package/src/utils/task/diskOutput.ts +451 -0
  1850. package/src/utils/task/framework.ts +308 -0
  1851. package/src/utils/task/outputFormatting.ts +38 -0
  1852. package/src/utils/task/sdkProgress.ts +36 -0
  1853. package/src/utils/tasks.ts +862 -0
  1854. package/src/utils/teamDiscovery.ts +81 -0
  1855. package/src/utils/teamMemoryOps.ts +88 -0
  1856. package/src/utils/teammate.ts +292 -0
  1857. package/src/utils/teammateContext.ts +96 -0
  1858. package/src/utils/teammateMailbox.ts +1183 -0
  1859. package/src/utils/telemetry/betaSessionTracing.ts +491 -0
  1860. package/src/utils/telemetry/bigqueryExporter.ts +252 -0
  1861. package/src/utils/telemetry/events.ts +75 -0
  1862. package/src/utils/telemetry/instrumentation.ts +825 -0
  1863. package/src/utils/telemetry/logger.ts +26 -0
  1864. package/src/utils/telemetry/perfettoTracing.ts +1120 -0
  1865. package/src/utils/telemetry/pluginTelemetry.ts +289 -0
  1866. package/src/utils/telemetry/sessionTracing.ts +927 -0
  1867. package/src/utils/telemetry/skillLoadedEvent.ts +39 -0
  1868. package/src/utils/telemetryAttributes.ts +71 -0
  1869. package/src/utils/teleport/api.ts +466 -0
  1870. package/src/utils/teleport/environmentSelection.ts +77 -0
  1871. package/src/utils/teleport/environments.ts +120 -0
  1872. package/src/utils/teleport/gitBundle.ts +292 -0
  1873. package/src/utils/teleport.tsx +1226 -0
  1874. package/src/utils/tempfile.ts +31 -0
  1875. package/src/utils/terminal.ts +131 -0
  1876. package/src/utils/terminalPanel.ts +191 -0
  1877. package/src/utils/textHighlighting.ts +166 -0
  1878. package/src/utils/theme.ts +639 -0
  1879. package/src/utils/thinking.ts +162 -0
  1880. package/src/utils/timeouts.ts +39 -0
  1881. package/src/utils/tmuxSocket.ts +427 -0
  1882. package/src/utils/todo/types.ts +18 -0
  1883. package/src/utils/tokenBudget.ts +73 -0
  1884. package/src/utils/tokens.ts +261 -0
  1885. package/src/utils/toolErrors.ts +132 -0
  1886. package/src/utils/toolPool.ts +79 -0
  1887. package/src/utils/toolResultStorage.ts +1040 -0
  1888. package/src/utils/toolSchemaCache.ts +26 -0
  1889. package/src/utils/toolSearch.ts +756 -0
  1890. package/src/utils/transcriptSearch.ts +202 -0
  1891. package/src/utils/treeify.ts +170 -0
  1892. package/src/utils/truncate.ts +179 -0
  1893. package/src/utils/ultraplan/ccrSession.ts +349 -0
  1894. package/src/utils/ultraplan/keyword.ts +127 -0
  1895. package/src/utils/ultraplan/prompt.txt +1 -0
  1896. package/src/utils/unaryLogging.ts +39 -0
  1897. package/src/utils/undercover.ts +89 -0
  1898. package/src/utils/user.ts +194 -0
  1899. package/src/utils/userAgent.ts +10 -0
  1900. package/src/utils/userPromptKeywords.ts +27 -0
  1901. package/src/utils/uuid.ts +27 -0
  1902. package/src/utils/warningHandler.ts +121 -0
  1903. package/src/utils/which.ts +82 -0
  1904. package/src/utils/windowsPaths.ts +173 -0
  1905. package/src/utils/withResolvers.ts +13 -0
  1906. package/src/utils/words.ts +800 -0
  1907. package/src/utils/workloadContext.ts +57 -0
  1908. package/src/utils/worktree.ts +1519 -0
  1909. package/src/utils/worktreeModeEnabled.ts +11 -0
  1910. package/src/utils/xdg.ts +65 -0
  1911. package/src/utils/xml.ts +16 -0
  1912. package/src/utils/yaml.ts +15 -0
  1913. package/src/utils/zodToJsonSchema.ts +23 -0
  1914. package/src/vim/motions.ts +82 -0
  1915. package/src/vim/operators.ts +556 -0
  1916. package/src/vim/textObjects.ts +186 -0
  1917. package/src/vim/transitions.ts +490 -0
  1918. package/src/vim/types.ts +199 -0
  1919. package/src/voice/voiceModeEnabled.ts +54 -0
  1920. package/start.js +1 -0
@@ -0,0 +1,3348 @@
1
+ import { feature } from 'bun:bundle'
2
+ import type {
3
+ Base64ImageSource,
4
+ ContentBlockParam,
5
+ MessageParam,
6
+ } from '@anthropic-ai/sdk/resources/index.mjs'
7
+ import { Client } from '@modelcontextprotocol/sdk/client/index.js'
8
+ import {
9
+ SSEClientTransport,
10
+ type SSEClientTransportOptions,
11
+ } from '@modelcontextprotocol/sdk/client/sse.js'
12
+ import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'
13
+ import {
14
+ StreamableHTTPClientTransport,
15
+ type StreamableHTTPClientTransportOptions,
16
+ } from '@modelcontextprotocol/sdk/client/streamableHttp.js'
17
+ import {
18
+ createFetchWithInit,
19
+ type FetchLike,
20
+ type Transport,
21
+ } from '@modelcontextprotocol/sdk/shared/transport.js'
22
+ import {
23
+ CallToolResultSchema,
24
+ ElicitRequestSchema,
25
+ type ElicitRequestURLParams,
26
+ type ElicitResult,
27
+ ErrorCode,
28
+ type JSONRPCMessage,
29
+ type ListPromptsResult,
30
+ ListPromptsResultSchema,
31
+ ListResourcesResultSchema,
32
+ ListRootsRequestSchema,
33
+ type ListToolsResult,
34
+ ListToolsResultSchema,
35
+ McpError,
36
+ type PromptMessage,
37
+ type ResourceLink,
38
+ } from '@modelcontextprotocol/sdk/types.js'
39
+ import mapValues from 'lodash-es/mapValues.js'
40
+ import memoize from 'lodash-es/memoize.js'
41
+ import zipObject from 'lodash-es/zipObject.js'
42
+ import pMap from 'p-map'
43
+ import { getOriginalCwd, getSessionId } from '../../bootstrap/state.js'
44
+ import type { Command } from '../../commands.js'
45
+ import { getOauthConfig } from '../../constants/oauth.js'
46
+ import { PRODUCT_URL } from '../../constants/product.js'
47
+ import type { AppState } from '../../state/AppState.js'
48
+ import {
49
+ type Tool,
50
+ type ToolCallProgress,
51
+ toolMatchesName,
52
+ } from '../../Tool.js'
53
+ import { ListMcpResourcesTool } from '../../tools/ListMcpResourcesTool/ListMcpResourcesTool.js'
54
+ import { type MCPProgress, MCPTool } from '../../tools/MCPTool/MCPTool.js'
55
+ import { createMcpAuthTool } from '../../tools/McpAuthTool/McpAuthTool.js'
56
+ import { ReadMcpResourceTool } from '../../tools/ReadMcpResourceTool/ReadMcpResourceTool.js'
57
+ import { createAbortController } from '../../utils/abortController.js'
58
+ import { count } from '../../utils/array.js'
59
+ import {
60
+ checkAndRefreshOAuthTokenIfNeeded,
61
+ getClaudeAIOAuthTokens,
62
+ handleOAuth401Error,
63
+ } from '../../utils/auth.js'
64
+ import { registerCleanup } from '../../utils/cleanupRegistry.js'
65
+ import { detectCodeIndexingFromMcpServerName } from '../../utils/codeIndexing.js'
66
+ import { logForDebugging } from '../../utils/debug.js'
67
+ import { isEnvDefinedFalsy, isEnvTruthy } from '../../utils/envUtils.js'
68
+ import {
69
+ errorMessage,
70
+ TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
71
+ } from '../../utils/errors.js'
72
+ import { getMCPUserAgent } from '../../utils/http.js'
73
+ import { maybeNotifyIDEConnected } from '../../utils/ide.js'
74
+ import { maybeResizeAndDownsampleImageBuffer } from '../../utils/imageResizer.js'
75
+ import { logMCPDebug, logMCPError } from '../../utils/log.js'
76
+ import {
77
+ getBinaryBlobSavedMessage,
78
+ getFormatDescription,
79
+ getLargeOutputInstructions,
80
+ persistBinaryContent,
81
+ } from '../../utils/mcpOutputStorage.js'
82
+ import {
83
+ getContentSizeEstimate,
84
+ type MCPToolResult,
85
+ mcpContentNeedsTruncation,
86
+ truncateMcpContentIfNeeded,
87
+ } from '../../utils/mcpValidation.js'
88
+ import { WebSocketTransport } from '../../utils/mcpWebSocketTransport.js'
89
+ import { memoizeWithLRU } from '../../utils/memoize.js'
90
+ import { getWebSocketTLSOptions } from '../../utils/mtls.js'
91
+ import {
92
+ getProxyFetchOptions,
93
+ getWebSocketProxyAgent,
94
+ getWebSocketProxyUrl,
95
+ } from '../../utils/proxy.js'
96
+ import { recursivelySanitizeUnicode } from '../../utils/sanitization.js'
97
+ import { getSessionIngressAuthToken } from '../../utils/sessionIngressAuth.js'
98
+ import { subprocessEnv } from '../../utils/subprocessEnv.js'
99
+ import {
100
+ isPersistError,
101
+ persistToolResult,
102
+ } from '../../utils/toolResultStorage.js'
103
+ import {
104
+ type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
105
+ logEvent,
106
+ } from '../analytics/index.js'
107
+ import {
108
+ type ElicitationWaitingState,
109
+ runElicitationHooks,
110
+ runElicitationResultHooks,
111
+ } from './elicitationHandler.js'
112
+ import { buildMcpToolName } from './mcpStringUtils.js'
113
+ import { normalizeNameForMCP } from './normalization.js'
114
+ import { getLoggingSafeMcpBaseUrl } from './utils.js'
115
+
116
+ /* eslint-disable @typescript-eslint/no-require-imports */
117
+ const fetchMcpSkillsForClient = feature('MCP_SKILLS')
118
+ ? (
119
+ require('../../skills/mcpSkills.js') as typeof import('../../skills/mcpSkills.js')
120
+ ).fetchMcpSkillsForClient
121
+ : null
122
+
123
+ import { UnauthorizedError } from '@modelcontextprotocol/sdk/client/auth.js'
124
+ import type { AssistantMessage } from 'src/types/message.js'
125
+ /* eslint-enable @typescript-eslint/no-require-imports */
126
+ import { classifyMcpToolForCollapse } from '../../tools/MCPTool/classifyForCollapse.js'
127
+ import { clearKeychainCache } from '../../utils/secureStorage/macOsKeychainHelpers.js'
128
+ import { sleep } from '../../utils/sleep.js'
129
+ import {
130
+ ClaudeAuthProvider,
131
+ hasMcpDiscoveryButNoToken,
132
+ wrapFetchWithStepUpDetection,
133
+ } from './auth.js'
134
+ import { markClaudeAiMcpConnected } from './claudeai.js'
135
+ import { getAllMcpConfigs, isMcpServerDisabled } from './config.js'
136
+ import { getMcpServerHeaders } from './headersHelper.js'
137
+ import { SdkControlClientTransport } from './SdkControlTransport.js'
138
+ import type {
139
+ ConnectedMCPServer,
140
+ MCPServerConnection,
141
+ McpSdkServerConfig,
142
+ ScopedMcpServerConfig,
143
+ ServerResource,
144
+ } from './types.js'
145
+
146
+ /**
147
+ * Custom error class to indicate that an MCP tool call failed due to
148
+ * authentication issues (e.g., expired OAuth token returning 401).
149
+ * This error should be caught at the tool execution layer to update
150
+ * the client's status to 'needs-auth'.
151
+ */
152
+ export class McpAuthError extends Error {
153
+ serverName: string
154
+ constructor(serverName: string, message: string) {
155
+ super(message)
156
+ this.name = 'McpAuthError'
157
+ this.serverName = serverName
158
+ }
159
+ }
160
+
161
+ /**
162
+ * Thrown when an MCP session has expired and the connection cache has been cleared.
163
+ * The caller should get a fresh client via ensureConnectedClient and retry.
164
+ */
165
+ class McpSessionExpiredError extends Error {
166
+ constructor(serverName: string) {
167
+ super(`MCP server "${serverName}" session expired`)
168
+ this.name = 'McpSessionExpiredError'
169
+ }
170
+ }
171
+
172
+ /**
173
+ * Thrown when an MCP tool returns `isError: true`. Carries the result's `_meta`
174
+ * so SDK consumers can still receive it — per the MCP spec, `_meta` is on the
175
+ * base Result type and is valid on error results.
176
+ */
177
+ export class McpToolCallError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS extends TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {
178
+ constructor(
179
+ message: string,
180
+ telemetryMessage: string,
181
+ readonly mcpMeta?: { _meta?: Record<string, unknown> },
182
+ ) {
183
+ super(message, telemetryMessage)
184
+ this.name = 'McpToolCallError'
185
+ }
186
+ }
187
+
188
+ /**
189
+ * Detects whether an error is an MCP "Session not found" error (HTTP 404 + JSON-RPC code -32001).
190
+ * Per the MCP spec, servers return 404 when a session ID is no longer valid.
191
+ * We check both signals to avoid false positives from generic 404s (wrong URL, server gone, etc.).
192
+ */
193
+ export function isMcpSessionExpiredError(error: Error): boolean {
194
+ const httpStatus =
195
+ 'code' in error ? (error as Error & { code?: number }).code : undefined
196
+ if (httpStatus !== 404) {
197
+ return false
198
+ }
199
+ // The SDK embeds the response body text in the error message.
200
+ // MCP servers return: {"error":{"code":-32001,"message":"Session not found"},...}
201
+ // Check for the JSON-RPC error code to distinguish from generic web server 404s.
202
+ return (
203
+ error.message.includes('"code":-32001') ||
204
+ error.message.includes('"code": -32001')
205
+ )
206
+ }
207
+
208
+ /**
209
+ * Default timeout for MCP tool calls (effectively infinite - ~27.8 hours).
210
+ */
211
+ const DEFAULT_MCP_TOOL_TIMEOUT_MS = 100_000_000
212
+
213
+ /**
214
+ * Cap on MCP tool descriptions and server instructions sent to the model.
215
+ * OpenAPI-generated MCP servers have been observed dumping 15-60KB of endpoint
216
+ * docs into tool.description; this caps the p95 tail without losing the intent.
217
+ */
218
+ const MAX_MCP_DESCRIPTION_LENGTH = 2048
219
+
220
+ /**
221
+ * Gets the timeout for MCP tool calls in milliseconds.
222
+ * Uses MCP_TOOL_TIMEOUT environment variable if set, otherwise defaults to ~27.8 hours.
223
+ */
224
+ function getMcpToolTimeoutMs(): number {
225
+ return (
226
+ parseInt(process.env.MCP_TOOL_TIMEOUT || '', 10) ||
227
+ DEFAULT_MCP_TOOL_TIMEOUT_MS
228
+ )
229
+ }
230
+
231
+ import { isClaudeInChromeMCPServer } from '../../utils/claudeInChrome/common.js'
232
+
233
+ // Lazy: toolRendering.tsx pulls React/ink; only needed when Claude-in-Chrome MCP server is connected
234
+ /* eslint-disable @typescript-eslint/no-require-imports */
235
+ const claudeInChromeToolRendering =
236
+ (): typeof import('../../utils/claudeInChrome/toolRendering.js') =>
237
+ require('../../utils/claudeInChrome/toolRendering.js')
238
+ // Lazy: wrapper.tsx → hostAdapter.ts → executor.ts pulls both native modules
239
+ // (@ant/computer-use-input + @ant/computer-use-swift). Runtime-gated by
240
+ // GrowthBook tengu_malort_pedway (see gates.ts).
241
+ const computerUseWrapper = feature('CHICAGO_MCP')
242
+ ? (): typeof import('../../utils/computerUse/wrapper.js') =>
243
+ require('../../utils/computerUse/wrapper.js')
244
+ : undefined
245
+ const isComputerUseMCPServer = feature('CHICAGO_MCP')
246
+ ? (
247
+ require('../../utils/computerUse/common.js') as typeof import('../../utils/computerUse/common.js')
248
+ ).isComputerUseMCPServer
249
+ : undefined
250
+
251
+ import { mkdir, readFile, unlink, writeFile } from 'fs/promises'
252
+ import { dirname, join } from 'path'
253
+ import { getClaudeConfigHomeDir } from '../../utils/envUtils.js'
254
+ /* eslint-enable @typescript-eslint/no-require-imports */
255
+ import { jsonParse, jsonStringify } from '../../utils/slowOperations.js'
256
+
257
+ const MCP_AUTH_CACHE_TTL_MS = 15 * 60 * 1000 // 15 min
258
+
259
+ type McpAuthCacheData = Record<string, { timestamp: number }>
260
+
261
+ function getMcpAuthCachePath(): string {
262
+ return join(getClaudeConfigHomeDir(), 'mcp-needs-auth-cache.json')
263
+ }
264
+
265
+ // Memoized so N concurrent isMcpAuthCached() calls during batched connection
266
+ // share a single file read instead of N reads of the same file. Invalidated
267
+ // on write (setMcpAuthCacheEntry) and clear (clearMcpAuthCache). Not using
268
+ // lodash memoize because we need to null out the cache, not delete by key.
269
+ let authCachePromise: Promise<McpAuthCacheData> | null = null
270
+
271
+ function getMcpAuthCache(): Promise<McpAuthCacheData> {
272
+ if (!authCachePromise) {
273
+ authCachePromise = readFile(getMcpAuthCachePath(), 'utf-8')
274
+ .then(data => jsonParse(data) as McpAuthCacheData)
275
+ .catch(() => ({}))
276
+ }
277
+ return authCachePromise
278
+ }
279
+
280
+ async function isMcpAuthCached(serverId: string): Promise<boolean> {
281
+ const cache = await getMcpAuthCache()
282
+ const entry = cache[serverId]
283
+ if (!entry) {
284
+ return false
285
+ }
286
+ return Date.now() - entry.timestamp < MCP_AUTH_CACHE_TTL_MS
287
+ }
288
+
289
+ // Serialize cache writes through a promise chain to prevent concurrent
290
+ // read-modify-write races when multiple servers return 401 in the same batch
291
+ let writeChain = Promise.resolve()
292
+
293
+ function setMcpAuthCacheEntry(serverId: string): void {
294
+ writeChain = writeChain
295
+ .then(async () => {
296
+ const cache = await getMcpAuthCache()
297
+ cache[serverId] = { timestamp: Date.now() }
298
+ const cachePath = getMcpAuthCachePath()
299
+ await mkdir(dirname(cachePath), { recursive: true })
300
+ await writeFile(cachePath, jsonStringify(cache))
301
+ // Invalidate the read cache so subsequent reads see the new entry.
302
+ // Safe because writeChain serializes writes: the next write's
303
+ // getMcpAuthCache() call will re-read the file with this entry present.
304
+ authCachePromise = null
305
+ })
306
+ .catch(() => {
307
+ // Best-effort cache write
308
+ })
309
+ }
310
+
311
+ export function clearMcpAuthCache(): void {
312
+ authCachePromise = null
313
+ void unlink(getMcpAuthCachePath()).catch(() => {
314
+ // Cache file may not exist
315
+ })
316
+ }
317
+
318
+ /**
319
+ * Spread-ready analytics field for the server's base URL. Calls
320
+ * getLoggingSafeMcpBaseUrl once (not twice like the inline ternary it replaces).
321
+ * Typed as AnalyticsMetadata since the URL is query-stripped and safe to log.
322
+ */
323
+ function mcpBaseUrlAnalytics(serverRef: ScopedMcpServerConfig): {
324
+ mcpServerBaseUrl?: AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
325
+ } {
326
+ const url = getLoggingSafeMcpBaseUrl(serverRef)
327
+ return url
328
+ ? {
329
+ mcpServerBaseUrl:
330
+ url as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
331
+ }
332
+ : {}
333
+ }
334
+
335
+ /**
336
+ * Shared handler for sse/http/claudeai-proxy auth failures during connect:
337
+ * emits tengu_mcp_server_needs_auth, caches the needs-auth entry, and returns
338
+ * the needs-auth connection result.
339
+ */
340
+ function handleRemoteAuthFailure(
341
+ name: string,
342
+ serverRef: ScopedMcpServerConfig,
343
+ transportType: 'sse' | 'http' | 'claudeai-proxy',
344
+ ): MCPServerConnection {
345
+ logEvent('tengu_mcp_server_needs_auth', {
346
+ transportType:
347
+ transportType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
348
+ ...mcpBaseUrlAnalytics(serverRef),
349
+ })
350
+ const label: Record<typeof transportType, string> = {
351
+ sse: 'SSE',
352
+ http: 'HTTP',
353
+ 'claudeai-proxy': 'claude.ai proxy',
354
+ }
355
+ logMCPDebug(
356
+ name,
357
+ `Authentication required for ${label[transportType]} server`,
358
+ )
359
+ setMcpAuthCacheEntry(name)
360
+ return { name, type: 'needs-auth', config: serverRef }
361
+ }
362
+
363
+ /**
364
+ * Fetch wrapper for claude.ai proxy connections. Attaches the OAuth bearer
365
+ * token and retries once on 401 via handleOAuth401Error (force-refresh).
366
+ *
367
+ * The Anthropic API path has this retry (withRetry.ts, grove.ts) to handle
368
+ * memoize-cache staleness and clock drift. Without the same here, a single
369
+ * stale token mass-401s every claude.ai connector and sticks them all in the
370
+ * 15-min needs-auth cache.
371
+ */
372
+ export function createClaudeAiProxyFetch(innerFetch: FetchLike): FetchLike {
373
+ return async (url, init) => {
374
+ const doRequest = async () => {
375
+ await checkAndRefreshOAuthTokenIfNeeded()
376
+ const currentTokens = getClaudeAIOAuthTokens()
377
+ if (!currentTokens) {
378
+ throw new Error('No claude.ai OAuth token available')
379
+ }
380
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
381
+ const headers = new Headers(init?.headers)
382
+ headers.set('Authorization', `Bearer ${currentTokens.accessToken}`)
383
+ const response = await innerFetch(url, { ...init, headers })
384
+ // Return the exact token that was sent. Reading getClaudeAIOAuthTokens()
385
+ // again after the request is wrong under concurrent 401s: another
386
+ // connector's handleOAuth401Error clears the memoize cache, so we'd read
387
+ // the NEW token from keychain, pass it to handleOAuth401Error, which
388
+ // finds same-as-keychain → returns false → skips retry. Same pattern as
389
+ // bridgeApi.ts withOAuthRetry (token passed as fn param).
390
+ return { response, sentToken: currentTokens.accessToken }
391
+ }
392
+
393
+ const { response, sentToken } = await doRequest()
394
+ if (response.status !== 401) {
395
+ return response
396
+ }
397
+ // handleOAuth401Error returns true only if the token actually changed
398
+ // (keychain had a newer one, or force-refresh succeeded). Gate retry on
399
+ // that — otherwise we double round-trip time for every connector whose
400
+ // downstream service genuinely needs auth (the common case: 30+ servers
401
+ // with "MCP server requires authentication but no OAuth token configured").
402
+ const tokenChanged = await handleOAuth401Error(sentToken).catch(() => false)
403
+ logEvent('tengu_mcp_claudeai_proxy_401', {
404
+ tokenChanged:
405
+ tokenChanged as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
406
+ })
407
+ if (!tokenChanged) {
408
+ // ELOCKED contention: another connector may have won the lockfile and refreshed — check if token changed underneath us
409
+ const now = getClaudeAIOAuthTokens()?.accessToken
410
+ if (!now || now === sentToken) {
411
+ return response
412
+ }
413
+ }
414
+ try {
415
+ return (await doRequest()).response
416
+ } catch {
417
+ // Retry itself failed (network error). Return the original 401 so the
418
+ // outer handler can classify it.
419
+ return response
420
+ }
421
+ }
422
+ }
423
+
424
+ // Minimal interface for WebSocket instances passed to mcpWebSocketTransport
425
+ type WsClientLike = {
426
+ readonly readyState: number
427
+ close(): void
428
+ send(data: string): void
429
+ }
430
+
431
+ /**
432
+ * Create a ws.WebSocket client with the MCP protocol.
433
+ * Bun's ws shim types lack the 3-arg constructor (url, protocols, options)
434
+ * that the real ws package supports, so we cast the constructor here.
435
+ */
436
+ async function createNodeWsClient(
437
+ url: string,
438
+ options: Record<string, unknown>,
439
+ ): Promise<WsClientLike> {
440
+ const wsModule = await import('ws')
441
+ const WS = wsModule.default as unknown as new (
442
+ url: string,
443
+ protocols: string[],
444
+ options: Record<string, unknown>,
445
+ ) => WsClientLike
446
+ return new WS(url, ['mcp'], options)
447
+ }
448
+
449
+ const IMAGE_MIME_TYPES = new Set([
450
+ 'image/jpeg',
451
+ 'image/png',
452
+ 'image/gif',
453
+ 'image/webp',
454
+ ])
455
+
456
+ function getConnectionTimeoutMs(): number {
457
+ return parseInt(process.env.MCP_TIMEOUT || '', 10) || 30000
458
+ }
459
+
460
+ /**
461
+ * Default timeout for individual MCP requests (auth, tool calls, etc.)
462
+ */
463
+ const MCP_REQUEST_TIMEOUT_MS = 60000
464
+
465
+ /**
466
+ * MCP Streamable HTTP spec requires clients to advertise acceptance of both
467
+ * JSON and SSE on every POST. Servers that enforce this strictly reject
468
+ * requests without it (HTTP 406).
469
+ * https://modelcontextprotocol.io/specification/2025-03-26/basic/transports#sending-messages-to-the-server
470
+ */
471
+ const MCP_STREAMABLE_HTTP_ACCEPT = 'application/json, text/event-stream'
472
+
473
+ /**
474
+ * Wraps a fetch function to apply a fresh timeout signal to each request.
475
+ * This avoids the bug where a single AbortSignal.timeout() created at connection
476
+ * time becomes stale after 60 seconds, causing all subsequent requests to fail
477
+ * immediately with "The operation timed out." Uses a 60-second timeout.
478
+ *
479
+ * Also ensures the Accept header required by the MCP Streamable HTTP spec is
480
+ * present on POSTs. The MCP SDK sets this inside StreamableHTTPClientTransport.send(),
481
+ * but it is attached to a Headers instance that passes through an object spread here,
482
+ * and some runtimes/agents have been observed dropping it before it reaches the wire.
483
+ * See https://github.com/anthropics/claude-agent-sdk-typescript/issues/202.
484
+ * Normalizing here (the last wrapper before fetch()) guarantees it is sent.
485
+ *
486
+ * GET requests are excluded from the timeout since, for MCP transports, they are
487
+ * long-lived SSE streams meant to stay open indefinitely. (Auth-related GETs use
488
+ * a separate fetch wrapper with its own timeout in auth.ts.)
489
+ *
490
+ * @param baseFetch - The fetch function to wrap
491
+ */
492
+ export function wrapFetchWithTimeout(baseFetch: FetchLike): FetchLike {
493
+ return async (url: string | URL, init?: RequestInit) => {
494
+ const method = (init?.method ?? 'GET').toUpperCase()
495
+
496
+ // Skip timeout for GET requests - in MCP transports, these are long-lived SSE streams.
497
+ // (OAuth discovery GETs in auth.ts use a separate createAuthFetch() with its own timeout.)
498
+ if (method === 'GET') {
499
+ return baseFetch(url, init)
500
+ }
501
+
502
+ // Normalize headers and guarantee the Streamable-HTTP Accept value. new Headers()
503
+ // accepts HeadersInit | undefined and copies from plain objects, tuple arrays,
504
+ // and existing Headers instances — so whatever shape the SDK handed us, the
505
+ // Accept value survives the spread below as an own property of a concrete object.
506
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
507
+ const headers = new Headers(init?.headers)
508
+ if (!headers.has('accept')) {
509
+ headers.set('accept', MCP_STREAMABLE_HTTP_ACCEPT)
510
+ }
511
+
512
+ // Use setTimeout instead of AbortSignal.timeout() so we can clearTimeout on
513
+ // completion. AbortSignal.timeout's internal timer is only released when the
514
+ // signal is GC'd, which in Bun is lazy — ~2.4KB of native memory per request
515
+ // lingers for the full 60s even when the request completes in milliseconds.
516
+ const controller = new AbortController()
517
+ const timer = setTimeout(
518
+ c =>
519
+ c.abort(new DOMException('The operation timed out.', 'TimeoutError')),
520
+ MCP_REQUEST_TIMEOUT_MS,
521
+ controller,
522
+ )
523
+ timer.unref?.()
524
+
525
+ const parentSignal = init?.signal
526
+ const abort = () => controller.abort(parentSignal?.reason)
527
+ parentSignal?.addEventListener('abort', abort)
528
+ if (parentSignal?.aborted) {
529
+ controller.abort(parentSignal.reason)
530
+ }
531
+
532
+ const cleanup = () => {
533
+ clearTimeout(timer)
534
+ parentSignal?.removeEventListener('abort', abort)
535
+ }
536
+
537
+ try {
538
+ const response = await baseFetch(url, {
539
+ ...init,
540
+ headers,
541
+ signal: controller.signal,
542
+ })
543
+ cleanup()
544
+ return response
545
+ } catch (error) {
546
+ cleanup()
547
+ throw error
548
+ }
549
+ }
550
+ }
551
+
552
+ export function getMcpServerConnectionBatchSize(): number {
553
+ return parseInt(process.env.MCP_SERVER_CONNECTION_BATCH_SIZE || '', 10) || 3
554
+ }
555
+
556
+ function getRemoteMcpServerConnectionBatchSize(): number {
557
+ return (
558
+ parseInt(process.env.MCP_REMOTE_SERVER_CONNECTION_BATCH_SIZE || '', 10) ||
559
+ 20
560
+ )
561
+ }
562
+
563
+ function isLocalMcpServer(config: ScopedMcpServerConfig): boolean {
564
+ return !config.type || config.type === 'stdio' || config.type === 'sdk'
565
+ }
566
+
567
+ // For the IDE MCP servers, we only include specific tools
568
+ const ALLOWED_IDE_TOOLS = ['mcp__ide__executeCode', 'mcp__ide__getDiagnostics']
569
+ function isIncludedMcpTool(tool: Tool): boolean {
570
+ return (
571
+ !tool.name.startsWith('mcp__ide__') || ALLOWED_IDE_TOOLS.includes(tool.name)
572
+ )
573
+ }
574
+
575
+ /**
576
+ * Generates the cache key for a server connection
577
+ * @param name Server name
578
+ * @param serverRef Server configuration
579
+ * @returns Cache key string
580
+ */
581
+ export function getServerCacheKey(
582
+ name: string,
583
+ serverRef: ScopedMcpServerConfig,
584
+ ): string {
585
+ return `${name}-${jsonStringify(serverRef)}`
586
+ }
587
+
588
+ /**
589
+ * TODO (ollie): The memoization here increases complexity by a lot, and im not sure it really improves performance
590
+ * Attempts to connect to a single MCP server
591
+ * @param name Server name
592
+ * @param serverRef Scoped server configuration
593
+ * @returns A wrapped client (either connected or failed)
594
+ */
595
+ export const connectToServer = memoize(
596
+ async (
597
+ name: string,
598
+ serverRef: ScopedMcpServerConfig,
599
+ serverStats?: {
600
+ totalServers: number
601
+ stdioCount: number
602
+ sseCount: number
603
+ httpCount: number
604
+ sseIdeCount: number
605
+ wsIdeCount: number
606
+ },
607
+ ): Promise<MCPServerConnection> => {
608
+ const connectStartTime = Date.now()
609
+ let inProcessServer:
610
+ | { connect(t: Transport): Promise<void>; close(): Promise<void> }
611
+ | undefined
612
+ try {
613
+ let transport
614
+
615
+ // If we have the session ingress JWT, we will connect via the session ingress rather than
616
+ // to remote MCP's directly.
617
+ const sessionIngressToken = getSessionIngressAuthToken()
618
+
619
+ if (serverRef.type === 'sse') {
620
+ // Create an auth provider for this server
621
+ const authProvider = new ClaudeAuthProvider(name, serverRef)
622
+
623
+ // Get combined headers (static + dynamic)
624
+ const combinedHeaders = await getMcpServerHeaders(name, serverRef)
625
+
626
+ // Use the auth provider with SSEClientTransport
627
+ const transportOptions: SSEClientTransportOptions = {
628
+ authProvider,
629
+ // Use fresh timeout per request to avoid stale AbortSignal bug.
630
+ // Step-up detection wraps innermost so the 403 is seen before the
631
+ // SDK's handler calls auth() → tokens().
632
+ fetch: wrapFetchWithTimeout(
633
+ wrapFetchWithStepUpDetection(createFetchWithInit(), authProvider),
634
+ ),
635
+ requestInit: {
636
+ headers: {
637
+ 'User-Agent': getMCPUserAgent(),
638
+ ...combinedHeaders,
639
+ },
640
+ },
641
+ }
642
+
643
+ // IMPORTANT: Always set eventSourceInit with a fetch that does NOT use the
644
+ // timeout wrapper. The EventSource connection is long-lived (stays open indefinitely
645
+ // to receive server-sent events), so applying a 60-second timeout would kill it.
646
+ // The timeout is only meant for individual API requests (POST, auth refresh), not
647
+ // the persistent SSE stream.
648
+ transportOptions.eventSourceInit = {
649
+ fetch: async (url: string | URL, init?: RequestInit) => {
650
+ // Get auth headers from the auth provider
651
+ const authHeaders: Record<string, string> = {}
652
+ const tokens = await authProvider.tokens()
653
+ if (tokens) {
654
+ authHeaders.Authorization = `Bearer ${tokens.access_token}`
655
+ }
656
+
657
+ const proxyOptions = getProxyFetchOptions()
658
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
659
+ return fetch(url, {
660
+ ...init,
661
+ ...proxyOptions,
662
+ headers: {
663
+ 'User-Agent': getMCPUserAgent(),
664
+ ...authHeaders,
665
+ ...init?.headers,
666
+ ...combinedHeaders,
667
+ Accept: 'text/event-stream',
668
+ },
669
+ })
670
+ },
671
+ }
672
+
673
+ transport = new SSEClientTransport(
674
+ new URL(serverRef.url),
675
+ transportOptions,
676
+ )
677
+ logMCPDebug(name, `SSE transport initialized, awaiting connection`)
678
+ } else if (serverRef.type === 'sse-ide') {
679
+ logMCPDebug(name, `Setting up SSE-IDE transport to ${serverRef.url}`)
680
+ // IDE servers don't need authentication
681
+ // TODO: Use the auth token provided in the lockfile
682
+ const proxyOptions = getProxyFetchOptions()
683
+ const transportOptions: SSEClientTransportOptions =
684
+ proxyOptions.dispatcher
685
+ ? {
686
+ eventSourceInit: {
687
+ fetch: async (url: string | URL, init?: RequestInit) => {
688
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
689
+ return fetch(url, {
690
+ ...init,
691
+ ...proxyOptions,
692
+ headers: {
693
+ 'User-Agent': getMCPUserAgent(),
694
+ ...init?.headers,
695
+ },
696
+ })
697
+ },
698
+ },
699
+ }
700
+ : {}
701
+
702
+ transport = new SSEClientTransport(
703
+ new URL(serverRef.url),
704
+ Object.keys(transportOptions).length > 0
705
+ ? transportOptions
706
+ : undefined,
707
+ )
708
+ } else if (serverRef.type === 'ws-ide') {
709
+ const tlsOptions = getWebSocketTLSOptions()
710
+ const wsHeaders = {
711
+ 'User-Agent': getMCPUserAgent(),
712
+ ...(serverRef.authToken && {
713
+ 'X-Claude-Code-Ide-Authorization': serverRef.authToken,
714
+ }),
715
+ }
716
+
717
+ let wsClient: WsClientLike
718
+ if (typeof Bun !== 'undefined') {
719
+ // Bun's WebSocket supports headers/proxy/tls options but the DOM typings don't
720
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
721
+ wsClient = new globalThis.WebSocket(serverRef.url, {
722
+ protocols: ['mcp'],
723
+ headers: wsHeaders,
724
+ proxy: getWebSocketProxyUrl(serverRef.url),
725
+ tls: tlsOptions || undefined,
726
+ } as unknown as string[])
727
+ } else {
728
+ wsClient = await createNodeWsClient(serverRef.url, {
729
+ headers: wsHeaders,
730
+ agent: getWebSocketProxyAgent(serverRef.url),
731
+ ...(tlsOptions || {}),
732
+ })
733
+ }
734
+ transport = new WebSocketTransport(wsClient)
735
+ } else if (serverRef.type === 'ws') {
736
+ logMCPDebug(
737
+ name,
738
+ `Initializing WebSocket transport to ${serverRef.url}`,
739
+ )
740
+
741
+ const combinedHeaders = await getMcpServerHeaders(name, serverRef)
742
+
743
+ const tlsOptions = getWebSocketTLSOptions()
744
+ const wsHeaders = {
745
+ 'User-Agent': getMCPUserAgent(),
746
+ ...(sessionIngressToken && {
747
+ Authorization: `Bearer ${sessionIngressToken}`,
748
+ }),
749
+ ...combinedHeaders,
750
+ }
751
+
752
+ // Redact sensitive headers before logging
753
+ const wsHeadersForLogging = mapValues(wsHeaders, (value, key) =>
754
+ key.toLowerCase() === 'authorization' ? '[REDACTED]' : value,
755
+ )
756
+
757
+ logMCPDebug(
758
+ name,
759
+ `WebSocket transport options: ${jsonStringify({
760
+ url: serverRef.url,
761
+ headers: wsHeadersForLogging,
762
+ hasSessionAuth: !!sessionIngressToken,
763
+ })}`,
764
+ )
765
+
766
+ let wsClient: WsClientLike
767
+ if (typeof Bun !== 'undefined') {
768
+ // Bun's WebSocket supports headers/proxy/tls options but the DOM typings don't
769
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
770
+ wsClient = new globalThis.WebSocket(serverRef.url, {
771
+ protocols: ['mcp'],
772
+ headers: wsHeaders,
773
+ proxy: getWebSocketProxyUrl(serverRef.url),
774
+ tls: tlsOptions || undefined,
775
+ } as unknown as string[])
776
+ } else {
777
+ wsClient = await createNodeWsClient(serverRef.url, {
778
+ headers: wsHeaders,
779
+ agent: getWebSocketProxyAgent(serverRef.url),
780
+ ...(tlsOptions || {}),
781
+ })
782
+ }
783
+ transport = new WebSocketTransport(wsClient)
784
+ } else if (serverRef.type === 'http') {
785
+ logMCPDebug(name, `Initializing HTTP transport to ${serverRef.url}`)
786
+ logMCPDebug(
787
+ name,
788
+ `Node version: ${process.version}, Platform: ${process.platform}`,
789
+ )
790
+ logMCPDebug(
791
+ name,
792
+ `Environment: ${jsonStringify({
793
+ NODE_OPTIONS: process.env.NODE_OPTIONS || 'not set',
794
+ UV_THREADPOOL_SIZE: process.env.UV_THREADPOOL_SIZE || 'default',
795
+ HTTP_PROXY: process.env.HTTP_PROXY || 'not set',
796
+ HTTPS_PROXY: process.env.HTTPS_PROXY || 'not set',
797
+ NO_PROXY: process.env.NO_PROXY || 'not set',
798
+ })}`,
799
+ )
800
+
801
+ // Create an auth provider for this server
802
+ const authProvider = new ClaudeAuthProvider(name, serverRef)
803
+
804
+ // Get combined headers (static + dynamic)
805
+ const combinedHeaders = await getMcpServerHeaders(name, serverRef)
806
+
807
+ // Check if this server has stored OAuth tokens. If so, the SDK's
808
+ // authProvider will set Authorization — don't override with the
809
+ // session ingress token (SDK merges requestInit AFTER authProvider).
810
+ // CCR proxy URLs (ccr_shttp_mcp) have no stored OAuth, so they still
811
+ // get the ingress token. See PR #24454 discussion.
812
+ const hasOAuthTokens = !!(await authProvider.tokens())
813
+
814
+ // Use the auth provider with StreamableHTTPClientTransport
815
+ const proxyOptions = getProxyFetchOptions()
816
+ logMCPDebug(
817
+ name,
818
+ `Proxy options: ${proxyOptions.dispatcher ? 'custom dispatcher' : 'default'}`,
819
+ )
820
+
821
+ const transportOptions: StreamableHTTPClientTransportOptions = {
822
+ authProvider,
823
+ // Use fresh timeout per request to avoid stale AbortSignal bug.
824
+ // Step-up detection wraps innermost so the 403 is seen before the
825
+ // SDK's handler calls auth() → tokens().
826
+ fetch: wrapFetchWithTimeout(
827
+ wrapFetchWithStepUpDetection(createFetchWithInit(), authProvider),
828
+ ),
829
+ requestInit: {
830
+ ...proxyOptions,
831
+ headers: {
832
+ 'User-Agent': getMCPUserAgent(),
833
+ ...(sessionIngressToken &&
834
+ !hasOAuthTokens && {
835
+ Authorization: `Bearer ${sessionIngressToken}`,
836
+ }),
837
+ ...combinedHeaders,
838
+ },
839
+ },
840
+ }
841
+
842
+ // Redact sensitive headers before logging
843
+ const headersForLogging = transportOptions.requestInit?.headers
844
+ ? mapValues(
845
+ transportOptions.requestInit.headers as Record<string, string>,
846
+ (value, key) =>
847
+ key.toLowerCase() === 'authorization' ? '[REDACTED]' : value,
848
+ )
849
+ : undefined
850
+
851
+ logMCPDebug(
852
+ name,
853
+ `HTTP transport options: ${jsonStringify({
854
+ url: serverRef.url,
855
+ headers: headersForLogging,
856
+ hasAuthProvider: !!authProvider,
857
+ timeoutMs: MCP_REQUEST_TIMEOUT_MS,
858
+ })}`,
859
+ )
860
+
861
+ transport = new StreamableHTTPClientTransport(
862
+ new URL(serverRef.url),
863
+ transportOptions,
864
+ )
865
+ logMCPDebug(name, `HTTP transport created successfully`)
866
+ } else if (serverRef.type === 'sdk') {
867
+ throw new Error('SDK servers should be handled in print.ts')
868
+ } else if (serverRef.type === 'claudeai-proxy') {
869
+ logMCPDebug(
870
+ name,
871
+ `Initializing claude.ai proxy transport for server ${serverRef.id}`,
872
+ )
873
+
874
+ const tokens = getClaudeAIOAuthTokens()
875
+ if (!tokens) {
876
+ throw new Error('No claude.ai OAuth token found')
877
+ }
878
+
879
+ const oauthConfig = getOauthConfig()
880
+ const proxyUrl = `${oauthConfig.MCP_PROXY_URL}${oauthConfig.MCP_PROXY_PATH.replace('{server_id}', serverRef.id)}`
881
+
882
+ logMCPDebug(name, `Using claude.ai proxy at ${proxyUrl}`)
883
+
884
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
885
+ const fetchWithAuth = createClaudeAiProxyFetch(globalThis.fetch)
886
+
887
+ const proxyOptions = getProxyFetchOptions()
888
+ const transportOptions: StreamableHTTPClientTransportOptions = {
889
+ // Wrap fetchWithAuth with fresh timeout per request
890
+ fetch: wrapFetchWithTimeout(fetchWithAuth),
891
+ requestInit: {
892
+ ...proxyOptions,
893
+ headers: {
894
+ 'User-Agent': getMCPUserAgent(),
895
+ 'X-Mcp-Client-Session-Id': getSessionId(),
896
+ },
897
+ },
898
+ }
899
+
900
+ transport = new StreamableHTTPClientTransport(
901
+ new URL(proxyUrl),
902
+ transportOptions,
903
+ )
904
+ logMCPDebug(name, `claude.ai proxy transport created successfully`)
905
+ } else if (
906
+ (serverRef.type === 'stdio' || !serverRef.type) &&
907
+ isClaudeInChromeMCPServer(name)
908
+ ) {
909
+ // Run the Chrome MCP server in-process to avoid spawning a ~325 MB subprocess
910
+ const { createChromeContext } = await import(
911
+ '../../utils/claudeInChrome/mcpServer.js'
912
+ )
913
+ const { createClaudeForChromeMcpServer } = await import(
914
+ '@ant/claude-for-chrome-mcp'
915
+ )
916
+ const { createLinkedTransportPair } = await import(
917
+ './InProcessTransport.js'
918
+ )
919
+ const context = createChromeContext(serverRef.env)
920
+ inProcessServer = createClaudeForChromeMcpServer(context)
921
+ const [clientTransport, serverTransport] = createLinkedTransportPair()
922
+ await inProcessServer.connect(serverTransport)
923
+ transport = clientTransport
924
+ logMCPDebug(name, `In-process Chrome MCP server started`)
925
+ } else if (
926
+ feature('CHICAGO_MCP') &&
927
+ (serverRef.type === 'stdio' || !serverRef.type) &&
928
+ isComputerUseMCPServer!(name)
929
+ ) {
930
+ // Run the Computer Use MCP server in-process — same rationale as
931
+ // Chrome above. The package's CallTool handler is a stub; real
932
+ // dispatch goes through wrapper.tsx's .call() override.
933
+ const { createComputerUseMcpServerForCli } = await import(
934
+ '../../utils/computerUse/mcpServer.js'
935
+ )
936
+ const { createLinkedTransportPair } = await import(
937
+ './InProcessTransport.js'
938
+ )
939
+ inProcessServer = await createComputerUseMcpServerForCli()
940
+ const [clientTransport, serverTransport] = createLinkedTransportPair()
941
+ await inProcessServer.connect(serverTransport)
942
+ transport = clientTransport
943
+ logMCPDebug(name, `In-process Computer Use MCP server started`)
944
+ } else if (serverRef.type === 'stdio' || !serverRef.type) {
945
+ const finalCommand =
946
+ process.env.CLAUDE_CODE_SHELL_PREFIX || serverRef.command
947
+ const finalArgs = process.env.CLAUDE_CODE_SHELL_PREFIX
948
+ ? [[serverRef.command, ...serverRef.args].join(' ')]
949
+ : serverRef.args
950
+ transport = new StdioClientTransport({
951
+ command: finalCommand,
952
+ args: finalArgs,
953
+ env: {
954
+ ...subprocessEnv(),
955
+ ...serverRef.env,
956
+ } as Record<string, string>,
957
+ stderr: 'pipe', // prevents error output from the MCP server from printing to the UI
958
+ })
959
+ } else {
960
+ throw new Error(`Unsupported server type: ${serverRef.type}`)
961
+ }
962
+
963
+ // Set up stderr logging for stdio transport before connecting in case there are any stderr
964
+ // outputs emitted during the connection start (this can be useful for debugging failed connections).
965
+ // Store handler reference for cleanup to prevent memory leaks
966
+ let stderrHandler: ((data: Buffer) => void) | undefined
967
+ let stderrOutput = ''
968
+ if (serverRef.type === 'stdio' || !serverRef.type) {
969
+ const stdioTransport = transport as StdioClientTransport
970
+ if (stdioTransport.stderr) {
971
+ stderrHandler = (data: Buffer) => {
972
+ // Cap stderr accumulation to prevent unbounded memory growth
973
+ if (stderrOutput.length < 64 * 1024 * 1024) {
974
+ try {
975
+ stderrOutput += data.toString()
976
+ } catch {
977
+ // Ignore errors from exceeding max string length
978
+ }
979
+ }
980
+ }
981
+ stdioTransport.stderr.on('data', stderrHandler)
982
+ }
983
+ }
984
+
985
+ const client = new Client(
986
+ {
987
+ name: 'claude-code',
988
+ title: 'Claude Code',
989
+ version: MACRO.VERSION ?? 'unknown',
990
+ description: "Anthropic's agentic coding tool",
991
+ websiteUrl: PRODUCT_URL,
992
+ },
993
+ {
994
+ capabilities: {
995
+ roots: {},
996
+ // Empty object declares the capability. Sending {form:{},url:{}}
997
+ // breaks Java MCP SDK servers (Spring AI) whose Elicitation class
998
+ // has zero fields and fails on unknown properties.
999
+ elicitation: {},
1000
+ },
1001
+ },
1002
+ )
1003
+
1004
+ // Add debug logging for client events if available
1005
+ if (serverRef.type === 'http') {
1006
+ logMCPDebug(name, `Client created, setting up request handler`)
1007
+ }
1008
+
1009
+ client.setRequestHandler(ListRootsRequestSchema, async () => {
1010
+ logMCPDebug(name, `Received ListRoots request from server`)
1011
+ return {
1012
+ roots: [
1013
+ {
1014
+ uri: `file://${getOriginalCwd()}`,
1015
+ },
1016
+ ],
1017
+ }
1018
+ })
1019
+
1020
+ // Add a timeout to connection attempts to prevent tests from hanging indefinitely
1021
+ logMCPDebug(
1022
+ name,
1023
+ `Starting connection with timeout of ${getConnectionTimeoutMs()}ms`,
1024
+ )
1025
+
1026
+ // For HTTP transport, try a basic connectivity test first
1027
+ if (serverRef.type === 'http') {
1028
+ logMCPDebug(name, `Testing basic HTTP connectivity to ${serverRef.url}`)
1029
+ try {
1030
+ const testUrl = new URL(serverRef.url)
1031
+ logMCPDebug(
1032
+ name,
1033
+ `Parsed URL: host=${testUrl.hostname}, port=${testUrl.port || 'default'}, protocol=${testUrl.protocol}`,
1034
+ )
1035
+
1036
+ // Log DNS resolution attempt
1037
+ if (
1038
+ testUrl.hostname === '127.0.0.1' ||
1039
+ testUrl.hostname === 'localhost'
1040
+ ) {
1041
+ logMCPDebug(name, `Using loopback address: ${testUrl.hostname}`)
1042
+ }
1043
+ } catch (urlError) {
1044
+ logMCPDebug(name, `Failed to parse URL: ${urlError}`)
1045
+ }
1046
+ }
1047
+
1048
+ const connectPromise = client.connect(transport)
1049
+ const timeoutPromise = new Promise<never>((_, reject) => {
1050
+ const timeoutId = setTimeout(() => {
1051
+ const elapsed = Date.now() - connectStartTime
1052
+ logMCPDebug(
1053
+ name,
1054
+ `Connection timeout triggered after ${elapsed}ms (limit: ${getConnectionTimeoutMs()}ms)`,
1055
+ )
1056
+ if (inProcessServer) {
1057
+ inProcessServer.close().catch(() => {})
1058
+ }
1059
+ transport.close().catch(() => {})
1060
+ reject(
1061
+ new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
1062
+ `MCP server "${name}" connection timed out after ${getConnectionTimeoutMs()}ms`,
1063
+ 'MCP connection timeout',
1064
+ ),
1065
+ )
1066
+ }, getConnectionTimeoutMs())
1067
+
1068
+ // Clean up timeout if connect resolves or rejects
1069
+ connectPromise.then(
1070
+ () => {
1071
+ clearTimeout(timeoutId)
1072
+ },
1073
+ _error => {
1074
+ clearTimeout(timeoutId)
1075
+ },
1076
+ )
1077
+ })
1078
+
1079
+ try {
1080
+ await Promise.race([connectPromise, timeoutPromise])
1081
+ if (stderrOutput) {
1082
+ logMCPError(name, `Server stderr: ${stderrOutput}`)
1083
+ stderrOutput = '' // Release accumulated string to prevent memory growth
1084
+ }
1085
+ const elapsed = Date.now() - connectStartTime
1086
+ logMCPDebug(
1087
+ name,
1088
+ `Successfully connected (transport: ${serverRef.type || 'stdio'}) in ${elapsed}ms`,
1089
+ )
1090
+ } catch (error) {
1091
+ const elapsed = Date.now() - connectStartTime
1092
+ // SSE-specific error logging
1093
+ if (serverRef.type === 'sse' && error instanceof Error) {
1094
+ logMCPDebug(
1095
+ name,
1096
+ `SSE Connection failed after ${elapsed}ms: ${jsonStringify({
1097
+ url: serverRef.url,
1098
+ error: error.message,
1099
+ errorType: error.constructor.name,
1100
+ stack: error.stack,
1101
+ })}`,
1102
+ )
1103
+ logMCPError(name, error)
1104
+
1105
+ if (error instanceof UnauthorizedError) {
1106
+ return handleRemoteAuthFailure(name, serverRef, 'sse')
1107
+ }
1108
+ } else if (serverRef.type === 'http' && error instanceof Error) {
1109
+ const errorObj = error as Error & {
1110
+ cause?: unknown
1111
+ code?: string
1112
+ errno?: string | number
1113
+ syscall?: string
1114
+ }
1115
+ logMCPDebug(
1116
+ name,
1117
+ `HTTP Connection failed after ${elapsed}ms: ${error.message} (code: ${errorObj.code || 'none'}, errno: ${errorObj.errno || 'none'})`,
1118
+ )
1119
+ logMCPError(name, error)
1120
+
1121
+ if (error instanceof UnauthorizedError) {
1122
+ return handleRemoteAuthFailure(name, serverRef, 'http')
1123
+ }
1124
+ } else if (
1125
+ serverRef.type === 'claudeai-proxy' &&
1126
+ error instanceof Error
1127
+ ) {
1128
+ logMCPDebug(
1129
+ name,
1130
+ `claude.ai proxy connection failed after ${elapsed}ms: ${error.message}`,
1131
+ )
1132
+ logMCPError(name, error)
1133
+
1134
+ // StreamableHTTPError has a `code` property with the HTTP status
1135
+ const errorCode = (error as Error & { code?: number }).code
1136
+ if (errorCode === 401) {
1137
+ return handleRemoteAuthFailure(name, serverRef, 'claudeai-proxy')
1138
+ }
1139
+ } else if (
1140
+ serverRef.type === 'sse-ide' ||
1141
+ serverRef.type === 'ws-ide'
1142
+ ) {
1143
+ logEvent('tengu_mcp_ide_server_connection_failed', {
1144
+ connectionDurationMs: elapsed,
1145
+ })
1146
+ }
1147
+ if (inProcessServer) {
1148
+ inProcessServer.close().catch(() => {})
1149
+ }
1150
+ transport.close().catch(() => {})
1151
+ if (stderrOutput) {
1152
+ logMCPError(name, `Server stderr: ${stderrOutput}`)
1153
+ }
1154
+ throw error
1155
+ }
1156
+
1157
+ const capabilities = client.getServerCapabilities()
1158
+ const serverVersion = client.getServerVersion()
1159
+ const rawInstructions = client.getInstructions()
1160
+ let instructions = rawInstructions
1161
+ if (
1162
+ rawInstructions &&
1163
+ rawInstructions.length > MAX_MCP_DESCRIPTION_LENGTH
1164
+ ) {
1165
+ instructions =
1166
+ rawInstructions.slice(0, MAX_MCP_DESCRIPTION_LENGTH) + '… [truncated]'
1167
+ logMCPDebug(
1168
+ name,
1169
+ `Server instructions truncated from ${rawInstructions.length} to ${MAX_MCP_DESCRIPTION_LENGTH} chars`,
1170
+ )
1171
+ }
1172
+
1173
+ // Log successful connection details
1174
+ logMCPDebug(
1175
+ name,
1176
+ `Connection established with capabilities: ${jsonStringify({
1177
+ hasTools: !!capabilities?.tools,
1178
+ hasPrompts: !!capabilities?.prompts,
1179
+ hasResources: !!capabilities?.resources,
1180
+ hasResourceSubscribe: !!capabilities?.resources?.subscribe,
1181
+ serverVersion: serverVersion || 'unknown',
1182
+ })}`,
1183
+ )
1184
+ logForDebugging(
1185
+ `[MCP] Server "${name}" connected with subscribe=${!!capabilities?.resources?.subscribe}`,
1186
+ )
1187
+
1188
+ // Register default elicitation handler that returns cancel during the
1189
+ // window before registerElicitationHandler overwrites it in
1190
+ // onConnectionAttempt (useManageMCPConnections).
1191
+ client.setRequestHandler(ElicitRequestSchema, async request => {
1192
+ logMCPDebug(
1193
+ name,
1194
+ `Elicitation request received during initialization: ${jsonStringify(request)}`,
1195
+ )
1196
+ return { action: 'cancel' as const }
1197
+ })
1198
+
1199
+ if (serverRef.type === 'sse-ide' || serverRef.type === 'ws-ide') {
1200
+ const ideConnectionDurationMs = Date.now() - connectStartTime
1201
+ logEvent('tengu_mcp_ide_server_connection_succeeded', {
1202
+ connectionDurationMs: ideConnectionDurationMs,
1203
+ serverVersion:
1204
+ serverVersion as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1205
+ })
1206
+ try {
1207
+ void maybeNotifyIDEConnected(client)
1208
+ } catch (error) {
1209
+ logMCPError(
1210
+ name,
1211
+ `Failed to send ide_connected notification: ${error}`,
1212
+ )
1213
+ }
1214
+ }
1215
+
1216
+ // Enhanced connection drop detection and logging for all transport types
1217
+ const connectionStartTime = Date.now()
1218
+ let hasErrorOccurred = false
1219
+
1220
+ // Store original handlers
1221
+ const originalOnerror = client.onerror
1222
+ const originalOnclose = client.onclose
1223
+
1224
+ // The SDK's transport calls onerror on connection failures but doesn't call onclose,
1225
+ // which CC uses to trigger reconnection. We bridge this gap by tracking consecutive
1226
+ // terminal errors and manually closing after MAX_ERRORS_BEFORE_RECONNECT failures.
1227
+ let consecutiveConnectionErrors = 0
1228
+ const MAX_ERRORS_BEFORE_RECONNECT = 3
1229
+
1230
+ // Guard against re-entry: close() aborts in-flight streams which may fire
1231
+ // onerror again before the close chain completes.
1232
+ let hasTriggeredClose = false
1233
+
1234
+ // client.close() → transport.close() → transport.onclose → SDK's _onclose():
1235
+ // rejects all pending request handlers (so hung callTool() promises fail with
1236
+ // McpError -32000 "Connection closed") and then invokes our client.onclose
1237
+ // handler below (which clears the memo cache so the next call reconnects).
1238
+ // Calling client.onclose?.() directly would only clear the cache — pending
1239
+ // tool calls would stay hung.
1240
+ const closeTransportAndRejectPending = (reason: string) => {
1241
+ if (hasTriggeredClose) return
1242
+ hasTriggeredClose = true
1243
+ logMCPDebug(name, `Closing transport (${reason})`)
1244
+ void client.close().catch(e => {
1245
+ logMCPDebug(name, `Error during close: ${errorMessage(e)}`)
1246
+ })
1247
+ }
1248
+
1249
+ const isTerminalConnectionError = (msg: string): boolean => {
1250
+ return (
1251
+ msg.includes('ECONNRESET') ||
1252
+ msg.includes('ETIMEDOUT') ||
1253
+ msg.includes('EPIPE') ||
1254
+ msg.includes('EHOSTUNREACH') ||
1255
+ msg.includes('ECONNREFUSED') ||
1256
+ msg.includes('Body Timeout Error') ||
1257
+ msg.includes('terminated') ||
1258
+ // SDK SSE reconnection intermediate errors — may be wrapped around the
1259
+ // actual network error, so the substrings above won't match
1260
+ msg.includes('SSE stream disconnected') ||
1261
+ msg.includes('Failed to reconnect SSE stream')
1262
+ )
1263
+ }
1264
+
1265
+ // Enhanced error handler with detailed logging
1266
+ client.onerror = (error: Error) => {
1267
+ const uptime = Date.now() - connectionStartTime
1268
+ hasErrorOccurred = true
1269
+ const transportType = serverRef.type || 'stdio'
1270
+
1271
+ // Log the connection drop with context
1272
+ logMCPDebug(
1273
+ name,
1274
+ `${transportType.toUpperCase()} connection dropped after ${Math.floor(uptime / 1000)}s uptime`,
1275
+ )
1276
+
1277
+ // Log specific error details for debugging
1278
+ if (error.message) {
1279
+ if (error.message.includes('ECONNRESET')) {
1280
+ logMCPDebug(
1281
+ name,
1282
+ `Connection reset - server may have crashed or restarted`,
1283
+ )
1284
+ } else if (error.message.includes('ETIMEDOUT')) {
1285
+ logMCPDebug(
1286
+ name,
1287
+ `Connection timeout - network issue or server unresponsive`,
1288
+ )
1289
+ } else if (error.message.includes('ECONNREFUSED')) {
1290
+ logMCPDebug(name, `Connection refused - server may be down`)
1291
+ } else if (error.message.includes('EPIPE')) {
1292
+ logMCPDebug(
1293
+ name,
1294
+ `Broken pipe - server closed connection unexpectedly`,
1295
+ )
1296
+ } else if (error.message.includes('EHOSTUNREACH')) {
1297
+ logMCPDebug(name, `Host unreachable - network connectivity issue`)
1298
+ } else if (error.message.includes('ESRCH')) {
1299
+ logMCPDebug(
1300
+ name,
1301
+ `Process not found - stdio server process terminated`,
1302
+ )
1303
+ } else if (error.message.includes('spawn')) {
1304
+ logMCPDebug(
1305
+ name,
1306
+ `Failed to spawn process - check command and permissions`,
1307
+ )
1308
+ } else {
1309
+ logMCPDebug(name, `Connection error: ${error.message}`)
1310
+ }
1311
+ }
1312
+
1313
+ // For HTTP transports, detect session expiry (404 + JSON-RPC -32001)
1314
+ // and close the transport so pending tool calls reject and the next
1315
+ // call reconnects with a fresh session ID.
1316
+ if (
1317
+ (transportType === 'http' || transportType === 'claudeai-proxy') &&
1318
+ isMcpSessionExpiredError(error)
1319
+ ) {
1320
+ logMCPDebug(
1321
+ name,
1322
+ `MCP session expired (server returned 404 with session-not-found), triggering reconnection`,
1323
+ )
1324
+ closeTransportAndRejectPending('session expired')
1325
+ if (originalOnerror) {
1326
+ originalOnerror(error)
1327
+ }
1328
+ return
1329
+ }
1330
+
1331
+ // For remote transports (SSE/HTTP), track terminal connection errors
1332
+ // and trigger reconnection via close if we see repeated failures.
1333
+ if (
1334
+ transportType === 'sse' ||
1335
+ transportType === 'http' ||
1336
+ transportType === 'claudeai-proxy'
1337
+ ) {
1338
+ // The SDK's StreamableHTTP transport fires this after exhausting its
1339
+ // own SSE reconnect attempts (default maxRetries: 2) — but it never
1340
+ // calls onclose, so pending callTool() promises hang indefinitely.
1341
+ // This is the definitive "transport gave up" signal.
1342
+ if (error.message.includes('Maximum reconnection attempts')) {
1343
+ closeTransportAndRejectPending('SSE reconnection exhausted')
1344
+ if (originalOnerror) {
1345
+ originalOnerror(error)
1346
+ }
1347
+ return
1348
+ }
1349
+
1350
+ if (isTerminalConnectionError(error.message)) {
1351
+ consecutiveConnectionErrors++
1352
+ logMCPDebug(
1353
+ name,
1354
+ `Terminal connection error ${consecutiveConnectionErrors}/${MAX_ERRORS_BEFORE_RECONNECT}`,
1355
+ )
1356
+
1357
+ if (consecutiveConnectionErrors >= MAX_ERRORS_BEFORE_RECONNECT) {
1358
+ consecutiveConnectionErrors = 0
1359
+ closeTransportAndRejectPending('max consecutive terminal errors')
1360
+ }
1361
+ } else {
1362
+ // Non-terminal error (e.g., transient issue), reset counter
1363
+ consecutiveConnectionErrors = 0
1364
+ }
1365
+ }
1366
+
1367
+ // Call original handler
1368
+ if (originalOnerror) {
1369
+ originalOnerror(error)
1370
+ }
1371
+ }
1372
+
1373
+ // Enhanced close handler with connection drop context
1374
+ client.onclose = () => {
1375
+ const uptime = Date.now() - connectionStartTime
1376
+ const transportType = serverRef.type ?? 'unknown'
1377
+
1378
+ logMCPDebug(
1379
+ name,
1380
+ `${transportType.toUpperCase()} connection closed after ${Math.floor(uptime / 1000)}s (${hasErrorOccurred ? 'with errors' : 'cleanly'})`,
1381
+ )
1382
+
1383
+ // Clear the memoization cache so next operation reconnects
1384
+ const key = getServerCacheKey(name, serverRef)
1385
+
1386
+ // Also clear fetch caches (keyed by server name). Reconnection
1387
+ // creates a new connection object; without clearing, the next
1388
+ // fetch would return stale tools/resources from the old connection.
1389
+ fetchToolsForClient.cache.delete(name)
1390
+ fetchResourcesForClient.cache.delete(name)
1391
+ fetchCommandsForClient.cache.delete(name)
1392
+ if (feature('MCP_SKILLS')) {
1393
+ fetchMcpSkillsForClient!.cache.delete(name)
1394
+ }
1395
+
1396
+ connectToServer.cache.delete(key)
1397
+ logMCPDebug(name, `Cleared connection cache for reconnection`)
1398
+
1399
+ if (originalOnclose) {
1400
+ originalOnclose()
1401
+ }
1402
+ }
1403
+
1404
+ const cleanup = async () => {
1405
+ // In-process servers (e.g. Chrome MCP) don't have child processes or stderr
1406
+ if (inProcessServer) {
1407
+ try {
1408
+ await inProcessServer.close()
1409
+ } catch (error) {
1410
+ logMCPDebug(name, `Error closing in-process server: ${error}`)
1411
+ }
1412
+ try {
1413
+ await client.close()
1414
+ } catch (error) {
1415
+ logMCPDebug(name, `Error closing client: ${error}`)
1416
+ }
1417
+ return
1418
+ }
1419
+
1420
+ // Remove stderr event listener to prevent memory leaks
1421
+ if (stderrHandler && (serverRef.type === 'stdio' || !serverRef.type)) {
1422
+ const stdioTransport = transport as StdioClientTransport
1423
+ stdioTransport.stderr?.off('data', stderrHandler)
1424
+ }
1425
+
1426
+ // For stdio transports, explicitly terminate the child process with proper signals
1427
+ // NOTE: StdioClientTransport.close() only sends an abort signal, but many MCP servers
1428
+ // (especially Docker containers) need explicit SIGINT/SIGTERM signals to trigger graceful shutdown
1429
+ if (serverRef.type === 'stdio') {
1430
+ try {
1431
+ const stdioTransport = transport as StdioClientTransport
1432
+ const childPid = stdioTransport.pid
1433
+
1434
+ if (childPid) {
1435
+ logMCPDebug(name, 'Sending SIGINT to MCP server process')
1436
+
1437
+ // First try SIGINT (like Ctrl+C)
1438
+ try {
1439
+ process.kill(childPid, 'SIGINT')
1440
+ } catch (error) {
1441
+ logMCPDebug(name, `Error sending SIGINT: ${error}`)
1442
+ return
1443
+ }
1444
+
1445
+ // Wait for graceful shutdown with rapid escalation (total 500ms to keep CLI responsive)
1446
+ await new Promise<void>(async resolve => {
1447
+ let resolved = false
1448
+
1449
+ // Set up a timer to check if process still exists
1450
+ const checkInterval = setInterval(() => {
1451
+ try {
1452
+ // process.kill(pid, 0) checks if process exists without killing it
1453
+ process.kill(childPid, 0)
1454
+ } catch {
1455
+ // Process no longer exists
1456
+ if (!resolved) {
1457
+ resolved = true
1458
+ clearInterval(checkInterval)
1459
+ clearTimeout(failsafeTimeout)
1460
+ logMCPDebug(name, 'MCP server process exited cleanly')
1461
+ resolve()
1462
+ }
1463
+ }
1464
+ }, 50)
1465
+
1466
+ // Absolute failsafe: clear interval after 600ms no matter what
1467
+ const failsafeTimeout = setTimeout(() => {
1468
+ if (!resolved) {
1469
+ resolved = true
1470
+ clearInterval(checkInterval)
1471
+ logMCPDebug(
1472
+ name,
1473
+ 'Cleanup timeout reached, stopping process monitoring',
1474
+ )
1475
+ resolve()
1476
+ }
1477
+ }, 600)
1478
+
1479
+ try {
1480
+ // Wait 100ms for SIGINT to work (usually much faster)
1481
+ await sleep(100)
1482
+
1483
+ if (!resolved) {
1484
+ // Check if process still exists
1485
+ try {
1486
+ process.kill(childPid, 0)
1487
+ // Process still exists, SIGINT failed, try SIGTERM
1488
+ logMCPDebug(
1489
+ name,
1490
+ 'SIGINT failed, sending SIGTERM to MCP server process',
1491
+ )
1492
+ try {
1493
+ process.kill(childPid, 'SIGTERM')
1494
+ } catch (termError) {
1495
+ logMCPDebug(name, `Error sending SIGTERM: ${termError}`)
1496
+ resolved = true
1497
+ clearInterval(checkInterval)
1498
+ clearTimeout(failsafeTimeout)
1499
+ resolve()
1500
+ return
1501
+ }
1502
+ } catch {
1503
+ // Process already exited
1504
+ resolved = true
1505
+ clearInterval(checkInterval)
1506
+ clearTimeout(failsafeTimeout)
1507
+ resolve()
1508
+ return
1509
+ }
1510
+
1511
+ // Wait 400ms for SIGTERM to work (slower than SIGINT, often used for cleanup)
1512
+ await sleep(400)
1513
+
1514
+ if (!resolved) {
1515
+ // Check if process still exists
1516
+ try {
1517
+ process.kill(childPid, 0)
1518
+ // Process still exists, SIGTERM failed, force kill with SIGKILL
1519
+ logMCPDebug(
1520
+ name,
1521
+ 'SIGTERM failed, sending SIGKILL to MCP server process',
1522
+ )
1523
+ try {
1524
+ process.kill(childPid, 'SIGKILL')
1525
+ } catch (killError) {
1526
+ logMCPDebug(
1527
+ name,
1528
+ `Error sending SIGKILL: ${killError}`,
1529
+ )
1530
+ }
1531
+ } catch {
1532
+ // Process already exited
1533
+ resolved = true
1534
+ clearInterval(checkInterval)
1535
+ clearTimeout(failsafeTimeout)
1536
+ resolve()
1537
+ }
1538
+ }
1539
+ }
1540
+
1541
+ // Final timeout - always resolve after 500ms max (total cleanup time)
1542
+ if (!resolved) {
1543
+ resolved = true
1544
+ clearInterval(checkInterval)
1545
+ clearTimeout(failsafeTimeout)
1546
+ resolve()
1547
+ }
1548
+ } catch {
1549
+ // Handle any errors in the escalation sequence
1550
+ if (!resolved) {
1551
+ resolved = true
1552
+ clearInterval(checkInterval)
1553
+ clearTimeout(failsafeTimeout)
1554
+ resolve()
1555
+ }
1556
+ }
1557
+ })
1558
+ }
1559
+ } catch (processError) {
1560
+ logMCPDebug(name, `Error terminating process: ${processError}`)
1561
+ }
1562
+ }
1563
+
1564
+ // Close the client connection (which also closes the transport)
1565
+ try {
1566
+ await client.close()
1567
+ } catch (error) {
1568
+ logMCPDebug(name, `Error closing client: ${error}`)
1569
+ }
1570
+ }
1571
+
1572
+ // Register cleanup for all transport types - even network transports might need cleanup
1573
+ // This ensures all MCP servers get properly terminated, not just stdio ones
1574
+ const cleanupUnregister = registerCleanup(cleanup)
1575
+
1576
+ // Create the wrapped cleanup that includes unregistering
1577
+ const wrappedCleanup = async () => {
1578
+ cleanupUnregister?.()
1579
+ await cleanup()
1580
+ }
1581
+
1582
+ const connectionDurationMs = Date.now() - connectStartTime
1583
+ logEvent('tengu_mcp_server_connection_succeeded', {
1584
+ connectionDurationMs,
1585
+ transportType: (serverRef.type ??
1586
+ 'stdio') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1587
+ totalServers: serverStats?.totalServers,
1588
+ stdioCount: serverStats?.stdioCount,
1589
+ sseCount: serverStats?.sseCount,
1590
+ httpCount: serverStats?.httpCount,
1591
+ sseIdeCount: serverStats?.sseIdeCount,
1592
+ wsIdeCount: serverStats?.wsIdeCount,
1593
+ ...mcpBaseUrlAnalytics(serverRef),
1594
+ })
1595
+ return {
1596
+ name,
1597
+ client,
1598
+ type: 'connected' as const,
1599
+ capabilities: capabilities ?? {},
1600
+ serverInfo: serverVersion,
1601
+ instructions,
1602
+ config: serverRef,
1603
+ cleanup: wrappedCleanup,
1604
+ }
1605
+ } catch (error) {
1606
+ const connectionDurationMs = Date.now() - connectStartTime
1607
+ logEvent('tengu_mcp_server_connection_failed', {
1608
+ connectionDurationMs,
1609
+ totalServers: serverStats?.totalServers || 1,
1610
+ stdioCount:
1611
+ serverStats?.stdioCount || (serverRef.type === 'stdio' ? 1 : 0),
1612
+ sseCount: serverStats?.sseCount || (serverRef.type === 'sse' ? 1 : 0),
1613
+ httpCount:
1614
+ serverStats?.httpCount || (serverRef.type === 'http' ? 1 : 0),
1615
+ sseIdeCount:
1616
+ serverStats?.sseIdeCount || (serverRef.type === 'sse-ide' ? 1 : 0),
1617
+ wsIdeCount:
1618
+ serverStats?.wsIdeCount || (serverRef.type === 'ws-ide' ? 1 : 0),
1619
+ transportType: (serverRef.type ??
1620
+ 'stdio') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1621
+ ...mcpBaseUrlAnalytics(serverRef),
1622
+ })
1623
+ logMCPDebug(
1624
+ name,
1625
+ `Connection failed after ${connectionDurationMs}ms: ${errorMessage(error)}`,
1626
+ )
1627
+ logMCPError(name, `Connection failed: ${errorMessage(error)}`)
1628
+
1629
+ if (inProcessServer) {
1630
+ inProcessServer.close().catch(() => {})
1631
+ }
1632
+ return {
1633
+ name,
1634
+ type: 'failed' as const,
1635
+ config: serverRef,
1636
+ error: errorMessage(error),
1637
+ }
1638
+ }
1639
+ },
1640
+ getServerCacheKey,
1641
+ )
1642
+
1643
+ /**
1644
+ * Clears the memoize cache for a specific server
1645
+ * @param name Server name
1646
+ * @param serverRef Server configuration
1647
+ */
1648
+ export async function clearServerCache(
1649
+ name: string,
1650
+ serverRef: ScopedMcpServerConfig,
1651
+ ): Promise<void> {
1652
+ const key = getServerCacheKey(name, serverRef)
1653
+
1654
+ try {
1655
+ const wrappedClient = await connectToServer(name, serverRef)
1656
+
1657
+ if (wrappedClient.type === 'connected') {
1658
+ await wrappedClient.cleanup()
1659
+ }
1660
+ } catch {
1661
+ // Ignore errors - server might have failed to connect
1662
+ }
1663
+
1664
+ // Clear from cache (both connection and fetch caches so reconnect
1665
+ // fetches fresh tools/resources/commands instead of stale ones)
1666
+ connectToServer.cache.delete(key)
1667
+ fetchToolsForClient.cache.delete(name)
1668
+ fetchResourcesForClient.cache.delete(name)
1669
+ fetchCommandsForClient.cache.delete(name)
1670
+ if (feature('MCP_SKILLS')) {
1671
+ fetchMcpSkillsForClient!.cache.delete(name)
1672
+ }
1673
+ }
1674
+
1675
+ /**
1676
+ * Ensures a valid connected client for an MCP server.
1677
+ * For most server types, uses the memoization cache if available, or reconnects
1678
+ * if the cache was cleared (e.g., after onclose). This ensures tool/resource
1679
+ * calls always use a valid connection.
1680
+ *
1681
+ * SDK MCP servers run in-process and are handled separately via setupSdkMcpClients,
1682
+ * so they are returned as-is without going through connectToServer.
1683
+ *
1684
+ * @param client The connected MCP server client
1685
+ * @returns Connected MCP server client (same or reconnected)
1686
+ * @throws Error if server cannot be connected
1687
+ */
1688
+ export async function ensureConnectedClient(
1689
+ client: ConnectedMCPServer,
1690
+ ): Promise<ConnectedMCPServer> {
1691
+ // SDK MCP servers run in-process and are handled separately via setupSdkMcpClients
1692
+ if (client.config.type === 'sdk') {
1693
+ return client
1694
+ }
1695
+
1696
+ const connectedClient = await connectToServer(client.name, client.config)
1697
+ if (connectedClient.type !== 'connected') {
1698
+ throw new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
1699
+ `MCP server "${client.name}" is not connected`,
1700
+ 'MCP server not connected',
1701
+ )
1702
+ }
1703
+ return connectedClient
1704
+ }
1705
+
1706
+ /**
1707
+ * Compares two MCP server configurations to determine if they are equivalent.
1708
+ * Used to detect when a server needs to be reconnected due to config changes.
1709
+ */
1710
+ export function areMcpConfigsEqual(
1711
+ a: ScopedMcpServerConfig,
1712
+ b: ScopedMcpServerConfig,
1713
+ ): boolean {
1714
+ // Quick type check first
1715
+ if (a.type !== b.type) return false
1716
+
1717
+ // Compare by serializing - this handles all config variations
1718
+ // We exclude 'scope' from comparison since it's metadata, not connection config
1719
+ const { scope: _scopeA, ...configA } = a
1720
+ const { scope: _scopeB, ...configB } = b
1721
+ return jsonStringify(configA) === jsonStringify(configB)
1722
+ }
1723
+
1724
+ // Max cache size for fetch* caches. Keyed by server name (stable across
1725
+ // reconnects), bounded to prevent unbounded growth with many MCP servers.
1726
+ const MCP_FETCH_CACHE_SIZE = 20
1727
+
1728
+ /**
1729
+ * Encode MCP tool input for the auto-mode security classifier.
1730
+ * Exported so the auto-mode eval scripts can mirror production encoding
1731
+ * for `mcp__*` tool stubs without duplicating this logic.
1732
+ */
1733
+ export function mcpToolInputToAutoClassifierInput(
1734
+ input: Record<string, unknown>,
1735
+ toolName: string,
1736
+ ): string {
1737
+ const keys = Object.keys(input)
1738
+ return keys.length > 0
1739
+ ? keys.map(k => `${k}=${String(input[k])}`).join(' ')
1740
+ : toolName
1741
+ }
1742
+
1743
+ export const fetchToolsForClient = memoizeWithLRU(
1744
+ async (client: MCPServerConnection): Promise<Tool[]> => {
1745
+ if (client.type !== 'connected') return []
1746
+
1747
+ try {
1748
+ if (!client.capabilities?.tools) {
1749
+ return []
1750
+ }
1751
+
1752
+ const result = (await client.client.request(
1753
+ { method: 'tools/list' },
1754
+ ListToolsResultSchema,
1755
+ )) as ListToolsResult
1756
+
1757
+ // Sanitize tool data from MCP server
1758
+ const toolsToProcess = recursivelySanitizeUnicode(result.tools)
1759
+
1760
+ // Check if we should skip the mcp__ prefix for SDK MCP servers
1761
+ const skipPrefix =
1762
+ client.config.type === 'sdk' &&
1763
+ isEnvTruthy(process.env.CLAUDE_AGENT_SDK_MCP_NO_PREFIX)
1764
+
1765
+ // Convert MCP tools to our Tool format
1766
+ return toolsToProcess
1767
+ .map((tool): Tool => {
1768
+ const fullyQualifiedName = buildMcpToolName(client.name, tool.name)
1769
+ return {
1770
+ ...MCPTool,
1771
+ // In skip-prefix mode, use the original name for model invocation so MCP tools
1772
+ // can override builtins by name. mcpInfo is used for permission checking.
1773
+ name: skipPrefix ? tool.name : fullyQualifiedName,
1774
+ mcpInfo: { serverName: client.name, toolName: tool.name },
1775
+ isMcp: true,
1776
+ // Collapse whitespace: _meta is open to external MCP servers, and
1777
+ // a newline here would inject orphan lines into the deferred-tool
1778
+ // list (formatDeferredToolLine joins on '\n').
1779
+ searchHint:
1780
+ typeof tool._meta?.['anthropic/searchHint'] === 'string'
1781
+ ? tool._meta['anthropic/searchHint']
1782
+ .replace(/\s+/g, ' ')
1783
+ .trim() || undefined
1784
+ : undefined,
1785
+ alwaysLoad: tool._meta?.['anthropic/alwaysLoad'] === true,
1786
+ async description() {
1787
+ return tool.description ?? ''
1788
+ },
1789
+ async prompt() {
1790
+ const desc = tool.description ?? ''
1791
+ return desc.length > MAX_MCP_DESCRIPTION_LENGTH
1792
+ ? desc.slice(0, MAX_MCP_DESCRIPTION_LENGTH) + '… [truncated]'
1793
+ : desc
1794
+ },
1795
+ isConcurrencySafe() {
1796
+ return tool.annotations?.readOnlyHint ?? false
1797
+ },
1798
+ isReadOnly() {
1799
+ return tool.annotations?.readOnlyHint ?? false
1800
+ },
1801
+ toAutoClassifierInput(input) {
1802
+ return mcpToolInputToAutoClassifierInput(input, tool.name)
1803
+ },
1804
+ isDestructive() {
1805
+ return tool.annotations?.destructiveHint ?? false
1806
+ },
1807
+ isOpenWorld() {
1808
+ return tool.annotations?.openWorldHint ?? false
1809
+ },
1810
+ isSearchOrReadCommand() {
1811
+ return classifyMcpToolForCollapse(client.name, tool.name)
1812
+ },
1813
+ inputJSONSchema: tool.inputSchema as Tool['inputJSONSchema'],
1814
+ async checkPermissions() {
1815
+ return {
1816
+ behavior: 'passthrough' as const,
1817
+ message: 'MCPTool requires permission.',
1818
+ suggestions: [
1819
+ {
1820
+ type: 'addRules' as const,
1821
+ rules: [
1822
+ {
1823
+ toolName: fullyQualifiedName,
1824
+ ruleContent: undefined,
1825
+ },
1826
+ ],
1827
+ behavior: 'allow' as const,
1828
+ destination: 'localSettings' as const,
1829
+ },
1830
+ ],
1831
+ }
1832
+ },
1833
+ async call(
1834
+ args: Record<string, unknown>,
1835
+ context,
1836
+ _canUseTool,
1837
+ parentMessage,
1838
+ onProgress?: ToolCallProgress<MCPProgress>,
1839
+ ) {
1840
+ const toolUseId = extractToolUseId(parentMessage)
1841
+ const meta = toolUseId
1842
+ ? { 'claudecode/toolUseId': toolUseId }
1843
+ : {}
1844
+
1845
+ // Emit progress when tool starts
1846
+ if (onProgress && toolUseId) {
1847
+ onProgress({
1848
+ toolUseID: toolUseId,
1849
+ data: {
1850
+ type: 'mcp_progress',
1851
+ status: 'started',
1852
+ serverName: client.name,
1853
+ toolName: tool.name,
1854
+ },
1855
+ })
1856
+ }
1857
+
1858
+ const startTime = Date.now()
1859
+ const MAX_SESSION_RETRIES = 1
1860
+ for (let attempt = 0; ; attempt++) {
1861
+ try {
1862
+ const connectedClient = await ensureConnectedClient(client)
1863
+ const mcpResult = await callMCPToolWithUrlElicitationRetry({
1864
+ client: connectedClient,
1865
+ clientConnection: client,
1866
+ tool: tool.name,
1867
+ args,
1868
+ meta,
1869
+ signal: context.abortController.signal,
1870
+ setAppState: context.setAppState,
1871
+ onProgress:
1872
+ onProgress && toolUseId
1873
+ ? progressData => {
1874
+ onProgress({
1875
+ toolUseID: toolUseId,
1876
+ data: progressData,
1877
+ })
1878
+ }
1879
+ : undefined,
1880
+ handleElicitation: context.handleElicitation,
1881
+ })
1882
+
1883
+ // Emit progress when tool completes successfully
1884
+ if (onProgress && toolUseId) {
1885
+ onProgress({
1886
+ toolUseID: toolUseId,
1887
+ data: {
1888
+ type: 'mcp_progress',
1889
+ status: 'completed',
1890
+ serverName: client.name,
1891
+ toolName: tool.name,
1892
+ elapsedTimeMs: Date.now() - startTime,
1893
+ },
1894
+ })
1895
+ }
1896
+
1897
+ return {
1898
+ data: mcpResult.content,
1899
+ ...((mcpResult._meta || mcpResult.structuredContent) && {
1900
+ mcpMeta: {
1901
+ ...(mcpResult._meta && {
1902
+ _meta: mcpResult._meta,
1903
+ }),
1904
+ ...(mcpResult.structuredContent && {
1905
+ structuredContent: mcpResult.structuredContent,
1906
+ }),
1907
+ },
1908
+ }),
1909
+ }
1910
+ } catch (error) {
1911
+ // Session expired — the connection cache has been
1912
+ // cleared, so retry with a fresh client.
1913
+ if (
1914
+ error instanceof McpSessionExpiredError &&
1915
+ attempt < MAX_SESSION_RETRIES
1916
+ ) {
1917
+ logMCPDebug(
1918
+ client.name,
1919
+ `Retrying tool '${tool.name}' after session recovery`,
1920
+ )
1921
+ continue
1922
+ }
1923
+
1924
+ // Emit progress when tool fails
1925
+ if (onProgress && toolUseId) {
1926
+ onProgress({
1927
+ toolUseID: toolUseId,
1928
+ data: {
1929
+ type: 'mcp_progress',
1930
+ status: 'failed',
1931
+ serverName: client.name,
1932
+ toolName: tool.name,
1933
+ elapsedTimeMs: Date.now() - startTime,
1934
+ },
1935
+ })
1936
+ }
1937
+ // Wrap MCP SDK errors so telemetry gets useful context
1938
+ // instead of just "Error" or "McpError" (the constructor
1939
+ // name). MCP SDK errors are protocol-level messages and
1940
+ // don't contain user file paths or code.
1941
+ if (
1942
+ error instanceof Error &&
1943
+ !(
1944
+ error instanceof
1945
+ TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
1946
+ )
1947
+ ) {
1948
+ const name = error.constructor.name
1949
+ if (name === 'Error') {
1950
+ throw new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
1951
+ error.message,
1952
+ error.message.slice(0, 200),
1953
+ )
1954
+ }
1955
+ // McpError has a numeric `code` with the JSON-RPC error
1956
+ // code (e.g. -32000 ConnectionClosed, -32001 RequestTimeout)
1957
+ if (
1958
+ name === 'McpError' &&
1959
+ 'code' in error &&
1960
+ typeof error.code === 'number'
1961
+ ) {
1962
+ throw new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
1963
+ error.message,
1964
+ `McpError ${error.code}`,
1965
+ )
1966
+ }
1967
+ }
1968
+ throw error
1969
+ }
1970
+ }
1971
+ },
1972
+ userFacingName() {
1973
+ // Prefer title annotation if available, otherwise use tool name
1974
+ const displayName = tool.annotations?.title || tool.name
1975
+ return `${client.name} - ${displayName} (MCP)`
1976
+ },
1977
+ ...(isClaudeInChromeMCPServer(client.name) &&
1978
+ (client.config.type === 'stdio' || !client.config.type)
1979
+ ? claudeInChromeToolRendering().getClaudeInChromeMCPToolOverrides(
1980
+ tool.name,
1981
+ )
1982
+ : {}),
1983
+ ...(feature('CHICAGO_MCP') &&
1984
+ (client.config.type === 'stdio' || !client.config.type) &&
1985
+ isComputerUseMCPServer!(client.name)
1986
+ ? computerUseWrapper!().getComputerUseMCPToolOverrides(tool.name)
1987
+ : {}),
1988
+ }
1989
+ })
1990
+ .filter(isIncludedMcpTool)
1991
+ } catch (error) {
1992
+ logMCPError(client.name, `Failed to fetch tools: ${errorMessage(error)}`)
1993
+ return []
1994
+ }
1995
+ },
1996
+ (client: MCPServerConnection) => client.name,
1997
+ MCP_FETCH_CACHE_SIZE,
1998
+ )
1999
+
2000
+ export const fetchResourcesForClient = memoizeWithLRU(
2001
+ async (client: MCPServerConnection): Promise<ServerResource[]> => {
2002
+ if (client.type !== 'connected') return []
2003
+
2004
+ try {
2005
+ if (!client.capabilities?.resources) {
2006
+ return []
2007
+ }
2008
+
2009
+ const result = await client.client.request(
2010
+ { method: 'resources/list' },
2011
+ ListResourcesResultSchema,
2012
+ )
2013
+
2014
+ if (!result.resources) return []
2015
+
2016
+ // Add server name to each resource
2017
+ return result.resources.map(resource => ({
2018
+ ...resource,
2019
+ server: client.name,
2020
+ }))
2021
+ } catch (error) {
2022
+ logMCPError(
2023
+ client.name,
2024
+ `Failed to fetch resources: ${errorMessage(error)}`,
2025
+ )
2026
+ return []
2027
+ }
2028
+ },
2029
+ (client: MCPServerConnection) => client.name,
2030
+ MCP_FETCH_CACHE_SIZE,
2031
+ )
2032
+
2033
+ export const fetchCommandsForClient = memoizeWithLRU(
2034
+ async (client: MCPServerConnection): Promise<Command[]> => {
2035
+ if (client.type !== 'connected') return []
2036
+
2037
+ try {
2038
+ if (!client.capabilities?.prompts) {
2039
+ return []
2040
+ }
2041
+
2042
+ // Request prompts list from client
2043
+ const result = (await client.client.request(
2044
+ { method: 'prompts/list' },
2045
+ ListPromptsResultSchema,
2046
+ )) as ListPromptsResult
2047
+
2048
+ if (!result.prompts) return []
2049
+
2050
+ // Sanitize prompt data from MCP server
2051
+ const promptsToProcess = recursivelySanitizeUnicode(result.prompts)
2052
+
2053
+ // Convert MCP prompts to our Command format
2054
+ return promptsToProcess.map(prompt => {
2055
+ const argNames = Object.values(prompt.arguments ?? {}).map(k => k.name)
2056
+ return {
2057
+ type: 'prompt' as const,
2058
+ name: 'mcp__' + normalizeNameForMCP(client.name) + '__' + prompt.name,
2059
+ description: prompt.description ?? '',
2060
+ hasUserSpecifiedDescription: !!prompt.description,
2061
+ contentLength: 0, // Dynamic MCP content
2062
+ isEnabled: () => true,
2063
+ isHidden: false,
2064
+ isMcp: true,
2065
+ progressMessage: 'running',
2066
+ userFacingName() {
2067
+ // Use prompt.name (programmatic identifier) not prompt.title (display name)
2068
+ // to avoid spaces breaking slash command parsing
2069
+ return `${client.name}:${prompt.name} (MCP)`
2070
+ },
2071
+ argNames,
2072
+ source: 'mcp',
2073
+ async getPromptForCommand(args: string) {
2074
+ const argsArray = args.split(' ')
2075
+ try {
2076
+ const connectedClient = await ensureConnectedClient(client)
2077
+ const result = await connectedClient.client.getPrompt({
2078
+ name: prompt.name,
2079
+ arguments: zipObject(argNames, argsArray),
2080
+ })
2081
+ const transformed = await Promise.all(
2082
+ result.messages.map(message =>
2083
+ transformResultContent(message.content, connectedClient.name),
2084
+ ),
2085
+ )
2086
+ return transformed.flat()
2087
+ } catch (error) {
2088
+ logMCPError(
2089
+ client.name,
2090
+ `Error running command '${prompt.name}': ${errorMessage(error)}`,
2091
+ )
2092
+ throw error
2093
+ }
2094
+ },
2095
+ }
2096
+ })
2097
+ } catch (error) {
2098
+ logMCPError(
2099
+ client.name,
2100
+ `Failed to fetch commands: ${errorMessage(error)}`,
2101
+ )
2102
+ return []
2103
+ }
2104
+ },
2105
+ (client: MCPServerConnection) => client.name,
2106
+ MCP_FETCH_CACHE_SIZE,
2107
+ )
2108
+
2109
+ /**
2110
+ * Call an IDE tool directly as an RPC
2111
+ * @param toolName The name of the tool to call
2112
+ * @param args The arguments to pass to the tool
2113
+ * @param client The IDE client to use for the RPC call
2114
+ * @returns The result of the tool call
2115
+ */
2116
+ export async function callIdeRpc(
2117
+ toolName: string,
2118
+ args: Record<string, unknown>,
2119
+ client: ConnectedMCPServer,
2120
+ ): Promise<string | ContentBlockParam[] | undefined> {
2121
+ const result = await callMCPTool({
2122
+ client,
2123
+ tool: toolName,
2124
+ args,
2125
+ signal: createAbortController().signal,
2126
+ })
2127
+ return result.content
2128
+ }
2129
+
2130
+ /**
2131
+ * Note: This should not be called by UI components directly, they should use the reconnectMcpServer
2132
+ * function from useManageMcpConnections.
2133
+ * @param name Server name
2134
+ * @param config Server configuration
2135
+ * @returns Object containing the client connection and its resources
2136
+ */
2137
+ export async function reconnectMcpServerImpl(
2138
+ name: string,
2139
+ config: ScopedMcpServerConfig,
2140
+ ): Promise<{
2141
+ client: MCPServerConnection
2142
+ tools: Tool[]
2143
+ commands: Command[]
2144
+ resources?: ServerResource[]
2145
+ }> {
2146
+ try {
2147
+ // Invalidate the keychain cache so we read fresh credentials from disk.
2148
+ // This is necessary when another process (e.g. the VS Code extension host)
2149
+ // has modified stored tokens (cleared auth, saved new OAuth tokens) and then
2150
+ // asks the CLI subprocess to reconnect. Without this, the subprocess would
2151
+ // use stale cached data and never notice the tokens were removed.
2152
+ clearKeychainCache()
2153
+
2154
+ await clearServerCache(name, config)
2155
+ const client = await connectToServer(name, config)
2156
+
2157
+ if (client.type !== 'connected') {
2158
+ return {
2159
+ client,
2160
+ tools: [],
2161
+ commands: [],
2162
+ }
2163
+ }
2164
+
2165
+ if (config.type === 'claudeai-proxy') {
2166
+ markClaudeAiMcpConnected(name)
2167
+ }
2168
+
2169
+ const supportsResources = !!client.capabilities?.resources
2170
+
2171
+ const [tools, mcpCommands, mcpSkills, resources] = await Promise.all([
2172
+ fetchToolsForClient(client),
2173
+ fetchCommandsForClient(client),
2174
+ feature('MCP_SKILLS') && supportsResources
2175
+ ? fetchMcpSkillsForClient!(client)
2176
+ : Promise.resolve([]),
2177
+ supportsResources ? fetchResourcesForClient(client) : Promise.resolve([]),
2178
+ ])
2179
+ const commands = [...mcpCommands, ...mcpSkills]
2180
+
2181
+ // Check if we need to add resource tools
2182
+ const resourceTools: Tool[] = []
2183
+ if (supportsResources) {
2184
+ // Only add resource tools if no other server has them
2185
+ const hasResourceTools = [ListMcpResourcesTool, ReadMcpResourceTool].some(
2186
+ tool => tools.some(t => toolMatchesName(t, tool.name)),
2187
+ )
2188
+ if (!hasResourceTools) {
2189
+ resourceTools.push(ListMcpResourcesTool, ReadMcpResourceTool)
2190
+ }
2191
+ }
2192
+
2193
+ return {
2194
+ client,
2195
+ tools: [...tools, ...resourceTools],
2196
+ commands,
2197
+ resources: resources.length > 0 ? resources : undefined,
2198
+ }
2199
+ } catch (error) {
2200
+ // Handle errors gracefully - connection might have closed during fetch
2201
+ logMCPError(name, `Error during reconnection: ${errorMessage(error)}`)
2202
+
2203
+ // Return with failed status
2204
+ return {
2205
+ client: { name, type: 'failed' as const, config },
2206
+ tools: [],
2207
+ commands: [],
2208
+ }
2209
+ }
2210
+ }
2211
+
2212
+ // Replaced 2026-03: previous implementation ran fixed-size sequential batches
2213
+ // (await batch 1 fully, then start batch 2). That meant one slow server in
2214
+ // batch N held up ALL servers in batch N+1, even if the other 19 slots were
2215
+ // idle. pMap frees each slot as soon as its server completes, so a single
2216
+ // slow server only occupies one slot instead of blocking an entire batch
2217
+ // boundary. Same concurrency ceiling, same results, better scheduling.
2218
+ async function processBatched<T>(
2219
+ items: T[],
2220
+ concurrency: number,
2221
+ processor: (item: T) => Promise<void>,
2222
+ ): Promise<void> {
2223
+ await pMap(items, processor, { concurrency })
2224
+ }
2225
+
2226
+ export async function getMcpToolsCommandsAndResources(
2227
+ onConnectionAttempt: (params: {
2228
+ client: MCPServerConnection
2229
+ tools: Tool[]
2230
+ commands: Command[]
2231
+ resources?: ServerResource[]
2232
+ }) => void,
2233
+ mcpConfigs?: Record<string, ScopedMcpServerConfig>,
2234
+ ): Promise<void> {
2235
+ let resourceToolsAdded = false
2236
+
2237
+ const allConfigEntries = Object.entries(
2238
+ mcpConfigs ?? (await getAllMcpConfigs()).servers,
2239
+ )
2240
+
2241
+ // Partition into disabled and active entries — disabled servers should
2242
+ // never generate HTTP connections or flow through batch processing
2243
+ const configEntries: typeof allConfigEntries = []
2244
+ for (const entry of allConfigEntries) {
2245
+ if (isMcpServerDisabled(entry[0])) {
2246
+ onConnectionAttempt({
2247
+ client: { name: entry[0], type: 'disabled', config: entry[1] },
2248
+ tools: [],
2249
+ commands: [],
2250
+ })
2251
+ } else {
2252
+ configEntries.push(entry)
2253
+ }
2254
+ }
2255
+
2256
+ // Calculate transport counts for logging
2257
+ const totalServers = configEntries.length
2258
+ const stdioCount = count(configEntries, ([_, c]) => c.type === 'stdio')
2259
+ const sseCount = count(configEntries, ([_, c]) => c.type === 'sse')
2260
+ const httpCount = count(configEntries, ([_, c]) => c.type === 'http')
2261
+ const sseIdeCount = count(configEntries, ([_, c]) => c.type === 'sse-ide')
2262
+ const wsIdeCount = count(configEntries, ([_, c]) => c.type === 'ws-ide')
2263
+
2264
+ // Split servers by type: local (stdio/sdk) need lower concurrency due to
2265
+ // process spawning, remote servers can connect with higher concurrency
2266
+ const localServers = configEntries.filter(([_, config]) =>
2267
+ isLocalMcpServer(config),
2268
+ )
2269
+ const remoteServers = configEntries.filter(
2270
+ ([_, config]) => !isLocalMcpServer(config),
2271
+ )
2272
+
2273
+ const serverStats = {
2274
+ totalServers,
2275
+ stdioCount,
2276
+ sseCount,
2277
+ httpCount,
2278
+ sseIdeCount,
2279
+ wsIdeCount,
2280
+ }
2281
+
2282
+ const processServer = async ([name, config]: [
2283
+ string,
2284
+ ScopedMcpServerConfig,
2285
+ ]): Promise<void> => {
2286
+ try {
2287
+ // Check if server is disabled - if so, just add it to state without connecting
2288
+ if (isMcpServerDisabled(name)) {
2289
+ onConnectionAttempt({
2290
+ client: {
2291
+ name,
2292
+ type: 'disabled',
2293
+ config,
2294
+ },
2295
+ tools: [],
2296
+ commands: [],
2297
+ })
2298
+ return
2299
+ }
2300
+
2301
+ // Skip connection for servers that recently returned 401 (15min TTL),
2302
+ // or that we have probed before but hold no token for. The second
2303
+ // check closes the gap the TTL leaves open: without it, every 15min
2304
+ // we re-probe servers that cannot succeed until the user runs /mcp.
2305
+ // Each probe is a network round-trip for connect-401 plus OAuth
2306
+ // discovery, and print mode awaits the whole batch (main.tsx:3503).
2307
+ if (
2308
+ (config.type === 'claudeai-proxy' ||
2309
+ config.type === 'http' ||
2310
+ config.type === 'sse') &&
2311
+ ((await isMcpAuthCached(name)) ||
2312
+ ((config.type === 'http' || config.type === 'sse') &&
2313
+ hasMcpDiscoveryButNoToken(name, config)))
2314
+ ) {
2315
+ logMCPDebug(name, `Skipping connection (cached needs-auth)`)
2316
+ onConnectionAttempt({
2317
+ client: { name, type: 'needs-auth' as const, config },
2318
+ tools: [createMcpAuthTool(name, config)],
2319
+ commands: [],
2320
+ })
2321
+ return
2322
+ }
2323
+
2324
+ const client = await connectToServer(name, config, serverStats)
2325
+
2326
+ if (client.type !== 'connected') {
2327
+ onConnectionAttempt({
2328
+ client,
2329
+ tools:
2330
+ client.type === 'needs-auth'
2331
+ ? [createMcpAuthTool(name, config)]
2332
+ : [],
2333
+ commands: [],
2334
+ })
2335
+ return
2336
+ }
2337
+
2338
+ if (config.type === 'claudeai-proxy') {
2339
+ markClaudeAiMcpConnected(name)
2340
+ }
2341
+
2342
+ const supportsResources = !!client.capabilities?.resources
2343
+
2344
+ const [tools, mcpCommands, mcpSkills, resources] = await Promise.all([
2345
+ fetchToolsForClient(client),
2346
+ fetchCommandsForClient(client),
2347
+ // Discover skills from skill:// resources
2348
+ feature('MCP_SKILLS') && supportsResources
2349
+ ? fetchMcpSkillsForClient!(client)
2350
+ : Promise.resolve([]),
2351
+ // Fetch resources if supported
2352
+ supportsResources
2353
+ ? fetchResourcesForClient(client)
2354
+ : Promise.resolve([]),
2355
+ ])
2356
+ const commands = [...mcpCommands, ...mcpSkills]
2357
+
2358
+ // If this server resources and we haven't added resource tools yet,
2359
+ // include our resource tools with this client's tools
2360
+ const resourceTools: Tool[] = []
2361
+ if (supportsResources && !resourceToolsAdded) {
2362
+ resourceToolsAdded = true
2363
+ resourceTools.push(ListMcpResourcesTool, ReadMcpResourceTool)
2364
+ }
2365
+
2366
+ onConnectionAttempt({
2367
+ client,
2368
+ tools: [...tools, ...resourceTools],
2369
+ commands,
2370
+ resources: resources.length > 0 ? resources : undefined,
2371
+ })
2372
+ } catch (error) {
2373
+ // Handle errors gracefully - connection might have closed during fetch
2374
+ logMCPError(
2375
+ name,
2376
+ `Error fetching tools/commands/resources: ${errorMessage(error)}`,
2377
+ )
2378
+
2379
+ // Still update with the client but no tools/commands
2380
+ onConnectionAttempt({
2381
+ client: { name, type: 'failed' as const, config },
2382
+ tools: [],
2383
+ commands: [],
2384
+ })
2385
+ }
2386
+ }
2387
+
2388
+ // Process both groups concurrently, each with their own concurrency limits:
2389
+ // - Local servers (stdio/sdk): lower concurrency to avoid process spawning resource contention
2390
+ // - Remote servers: higher concurrency since they're just network connections
2391
+ await Promise.all([
2392
+ processBatched(
2393
+ localServers,
2394
+ getMcpServerConnectionBatchSize(),
2395
+ processServer,
2396
+ ),
2397
+ processBatched(
2398
+ remoteServers,
2399
+ getRemoteMcpServerConnectionBatchSize(),
2400
+ processServer,
2401
+ ),
2402
+ ])
2403
+ }
2404
+
2405
+ // Not memoized: called only 2-3 times at startup/reconfig. The inner work
2406
+ // (connectToServer, fetch*ForClient) is already cached. Memoizing here by
2407
+ // mcpConfigs object ref leaked — main.tsx creates fresh config objects each call.
2408
+ export function prefetchAllMcpResources(
2409
+ mcpConfigs: Record<string, ScopedMcpServerConfig>,
2410
+ ): Promise<{
2411
+ clients: MCPServerConnection[]
2412
+ tools: Tool[]
2413
+ commands: Command[]
2414
+ }> {
2415
+ return new Promise(resolve => {
2416
+ let pendingCount = 0
2417
+ let completedCount = 0
2418
+
2419
+ pendingCount = Object.keys(mcpConfigs).length
2420
+
2421
+ if (pendingCount === 0) {
2422
+ void resolve({
2423
+ clients: [],
2424
+ tools: [],
2425
+ commands: [],
2426
+ })
2427
+ return
2428
+ }
2429
+
2430
+ const clients: MCPServerConnection[] = []
2431
+ const tools: Tool[] = []
2432
+ const commands: Command[] = []
2433
+
2434
+ getMcpToolsCommandsAndResources(result => {
2435
+ clients.push(result.client)
2436
+ tools.push(...result.tools)
2437
+ commands.push(...result.commands)
2438
+
2439
+ completedCount++
2440
+ if (completedCount >= pendingCount) {
2441
+ const commandsMetadataLength = commands.reduce((sum, command) => {
2442
+ const commandMetadataLength =
2443
+ command.name.length +
2444
+ (command.description ?? '').length +
2445
+ (command.argumentHint ?? '').length
2446
+ return sum + commandMetadataLength
2447
+ }, 0)
2448
+ logEvent('tengu_mcp_tools_commands_loaded', {
2449
+ tools_count: tools.length,
2450
+ commands_count: commands.length,
2451
+ commands_metadata_length: commandsMetadataLength,
2452
+ })
2453
+
2454
+ void resolve({
2455
+ clients,
2456
+ tools,
2457
+ commands,
2458
+ })
2459
+ }
2460
+ }, mcpConfigs).catch(error => {
2461
+ logMCPError(
2462
+ 'prefetchAllMcpResources',
2463
+ `Failed to get MCP resources: ${errorMessage(error)}`,
2464
+ )
2465
+ // Still resolve with empty results
2466
+ void resolve({
2467
+ clients: [],
2468
+ tools: [],
2469
+ commands: [],
2470
+ })
2471
+ })
2472
+ })
2473
+ }
2474
+
2475
+ /**
2476
+ * Transform result content from an MCP tool or MCP prompt into message blocks
2477
+ */
2478
+ export async function transformResultContent(
2479
+ resultContent: PromptMessage['content'],
2480
+ serverName: string,
2481
+ ): Promise<Array<ContentBlockParam>> {
2482
+ switch (resultContent.type) {
2483
+ case 'text':
2484
+ return [
2485
+ {
2486
+ type: 'text',
2487
+ text: resultContent.text,
2488
+ },
2489
+ ]
2490
+ case 'audio': {
2491
+ const audioData = resultContent as {
2492
+ type: 'audio'
2493
+ data: string
2494
+ mimeType?: string
2495
+ }
2496
+ return await persistBlobToTextBlock(
2497
+ Buffer.from(audioData.data, 'base64'),
2498
+ audioData.mimeType,
2499
+ serverName,
2500
+ `[Audio from ${serverName}] `,
2501
+ )
2502
+ }
2503
+ case 'image': {
2504
+ // Resize and compress image data, enforcing API dimension limits
2505
+ const imageBuffer = Buffer.from(String(resultContent.data), 'base64')
2506
+ const ext = resultContent.mimeType?.split('/')[1] || 'png'
2507
+ const resized = await maybeResizeAndDownsampleImageBuffer(
2508
+ imageBuffer,
2509
+ imageBuffer.length,
2510
+ ext,
2511
+ )
2512
+ return [
2513
+ {
2514
+ type: 'image',
2515
+ source: {
2516
+ data: resized.buffer.toString('base64'),
2517
+ media_type:
2518
+ `image/${resized.mediaType}` as Base64ImageSource['media_type'],
2519
+ type: 'base64',
2520
+ },
2521
+ },
2522
+ ]
2523
+ }
2524
+ case 'resource': {
2525
+ const resource = resultContent.resource
2526
+ const prefix = `[Resource from ${serverName} at ${resource.uri}] `
2527
+
2528
+ if ('text' in resource) {
2529
+ return [
2530
+ {
2531
+ type: 'text',
2532
+ text: `${prefix}${resource.text}`,
2533
+ },
2534
+ ]
2535
+ } else if ('blob' in resource) {
2536
+ const isImage = IMAGE_MIME_TYPES.has(resource.mimeType ?? '')
2537
+
2538
+ if (isImage) {
2539
+ // Resize and compress image blob, enforcing API dimension limits
2540
+ const imageBuffer = Buffer.from(resource.blob, 'base64')
2541
+ const ext = resource.mimeType?.split('/')[1] || 'png'
2542
+ const resized = await maybeResizeAndDownsampleImageBuffer(
2543
+ imageBuffer,
2544
+ imageBuffer.length,
2545
+ ext,
2546
+ )
2547
+ const content: MessageParam['content'] = []
2548
+ if (prefix) {
2549
+ content.push({
2550
+ type: 'text',
2551
+ text: prefix,
2552
+ })
2553
+ }
2554
+ content.push({
2555
+ type: 'image',
2556
+ source: {
2557
+ data: resized.buffer.toString('base64'),
2558
+ media_type:
2559
+ `image/${resized.mediaType}` as Base64ImageSource['media_type'],
2560
+ type: 'base64',
2561
+ },
2562
+ })
2563
+ return content
2564
+ } else {
2565
+ return await persistBlobToTextBlock(
2566
+ Buffer.from(resource.blob, 'base64'),
2567
+ resource.mimeType,
2568
+ serverName,
2569
+ prefix,
2570
+ )
2571
+ }
2572
+ }
2573
+ return []
2574
+ }
2575
+ case 'resource_link': {
2576
+ const resourceLink = resultContent as ResourceLink
2577
+ let text = `[Resource link: ${resourceLink.name}] ${resourceLink.uri}`
2578
+ if (resourceLink.description) {
2579
+ text += ` (${resourceLink.description})`
2580
+ }
2581
+ return [
2582
+ {
2583
+ type: 'text',
2584
+ text,
2585
+ },
2586
+ ]
2587
+ }
2588
+ default:
2589
+ return []
2590
+ }
2591
+ }
2592
+
2593
+ /**
2594
+ * Decode base64 binary content, write it to disk with the proper extension,
2595
+ * and return a small text block with the file path. Replaces the old behavior
2596
+ * of dumping raw base64 into the context.
2597
+ */
2598
+ async function persistBlobToTextBlock(
2599
+ bytes: Buffer,
2600
+ mimeType: string | undefined,
2601
+ serverName: string,
2602
+ sourceDescription: string,
2603
+ ): Promise<Array<ContentBlockParam>> {
2604
+ const persistId = `mcp-${normalizeNameForMCP(serverName)}-blob-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`
2605
+ const result = await persistBinaryContent(bytes, mimeType, persistId)
2606
+
2607
+ if ('error' in result) {
2608
+ return [
2609
+ {
2610
+ type: 'text',
2611
+ text: `${sourceDescription}Binary content (${mimeType || 'unknown type'}, ${bytes.length} bytes) could not be saved to disk: ${result.error}`,
2612
+ },
2613
+ ]
2614
+ }
2615
+
2616
+ return [
2617
+ {
2618
+ type: 'text',
2619
+ text: getBinaryBlobSavedMessage(
2620
+ result.filepath,
2621
+ mimeType,
2622
+ result.size,
2623
+ sourceDescription,
2624
+ ),
2625
+ },
2626
+ ]
2627
+ }
2628
+
2629
+ /**
2630
+ * Processes MCP tool result into a normalized format.
2631
+ */
2632
+ export type MCPResultType = 'toolResult' | 'structuredContent' | 'contentArray'
2633
+
2634
+ export type TransformedMCPResult = {
2635
+ content: MCPToolResult
2636
+ type: MCPResultType
2637
+ schema?: string
2638
+ }
2639
+
2640
+ /**
2641
+ * Generates a compact, jq-friendly type signature for a value.
2642
+ * e.g. "{title: string, items: [{id: number, name: string}]}"
2643
+ */
2644
+ export function inferCompactSchema(value: unknown, depth = 2): string {
2645
+ if (value === null) return 'null'
2646
+ if (Array.isArray(value)) {
2647
+ if (value.length === 0) return '[]'
2648
+ return `[${inferCompactSchema(value[0], depth - 1)}]`
2649
+ }
2650
+ if (typeof value === 'object') {
2651
+ if (depth <= 0) return '{...}'
2652
+ const entries = Object.entries(value).slice(0, 10)
2653
+ const props = entries.map(
2654
+ ([k, v]) => `${k}: ${inferCompactSchema(v, depth - 1)}`,
2655
+ )
2656
+ const suffix = Object.keys(value).length > 10 ? ', ...' : ''
2657
+ return `{${props.join(', ')}${suffix}}`
2658
+ }
2659
+ return typeof value
2660
+ }
2661
+
2662
+ export async function transformMCPResult(
2663
+ result: unknown,
2664
+ tool: string, // Tool name for validation (e.g., "search")
2665
+ name: string, // Server name for transformation (e.g., "slack")
2666
+ ): Promise<TransformedMCPResult> {
2667
+ if (result && typeof result === 'object') {
2668
+ if ('toolResult' in result) {
2669
+ return {
2670
+ content: String(result.toolResult),
2671
+ type: 'toolResult',
2672
+ }
2673
+ }
2674
+
2675
+ if (
2676
+ 'structuredContent' in result &&
2677
+ result.structuredContent !== undefined
2678
+ ) {
2679
+ return {
2680
+ content: jsonStringify(result.structuredContent),
2681
+ type: 'structuredContent',
2682
+ schema: inferCompactSchema(result.structuredContent),
2683
+ }
2684
+ }
2685
+
2686
+ if ('content' in result && Array.isArray(result.content)) {
2687
+ const transformedContent = (
2688
+ await Promise.all(
2689
+ result.content.map(item => transformResultContent(item, name)),
2690
+ )
2691
+ ).flat()
2692
+ return {
2693
+ content: transformedContent,
2694
+ type: 'contentArray',
2695
+ schema: inferCompactSchema(transformedContent),
2696
+ }
2697
+ }
2698
+ }
2699
+
2700
+ const errorMessage = `MCP server "${name}" tool "${tool}": unexpected response format`
2701
+ logMCPError(name, errorMessage)
2702
+ throw new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
2703
+ errorMessage,
2704
+ 'MCP tool unexpected response format',
2705
+ )
2706
+ }
2707
+
2708
+ /**
2709
+ * Check if MCP content contains any image blocks.
2710
+ * Used to decide whether to persist to file (images should use truncation instead
2711
+ * to preserve image compression and viewability).
2712
+ */
2713
+ function contentContainsImages(content: MCPToolResult): boolean {
2714
+ if (!content || typeof content === 'string') {
2715
+ return false
2716
+ }
2717
+ return content.some(block => block.type === 'image')
2718
+ }
2719
+
2720
+ export async function processMCPResult(
2721
+ result: unknown,
2722
+ tool: string, // Tool name for validation (e.g., "search")
2723
+ name: string, // Server name for IDE check and transformation (e.g., "slack")
2724
+ ): Promise<MCPToolResult> {
2725
+ const { content, type, schema } = await transformMCPResult(result, tool, name)
2726
+
2727
+ // IDE tools are not going to the model directly, so we don't need to
2728
+ // handle large output.
2729
+ if (name === 'ide') {
2730
+ return content
2731
+ }
2732
+
2733
+ // Check if content needs truncation (i.e., is too large)
2734
+ if (!(await mcpContentNeedsTruncation(content))) {
2735
+ return content
2736
+ }
2737
+
2738
+ const sizeEstimateTokens = getContentSizeEstimate(content)
2739
+
2740
+ // If large output files feature is disabled, fall back to old truncation behavior
2741
+ if (isEnvDefinedFalsy(process.env.ENABLE_MCP_LARGE_OUTPUT_FILES)) {
2742
+ logEvent('tengu_mcp_large_result_handled', {
2743
+ outcome: 'truncated',
2744
+ reason: 'env_disabled',
2745
+ sizeEstimateTokens,
2746
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2747
+ return await truncateMcpContentIfNeeded(content)
2748
+ }
2749
+
2750
+ // Save large output to file and return instructions for reading it
2751
+ // Content is guaranteed to exist at this point (we checked mcpContentNeedsTruncation)
2752
+ if (!content) {
2753
+ return content
2754
+ }
2755
+
2756
+ // If content contains images, fall back to truncation - persisting images as JSON
2757
+ // defeats the image compression logic and makes them non-viewable
2758
+ if (contentContainsImages(content)) {
2759
+ logEvent('tengu_mcp_large_result_handled', {
2760
+ outcome: 'truncated',
2761
+ reason: 'contains_images',
2762
+ sizeEstimateTokens,
2763
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2764
+ return await truncateMcpContentIfNeeded(content)
2765
+ }
2766
+
2767
+ // Generate a unique ID for the persisted file (server__tool-timestamp)
2768
+ const timestamp = Date.now()
2769
+ const persistId = `mcp-${normalizeNameForMCP(name)}-${normalizeNameForMCP(tool)}-${timestamp}`
2770
+ // Convert to string for persistence (persistToolResult expects string or specific block types)
2771
+ const contentStr =
2772
+ typeof content === 'string' ? content : jsonStringify(content, null, 2)
2773
+ const persistResult = await persistToolResult(contentStr, persistId)
2774
+
2775
+ if (isPersistError(persistResult)) {
2776
+ // If file save failed, fall back to returning truncated content info
2777
+ const contentLength = contentStr.length
2778
+ logEvent('tengu_mcp_large_result_handled', {
2779
+ outcome: 'truncated',
2780
+ reason: 'persist_failed',
2781
+ sizeEstimateTokens,
2782
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2783
+ return `Error: result (${contentLength.toLocaleString()} characters) exceeds maximum allowed tokens. Failed to save output to file: ${persistResult.error}. If this MCP server provides pagination or filtering tools, use them to retrieve specific portions of the data.`
2784
+ }
2785
+
2786
+ logEvent('tengu_mcp_large_result_handled', {
2787
+ outcome: 'persisted',
2788
+ reason: 'file_saved',
2789
+ sizeEstimateTokens,
2790
+ persistedSizeChars: persistResult.originalSize,
2791
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2792
+
2793
+ const formatDescription = getFormatDescription(type, schema)
2794
+ return getLargeOutputInstructions(
2795
+ persistResult.filepath,
2796
+ persistResult.originalSize,
2797
+ formatDescription,
2798
+ )
2799
+ }
2800
+
2801
+ /**
2802
+ * Call an MCP tool, handling UrlElicitationRequiredError (-32042) by
2803
+ * displaying the URL elicitation to the user, waiting for the completion
2804
+ * notification, and retrying the tool call.
2805
+ */
2806
+ type MCPToolCallResult = {
2807
+ content: MCPToolResult
2808
+ _meta?: Record<string, unknown>
2809
+ structuredContent?: Record<string, unknown>
2810
+ }
2811
+
2812
+ /** @internal Exported for testing. */
2813
+ export async function callMCPToolWithUrlElicitationRetry({
2814
+ client: connectedClient,
2815
+ clientConnection,
2816
+ tool,
2817
+ args,
2818
+ meta,
2819
+ signal,
2820
+ setAppState,
2821
+ onProgress,
2822
+ callToolFn = callMCPTool,
2823
+ handleElicitation,
2824
+ }: {
2825
+ client: ConnectedMCPServer
2826
+ clientConnection: MCPServerConnection
2827
+ tool: string
2828
+ args: Record<string, unknown>
2829
+ meta?: Record<string, unknown>
2830
+ signal: AbortSignal
2831
+ setAppState: (f: (prev: AppState) => AppState) => void
2832
+ onProgress?: (data: MCPProgress) => void
2833
+ /** Injectable for testing. Defaults to callMCPTool. */
2834
+ callToolFn?: (opts: {
2835
+ client: ConnectedMCPServer
2836
+ tool: string
2837
+ args: Record<string, unknown>
2838
+ meta?: Record<string, unknown>
2839
+ signal: AbortSignal
2840
+ onProgress?: (data: MCPProgress) => void
2841
+ }) => Promise<MCPToolCallResult>
2842
+ /** Handler for URL elicitations when no hook handles them.
2843
+ * In print/SDK mode, delegates to structuredIO. In REPL, falls back to queue. */
2844
+ handleElicitation?: (
2845
+ serverName: string,
2846
+ params: ElicitRequestURLParams,
2847
+ signal: AbortSignal,
2848
+ ) => Promise<ElicitResult>
2849
+ }): Promise<MCPToolCallResult> {
2850
+ const MAX_URL_ELICITATION_RETRIES = 3
2851
+ for (let attempt = 0; ; attempt++) {
2852
+ try {
2853
+ return await callToolFn({
2854
+ client: connectedClient,
2855
+ tool,
2856
+ args,
2857
+ meta,
2858
+ signal,
2859
+ onProgress,
2860
+ })
2861
+ } catch (error) {
2862
+ // The MCP SDK's Protocol creates plain McpError (not UrlElicitationRequiredError)
2863
+ // for error responses, so we check the error code instead of instanceof.
2864
+ if (
2865
+ !(error instanceof McpError) ||
2866
+ error.code !== ErrorCode.UrlElicitationRequired
2867
+ ) {
2868
+ throw error
2869
+ }
2870
+
2871
+ // Limit the number of URL elicitation retries
2872
+ if (attempt >= MAX_URL_ELICITATION_RETRIES) {
2873
+ throw error
2874
+ }
2875
+
2876
+ const errorData = error.data
2877
+ const rawElicitations =
2878
+ errorData != null &&
2879
+ typeof errorData === 'object' &&
2880
+ 'elicitations' in errorData &&
2881
+ Array.isArray(errorData.elicitations)
2882
+ ? (errorData.elicitations as unknown[])
2883
+ : []
2884
+
2885
+ // Validate each element has the required fields for ElicitRequestURLParams
2886
+ const elicitations = rawElicitations.filter(
2887
+ (e): e is ElicitRequestURLParams => {
2888
+ if (e == null || typeof e !== 'object') return false
2889
+ const obj = e as Record<string, unknown>
2890
+ return (
2891
+ obj.mode === 'url' &&
2892
+ typeof obj.url === 'string' &&
2893
+ typeof obj.elicitationId === 'string' &&
2894
+ typeof obj.message === 'string'
2895
+ )
2896
+ },
2897
+ )
2898
+
2899
+ const serverName =
2900
+ clientConnection.type === 'connected'
2901
+ ? clientConnection.name
2902
+ : 'unknown'
2903
+
2904
+ if (elicitations.length === 0) {
2905
+ logMCPDebug(
2906
+ serverName,
2907
+ `Tool '${tool}' returned -32042 but no valid elicitations in error data`,
2908
+ )
2909
+ throw error
2910
+ }
2911
+
2912
+ logMCPDebug(
2913
+ serverName,
2914
+ `Tool '${tool}' requires URL elicitation (error -32042, attempt ${attempt + 1}), processing ${elicitations.length} elicitation(s)`,
2915
+ )
2916
+
2917
+ // Process each URL elicitation from the error.
2918
+ // The completion notification handler (in registerElicitationHandler) sets
2919
+ // `completed: true` on the matching queue event; the dialog reacts to this flag.
2920
+ for (const elicitation of elicitations) {
2921
+ const { elicitationId } = elicitation
2922
+
2923
+ // Run elicitation hooks — they can resolve URL elicitations programmatically
2924
+ const hookResponse = await runElicitationHooks(
2925
+ serverName,
2926
+ elicitation,
2927
+ signal,
2928
+ )
2929
+ if (hookResponse) {
2930
+ logMCPDebug(
2931
+ serverName,
2932
+ `URL elicitation ${elicitationId} resolved by hook: ${jsonStringify(hookResponse)}`,
2933
+ )
2934
+ if (hookResponse.action !== 'accept') {
2935
+ return {
2936
+ content: `URL elicitation was ${hookResponse.action === 'decline' ? 'declined' : hookResponse.action + 'ed'} by a hook. The tool "${tool}" could not complete because it requires the user to open a URL.`,
2937
+ }
2938
+ }
2939
+ // Hook accepted — skip the UI and proceed to retry
2940
+ continue
2941
+ }
2942
+
2943
+ // Resolve the URL elicitation via callback (print/SDK mode) or queue (REPL mode).
2944
+ let userResult: ElicitResult
2945
+ if (handleElicitation) {
2946
+ // Print/SDK mode: delegate to structuredIO which sends a control request
2947
+ userResult = await handleElicitation(serverName, elicitation, signal)
2948
+ } else {
2949
+ // REPL mode: queue for ElicitationDialog with two-phase consent/waiting flow
2950
+ const waitingState: ElicitationWaitingState = {
2951
+ actionLabel: 'Retry now',
2952
+ showCancel: true,
2953
+ }
2954
+ userResult = await new Promise<ElicitResult>(resolve => {
2955
+ const onAbort = () => {
2956
+ void resolve({ action: 'cancel' })
2957
+ }
2958
+ if (signal.aborted) {
2959
+ onAbort()
2960
+ return
2961
+ }
2962
+ signal.addEventListener('abort', onAbort, { once: true })
2963
+
2964
+ setAppState(prev => ({
2965
+ ...prev,
2966
+ elicitation: {
2967
+ queue: [
2968
+ ...prev.elicitation.queue,
2969
+ {
2970
+ serverName,
2971
+ requestId: `error-elicit-${elicitationId}`,
2972
+ params: elicitation,
2973
+ signal,
2974
+ waitingState,
2975
+ respond: result => {
2976
+ // Phase 1 consent: accept is a no-op (doesn't resolve retry Promise)
2977
+ if (result.action === 'accept') {
2978
+ return
2979
+ }
2980
+ // Decline or cancel: resolve the retry Promise
2981
+ signal.removeEventListener('abort', onAbort)
2982
+ void resolve(result)
2983
+ },
2984
+ onWaitingDismiss: action => {
2985
+ signal.removeEventListener('abort', onAbort)
2986
+ if (action === 'retry') {
2987
+ void resolve({ action: 'accept' })
2988
+ } else {
2989
+ void resolve({ action: 'cancel' })
2990
+ }
2991
+ },
2992
+ },
2993
+ ],
2994
+ },
2995
+ }))
2996
+ })
2997
+ }
2998
+
2999
+ // Run ElicitationResult hooks — they can modify or block the response
3000
+ const finalResult = await runElicitationResultHooks(
3001
+ serverName,
3002
+ userResult,
3003
+ signal,
3004
+ 'url',
3005
+ elicitationId,
3006
+ )
3007
+
3008
+ if (finalResult.action !== 'accept') {
3009
+ logMCPDebug(
3010
+ serverName,
3011
+ `User ${finalResult.action === 'decline' ? 'declined' : finalResult.action + 'ed'} URL elicitation ${elicitationId}`,
3012
+ )
3013
+ return {
3014
+ content: `URL elicitation was ${finalResult.action === 'decline' ? 'declined' : finalResult.action + 'ed'} by the user. The tool "${tool}" could not complete because it requires the user to open a URL.`,
3015
+ }
3016
+ }
3017
+
3018
+ logMCPDebug(
3019
+ serverName,
3020
+ `Elicitation ${elicitationId} completed, retrying tool call`,
3021
+ )
3022
+ }
3023
+
3024
+ // Loop back to retry the tool call
3025
+ }
3026
+ }
3027
+ }
3028
+
3029
+ async function callMCPTool({
3030
+ client: { client, name, config },
3031
+ tool,
3032
+ args,
3033
+ meta,
3034
+ signal,
3035
+ onProgress,
3036
+ }: {
3037
+ client: ConnectedMCPServer
3038
+ tool: string
3039
+ args: Record<string, unknown>
3040
+ meta?: Record<string, unknown>
3041
+ signal: AbortSignal
3042
+ onProgress?: (data: MCPProgress) => void
3043
+ }): Promise<{
3044
+ content: MCPToolResult
3045
+ _meta?: Record<string, unknown>
3046
+ structuredContent?: Record<string, unknown>
3047
+ }> {
3048
+ const toolStartTime = Date.now()
3049
+ let progressInterval: NodeJS.Timeout | undefined
3050
+
3051
+ try {
3052
+ logMCPDebug(name, `Calling MCP tool: ${tool}`)
3053
+
3054
+ // Set up progress logging for long-running tools (every 30 seconds)
3055
+ progressInterval = setInterval(
3056
+ (startTime, name, tool) => {
3057
+ const elapsed = Date.now() - startTime
3058
+ const elapsedSeconds = Math.floor(elapsed / 1000)
3059
+ const duration = `${elapsedSeconds}s`
3060
+ logMCPDebug(name, `Tool '${tool}' still running (${duration} elapsed)`)
3061
+ },
3062
+ 30000, // Log every 30 seconds
3063
+ toolStartTime,
3064
+ name,
3065
+ tool,
3066
+ )
3067
+
3068
+ // Use Promise.race with our own timeout to handle cases where SDK's
3069
+ // internal timeout doesn't work (e.g., SSE stream breaks mid-request)
3070
+ const timeoutMs = getMcpToolTimeoutMs()
3071
+ let timeoutId: NodeJS.Timeout | undefined
3072
+
3073
+ const timeoutPromise = new Promise<never>((_, reject) => {
3074
+ timeoutId = setTimeout(
3075
+ (reject, name, tool, timeoutMs) => {
3076
+ reject(
3077
+ new TelemetrySafeError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
3078
+ `MCP server "${name}" tool "${tool}" timed out after ${Math.floor(timeoutMs / 1000)}s`,
3079
+ 'MCP tool timeout',
3080
+ ),
3081
+ )
3082
+ },
3083
+ timeoutMs,
3084
+ reject,
3085
+ name,
3086
+ tool,
3087
+ timeoutMs,
3088
+ )
3089
+ })
3090
+
3091
+ const result = await Promise.race([
3092
+ client.callTool(
3093
+ {
3094
+ name: tool,
3095
+ arguments: args,
3096
+ _meta: meta,
3097
+ },
3098
+ CallToolResultSchema,
3099
+ {
3100
+ signal,
3101
+ timeout: timeoutMs,
3102
+ onprogress: onProgress
3103
+ ? sdkProgress => {
3104
+ onProgress({
3105
+ type: 'mcp_progress',
3106
+ status: 'progress',
3107
+ serverName: name,
3108
+ toolName: tool,
3109
+ progress: sdkProgress.progress,
3110
+ total: sdkProgress.total,
3111
+ progressMessage: sdkProgress.message,
3112
+ })
3113
+ }
3114
+ : undefined,
3115
+ },
3116
+ ),
3117
+ timeoutPromise,
3118
+ ]).finally(() => {
3119
+ if (timeoutId) {
3120
+ clearTimeout(timeoutId)
3121
+ }
3122
+ })
3123
+
3124
+ if ('isError' in result && result.isError) {
3125
+ let errorDetails = 'Unknown error'
3126
+ if (
3127
+ 'content' in result &&
3128
+ Array.isArray(result.content) &&
3129
+ result.content.length > 0
3130
+ ) {
3131
+ const firstContent = result.content[0]
3132
+ if (
3133
+ firstContent &&
3134
+ typeof firstContent === 'object' &&
3135
+ 'text' in firstContent
3136
+ ) {
3137
+ errorDetails = firstContent.text
3138
+ }
3139
+ } else if ('error' in result) {
3140
+ // Fallback for legacy error format
3141
+ errorDetails = String(result.error)
3142
+ }
3143
+ logMCPError(name, errorDetails)
3144
+ throw new McpToolCallError_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS(
3145
+ errorDetails,
3146
+ 'MCP tool returned error',
3147
+ '_meta' in result && result._meta ? { _meta: result._meta } : undefined,
3148
+ )
3149
+ }
3150
+ const elapsed = Date.now() - toolStartTime
3151
+ const duration =
3152
+ elapsed < 1000
3153
+ ? `${elapsed}ms`
3154
+ : elapsed < 60000
3155
+ ? `${Math.floor(elapsed / 1000)}s`
3156
+ : `${Math.floor(elapsed / 60000)}m ${Math.floor((elapsed % 60000) / 1000)}s`
3157
+
3158
+ logMCPDebug(name, `Tool '${tool}' completed successfully in ${duration}`)
3159
+
3160
+ // Log code indexing tool usage
3161
+ const codeIndexingTool = detectCodeIndexingFromMcpServerName(name)
3162
+ if (codeIndexingTool) {
3163
+ logEvent('tengu_code_indexing_tool_used', {
3164
+ tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3165
+ source:
3166
+ 'mcp' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3167
+ success: true,
3168
+ })
3169
+ }
3170
+
3171
+ const content = await processMCPResult(result, tool, name)
3172
+ return {
3173
+ content,
3174
+ _meta: result._meta as Record<string, unknown> | undefined,
3175
+ structuredContent: result.structuredContent as
3176
+ | Record<string, unknown>
3177
+ | undefined,
3178
+ }
3179
+ } catch (e) {
3180
+ // Clear intervals on error
3181
+ if (progressInterval !== undefined) {
3182
+ clearInterval(progressInterval)
3183
+ }
3184
+
3185
+ const elapsed = Date.now() - toolStartTime
3186
+
3187
+ if (e instanceof Error && e.name !== 'AbortError') {
3188
+ logMCPDebug(
3189
+ name,
3190
+ `Tool '${tool}' failed after ${Math.floor(elapsed / 1000)}s: ${e.message}`,
3191
+ )
3192
+ }
3193
+
3194
+ // Check for 401 errors indicating expired/invalid OAuth tokens
3195
+ // The MCP SDK's StreamableHTTPError has a `code` property with the HTTP status
3196
+ if (e instanceof Error) {
3197
+ const errorCode = 'code' in e ? (e.code as number | undefined) : undefined
3198
+ if (errorCode === 401 || e instanceof UnauthorizedError) {
3199
+ logMCPDebug(
3200
+ name,
3201
+ `Tool call returned 401 Unauthorized - token may have expired`,
3202
+ )
3203
+ logEvent('tengu_mcp_tool_call_auth_error', {})
3204
+ throw new McpAuthError(
3205
+ name,
3206
+ `MCP server "${name}" requires re-authorization (token expired)`,
3207
+ )
3208
+ }
3209
+
3210
+ // Check for session expiry — two error shapes can surface here:
3211
+ // 1. Direct 404 + JSON-RPC -32001 from the server (StreamableHTTPError)
3212
+ // 2. -32000 "Connection closed" (McpError) — the SDK closes the transport
3213
+ // after the onerror handler fires, so the pending callTool() rejects
3214
+ // with this derived error instead of the original 404.
3215
+ // In both cases, clear the connection cache so the next tool call
3216
+ // creates a fresh session.
3217
+ const isSessionExpired = isMcpSessionExpiredError(e)
3218
+ const isConnectionClosedOnHttp =
3219
+ 'code' in e &&
3220
+ (e as Error & { code?: number }).code === -32000 &&
3221
+ e.message.includes('Connection closed') &&
3222
+ (config.type === 'http' || config.type === 'claudeai-proxy')
3223
+ if (isSessionExpired || isConnectionClosedOnHttp) {
3224
+ logMCPDebug(
3225
+ name,
3226
+ `MCP session expired during tool call (${isSessionExpired ? '404/-32001' : 'connection closed'}), clearing connection cache for re-initialization`,
3227
+ )
3228
+ logEvent('tengu_mcp_session_expired', {})
3229
+ await clearServerCache(name, config)
3230
+ throw new McpSessionExpiredError(name)
3231
+ }
3232
+ }
3233
+
3234
+ // When the users hits esc, avoid logspew
3235
+ if (!(e instanceof Error) || e.name !== 'AbortError') {
3236
+ throw e
3237
+ }
3238
+ return { content: undefined }
3239
+ } finally {
3240
+ // Always clear intervals
3241
+ if (progressInterval !== undefined) {
3242
+ clearInterval(progressInterval)
3243
+ }
3244
+ }
3245
+ }
3246
+
3247
+ function extractToolUseId(message: AssistantMessage): string | undefined {
3248
+ if (message.message.content[0]?.type !== 'tool_use') {
3249
+ return undefined
3250
+ }
3251
+ return message.message.content[0].id
3252
+ }
3253
+
3254
+ /**
3255
+ * Sets up SDK MCP clients by creating transports and connecting them.
3256
+ * This is used for SDK MCP servers that run in the same process as the SDK.
3257
+ *
3258
+ * @param sdkMcpConfigs - The SDK MCP server configurations
3259
+ * @param sendMcpMessage - Callback to send MCP messages through the control channel
3260
+ * @returns Connected clients, their tools, and transport map for message routing
3261
+ */
3262
+ export async function setupSdkMcpClients(
3263
+ sdkMcpConfigs: Record<string, McpSdkServerConfig>,
3264
+ sendMcpMessage: (
3265
+ serverName: string,
3266
+ message: JSONRPCMessage,
3267
+ ) => Promise<JSONRPCMessage>,
3268
+ ): Promise<{
3269
+ clients: MCPServerConnection[]
3270
+ tools: Tool[]
3271
+ }> {
3272
+ const clients: MCPServerConnection[] = []
3273
+ const tools: Tool[] = []
3274
+
3275
+ // Connect to all servers in parallel
3276
+ const results = await Promise.allSettled(
3277
+ Object.entries(sdkMcpConfigs).map(async ([name, config]) => {
3278
+ const transport = new SdkControlClientTransport(name, sendMcpMessage)
3279
+
3280
+ const client = new Client(
3281
+ {
3282
+ name: 'claude-code',
3283
+ title: 'Claude Code',
3284
+ version: MACRO.VERSION ?? 'unknown',
3285
+ description: "Anthropic's agentic coding tool",
3286
+ websiteUrl: PRODUCT_URL,
3287
+ },
3288
+ {
3289
+ capabilities: {},
3290
+ },
3291
+ )
3292
+
3293
+ try {
3294
+ // Connect the client
3295
+ await client.connect(transport)
3296
+
3297
+ // Get capabilities from the server
3298
+ const capabilities = client.getServerCapabilities()
3299
+
3300
+ // Create the connected client object
3301
+ const connectedClient: MCPServerConnection = {
3302
+ type: 'connected',
3303
+ name,
3304
+ capabilities: capabilities || {},
3305
+ client,
3306
+ config: { ...config, scope: 'dynamic' as const },
3307
+ cleanup: async () => {
3308
+ await client.close()
3309
+ },
3310
+ }
3311
+
3312
+ // Fetch tools if the server has them
3313
+ const serverTools: Tool[] = []
3314
+ if (capabilities?.tools) {
3315
+ const sdkTools = await fetchToolsForClient(connectedClient)
3316
+ serverTools.push(...sdkTools)
3317
+ }
3318
+
3319
+ return {
3320
+ client: connectedClient,
3321
+ tools: serverTools,
3322
+ }
3323
+ } catch (error) {
3324
+ // If connection fails, return failed server
3325
+ logMCPError(name, `Failed to connect SDK MCP server: ${error}`)
3326
+ return {
3327
+ client: {
3328
+ type: 'failed' as const,
3329
+ name,
3330
+ config: { ...config, scope: 'user' as const },
3331
+ },
3332
+ tools: [],
3333
+ }
3334
+ }
3335
+ }),
3336
+ )
3337
+
3338
+ // Process results and collect clients and tools
3339
+ for (const result of results) {
3340
+ if (result.status === 'fulfilled') {
3341
+ clients.push(result.value.client)
3342
+ tools.push(...result.value.tools)
3343
+ }
3344
+ // If rejected (unexpected), the error was already logged inside the promise
3345
+ }
3346
+
3347
+ return { clients, tools }
3348
+ }