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