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