@within-7/minto 0.4.0 → 0.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (391) hide show
  1. package/dist/Tool.js +7 -0
  2. package/dist/Tool.js.map +2 -2
  3. package/dist/commands/agents/AgentsCommand.js +1 -1
  4. package/dist/commands/agents/AgentsCommand.js.map +2 -2
  5. package/dist/commands/agents/constants.js +2 -2
  6. package/dist/commands/agents/constants.js.map +2 -2
  7. package/dist/commands/clear.js +4 -3
  8. package/dist/commands/clear.js.map +2 -2
  9. package/dist/commands/compact.js +2 -2
  10. package/dist/commands/compact.js.map +1 -1
  11. package/dist/commands/context.js +3 -1
  12. package/dist/commands/context.js.map +2 -2
  13. package/dist/commands/login.js +128 -0
  14. package/dist/commands/login.js.map +7 -0
  15. package/dist/commands/memory.js +33 -82
  16. package/dist/commands/memory.js.map +2 -2
  17. package/dist/commands/quit.js +3 -1
  18. package/dist/commands/quit.js.map +2 -2
  19. package/dist/commands/resume.js +39 -239
  20. package/dist/commands/resume.js.map +2 -2
  21. package/dist/commands/tasks.js +1 -1
  22. package/dist/commands/tasks.js.map +2 -2
  23. package/dist/commands/terminalSetup.js +6 -2
  24. package/dist/commands/terminalSetup.js.map +2 -2
  25. package/dist/commands.js +2 -0
  26. package/dist/commands.js.map +2 -2
  27. package/dist/components/AgentDetailView.js +126 -0
  28. package/dist/components/AgentDetailView.js.map +7 -0
  29. package/dist/components/AgentThinkingBlock.js +1 -1
  30. package/dist/components/AgentThinkingBlock.js.map +2 -2
  31. package/dist/components/AgentViewBanner.js +22 -0
  32. package/dist/components/AgentViewBanner.js.map +7 -0
  33. package/dist/components/HeaderBar.js +1 -1
  34. package/dist/components/HeaderBar.js.map +2 -2
  35. package/dist/components/Help.js +8 -1
  36. package/dist/components/Help.js.map +2 -2
  37. package/dist/components/HotkeyHelpPanel.js +26 -8
  38. package/dist/components/HotkeyHelpPanel.js.map +2 -2
  39. package/dist/components/IdleNotificationBar.js +10 -0
  40. package/dist/components/IdleNotificationBar.js.map +7 -0
  41. package/dist/components/ModelSelector/ModelSelector.js +55 -20
  42. package/dist/components/ModelSelector/ModelSelector.js.map +2 -2
  43. package/dist/components/PromptInput.js +186 -115
  44. package/dist/components/PromptInput.js.map +2 -2
  45. package/dist/components/RewindPanel.js +272 -0
  46. package/dist/components/RewindPanel.js.map +7 -0
  47. package/dist/components/Spinner.js +10 -21
  48. package/dist/components/Spinner.js.map +2 -2
  49. package/dist/components/StreamingTextPreview.js +29 -0
  50. package/dist/components/StreamingTextPreview.js.map +7 -0
  51. package/dist/components/SubagentBlock.js +3 -2
  52. package/dist/components/SubagentBlock.js.map +2 -2
  53. package/dist/components/SubagentProgress.js +4 -4
  54. package/dist/components/SubagentProgress.js.map +2 -2
  55. package/dist/components/TabbedListView/SearchInput.js +1 -1
  56. package/dist/components/TabbedListView/SearchInput.js.map +2 -2
  57. package/dist/components/TabbedListView/TabbedListView.js +87 -41
  58. package/dist/components/TabbedListView/TabbedListView.js.map +2 -2
  59. package/dist/components/TaskCard.js +4 -4
  60. package/dist/components/TaskCard.js.map +2 -2
  61. package/dist/components/TeamMemberPanel.js +107 -0
  62. package/dist/components/TeamMemberPanel.js.map +7 -0
  63. package/dist/components/ThinkingSelector.js +84 -0
  64. package/dist/components/ThinkingSelector.js.map +7 -0
  65. package/dist/components/TitledDivider.js +26 -0
  66. package/dist/components/TitledDivider.js.map +7 -0
  67. package/dist/components/TodoPanel.js +31 -30
  68. package/dist/components/TodoPanel.js.map +2 -2
  69. package/dist/components/TokenWarning.js +28 -7
  70. package/dist/components/TokenWarning.js.map +2 -2
  71. package/dist/components/messages/AssistantTextMessage.js +5 -2
  72. package/dist/components/messages/AssistantTextMessage.js.map +2 -2
  73. package/dist/components/messages/AssistantToolUseMessage.js +9 -1
  74. package/dist/components/messages/AssistantToolUseMessage.js.map +2 -2
  75. package/dist/components/messages/DefaultToolResultFallback.js +11 -0
  76. package/dist/components/messages/DefaultToolResultFallback.js.map +7 -0
  77. package/dist/components/messages/ParallelTasksGroupView.js +14 -6
  78. package/dist/components/messages/ParallelTasksGroupView.js.map +2 -2
  79. package/dist/components/messages/TaskInModuleView.js +27 -27
  80. package/dist/components/messages/TaskInModuleView.js.map +2 -2
  81. package/dist/components/messages/UserGuidanceMessage.js +26 -0
  82. package/dist/components/messages/UserGuidanceMessage.js.map +7 -0
  83. package/dist/components/messages/UserPromptMessage.js +2 -1
  84. package/dist/components/messages/UserPromptMessage.js.map +2 -2
  85. package/dist/components/messages/UserTeamNotificationMessage.js +91 -0
  86. package/dist/components/messages/UserTeamNotificationMessage.js.map +7 -0
  87. package/dist/components/messages/UserTextMessage.js +8 -0
  88. package/dist/components/messages/UserTextMessage.js.map +2 -2
  89. package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js +4 -2
  90. package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js.map +2 -2
  91. package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js +18 -1
  92. package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js.map +2 -2
  93. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +12 -1
  94. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +2 -2
  95. package/dist/components/permissions/PermissionRequest.js +4 -0
  96. package/dist/components/permissions/PermissionRequest.js.map +2 -2
  97. package/dist/components/permissions/PlanApprovalRequest.js +164 -0
  98. package/dist/components/permissions/PlanApprovalRequest.js.map +7 -0
  99. package/dist/constants/agentTeams.js +17 -0
  100. package/dist/constants/agentTeams.js.map +7 -0
  101. package/dist/constants/macros.js +2 -1
  102. package/dist/constants/macros.js.map +2 -2
  103. package/dist/constants/prompts/agentPrompt.js +1 -0
  104. package/dist/constants/prompts/agentPrompt.js.map +2 -2
  105. package/dist/constants/prompts/autoMemory.js +39 -0
  106. package/dist/constants/prompts/autoMemory.js.map +7 -0
  107. package/dist/constants/prompts/codeConventions.js +1 -13
  108. package/dist/constants/prompts/codeConventions.js.map +2 -2
  109. package/dist/constants/prompts/doingTasks.js +21 -2
  110. package/dist/constants/prompts/doingTasks.js.map +2 -2
  111. package/dist/constants/prompts/envInfo.js +6 -7
  112. package/dist/constants/prompts/envInfo.js.map +2 -2
  113. package/dist/constants/prompts/index.js +27 -5
  114. package/dist/constants/prompts/index.js.map +2 -2
  115. package/dist/constants/prompts/taskManagement.js +2 -43
  116. package/dist/constants/prompts/taskManagement.js.map +2 -2
  117. package/dist/constants/prompts/teamOverlays.js +50 -0
  118. package/dist/constants/prompts/teamOverlays.js.map +7 -0
  119. package/dist/constants/prompts/toneAndStyle.js +4 -29
  120. package/dist/constants/prompts/toneAndStyle.js.map +2 -2
  121. package/dist/constants/prompts/toolUsagePolicy.js +7 -22
  122. package/dist/constants/prompts/toolUsagePolicy.js.map +2 -2
  123. package/dist/constants/toolInputExamples.js +2 -2
  124. package/dist/constants/toolInputExamples.js.map +2 -2
  125. package/dist/context.js +39 -6
  126. package/dist/context.js.map +2 -2
  127. package/dist/core/backupManager.js +1 -1
  128. package/dist/core/backupManager.js.map +2 -2
  129. package/dist/core/permissions/rules/planModeRule.js +1 -1
  130. package/dist/core/permissions/rules/planModeRule.js.map +1 -1
  131. package/dist/core/permissions/rules/safeModeRule.js +1 -1
  132. package/dist/core/permissions/rules/safeModeRule.js.map +1 -1
  133. package/dist/engine/AgentEngine.js +902 -0
  134. package/dist/engine/AgentEngine.js.map +7 -0
  135. package/dist/engine/EngineRegistry.js +89 -0
  136. package/dist/engine/EngineRegistry.js.map +7 -0
  137. package/dist/engine/foregroundAdapter.js +191 -0
  138. package/dist/engine/foregroundAdapter.js.map +7 -0
  139. package/dist/engine/index.js +15 -0
  140. package/dist/engine/index.js.map +7 -0
  141. package/dist/engine/types.js +1 -0
  142. package/dist/engine/types.js.map +7 -0
  143. package/dist/entrypoints/cli.js +410 -79
  144. package/dist/entrypoints/cli.js.map +3 -3
  145. package/dist/hooks/useAgentEngine.js +129 -0
  146. package/dist/hooks/useAgentEngine.js.map +7 -0
  147. package/dist/hooks/useAgentTokenStats.js +0 -16
  148. package/dist/hooks/useAgentTokenStats.js.map +2 -2
  149. package/dist/hooks/useCanUseTool.js +47 -2
  150. package/dist/hooks/useCanUseTool.js.map +2 -2
  151. package/dist/hooks/useDeferredLoading.js +4 -1
  152. package/dist/hooks/useDeferredLoading.js.map +2 -2
  153. package/dist/hooks/useIdleNotifications.js +66 -0
  154. package/dist/hooks/useIdleNotifications.js.map +7 -0
  155. package/dist/hooks/useSessionTracking.js +9 -7
  156. package/dist/hooks/useSessionTracking.js.map +2 -2
  157. package/dist/hooks/useTeamMembers.js +51 -0
  158. package/dist/hooks/useTeamMembers.js.map +7 -0
  159. package/dist/i18n/locales/en.js +77 -12
  160. package/dist/i18n/locales/en.js.map +2 -2
  161. package/dist/i18n/locales/zh-CN.js +77 -12
  162. package/dist/i18n/locales/zh-CN.js.map +2 -2
  163. package/dist/i18n/types.js.map +1 -1
  164. package/dist/messages.js.map +2 -2
  165. package/dist/permissions.js +113 -7
  166. package/dist/permissions.js.map +2 -2
  167. package/dist/query.js +135 -37
  168. package/dist/query.js.map +2 -2
  169. package/dist/screens/REPL.js +504 -361
  170. package/dist/screens/REPL.js.map +3 -3
  171. package/dist/screens/ResumeConversation.js +199 -14
  172. package/dist/screens/ResumeConversation.js.map +2 -2
  173. package/dist/services/adapters/base.js.map +1 -1
  174. package/dist/services/agentTeams/backends/headless.js +108 -0
  175. package/dist/services/agentTeams/backends/headless.js.map +7 -0
  176. package/dist/services/agentTeams/backends/inProcess.js +102 -0
  177. package/dist/services/agentTeams/backends/inProcess.js.map +7 -0
  178. package/dist/services/agentTeams/backends/resolver.js +18 -0
  179. package/dist/services/agentTeams/backends/resolver.js.map +7 -0
  180. package/dist/services/agentTeams/backends/tmux.js +168 -0
  181. package/dist/services/agentTeams/backends/tmux.js.map +7 -0
  182. package/dist/services/agentTeams/backends/types.js +1 -0
  183. package/dist/services/agentTeams/backends/types.js.map +7 -0
  184. package/dist/services/agentTeams/heartbeat.js +88 -0
  185. package/dist/services/agentTeams/heartbeat.js.map +7 -0
  186. package/dist/services/agentTeams/index.js +42 -2
  187. package/dist/services/agentTeams/index.js.map +2 -2
  188. package/dist/services/agentTeams/injectionChannel.js +105 -0
  189. package/dist/services/agentTeams/injectionChannel.js.map +7 -0
  190. package/dist/services/agentTeams/mailbox.js +410 -30
  191. package/dist/services/agentTeams/mailbox.js.map +2 -2
  192. package/dist/services/agentTeams/messageFormatter.js +80 -0
  193. package/dist/services/agentTeams/messageFormatter.js.map +7 -0
  194. package/dist/services/agentTeams/permissionDelegation.js +71 -0
  195. package/dist/services/agentTeams/permissionDelegation.js.map +7 -0
  196. package/dist/services/agentTeams/teamEvents.js +45 -0
  197. package/dist/services/agentTeams/teamEvents.js.map +7 -0
  198. package/dist/services/agentTeams/teamManager.js +251 -34
  199. package/dist/services/agentTeams/teamManager.js.map +2 -2
  200. package/dist/services/agentTeams/teamTaskStore.js +290 -61
  201. package/dist/services/agentTeams/teamTaskStore.js.map +2 -2
  202. package/dist/services/agentTeams/teammateSpawner.js +99 -18
  203. package/dist/services/agentTeams/teammateSpawner.js.map +2 -2
  204. package/dist/services/hookExecutor.js +51 -8
  205. package/dist/services/hookExecutor.js.map +2 -2
  206. package/dist/services/llm/anthropicProvider.js +56 -59
  207. package/dist/services/llm/anthropicProvider.js.map +2 -2
  208. package/dist/services/llm/dispatch.js +24 -5
  209. package/dist/services/llm/dispatch.js.map +2 -2
  210. package/dist/services/llm/openaiProvider.js +115 -136
  211. package/dist/services/llm/openaiProvider.js.map +3 -3
  212. package/dist/services/llm/types.js +89 -15
  213. package/dist/services/llm/types.js.map +2 -2
  214. package/dist/services/mcpClient.js +80 -4
  215. package/dist/services/mcpClient.js.map +2 -2
  216. package/dist/services/mintoAuth.js +299 -0
  217. package/dist/services/mintoAuth.js.map +7 -0
  218. package/dist/services/oauth.js +3 -3
  219. package/dist/services/oauth.js.map +2 -2
  220. package/dist/services/openai.js +91 -20
  221. package/dist/services/openai.js.map +2 -2
  222. package/dist/services/plugins/pluginRuntime.js +11 -5
  223. package/dist/services/plugins/pluginRuntime.js.map +2 -2
  224. package/dist/services/plugins/pluginValidation.js +4 -2
  225. package/dist/services/plugins/pluginValidation.js.map +2 -2
  226. package/dist/services/sandbox/sandboxController.js +11 -3
  227. package/dist/services/sandbox/sandboxController.js.map +2 -2
  228. package/dist/services/sessionMemoryInjector.js +77 -0
  229. package/dist/services/sessionMemoryInjector.js.map +7 -0
  230. package/dist/services/systemReminder.js +130 -8
  231. package/dist/services/systemReminder.js.map +2 -2
  232. package/dist/services/taskStore.js +199 -8
  233. package/dist/services/taskStore.js.map +3 -3
  234. package/dist/services/topicDetector.js +169 -0
  235. package/dist/services/topicDetector.js.map +7 -0
  236. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +0 -13
  237. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +2 -2
  238. package/dist/tools/BashTool/BashTool.js +51 -28
  239. package/dist/tools/BashTool/BashTool.js.map +2 -2
  240. package/dist/tools/BashTool/prompt.js +95 -118
  241. package/dist/tools/BashTool/prompt.js.map +2 -2
  242. package/dist/tools/BashTool/utils.js +39 -1
  243. package/dist/tools/BashTool/utils.js.map +2 -2
  244. package/dist/tools/EnterWorktreeTool/EnterWorktreeTool.js +121 -0
  245. package/dist/tools/EnterWorktreeTool/EnterWorktreeTool.js.map +7 -0
  246. package/dist/tools/EnterWorktreeTool/prompt.js +22 -0
  247. package/dist/tools/EnterWorktreeTool/prompt.js.map +7 -0
  248. package/dist/tools/FileEditTool/FileEditTool.js +9 -4
  249. package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
  250. package/dist/tools/FileEditTool/prompt.js +3 -7
  251. package/dist/tools/FileEditTool/prompt.js.map +2 -2
  252. package/dist/tools/FileReadTool/FileReadTool.js +125 -3
  253. package/dist/tools/FileReadTool/FileReadTool.js.map +2 -2
  254. package/dist/tools/FileReadTool/prompt.js +1 -2
  255. package/dist/tools/FileReadTool/prompt.js.map +2 -2
  256. package/dist/tools/FileWriteTool/prompt.js +3 -5
  257. package/dist/tools/FileWriteTool/prompt.js.map +2 -2
  258. package/dist/tools/GlobTool/GlobTool.js +3 -2
  259. package/dist/tools/GlobTool/GlobTool.js.map +2 -2
  260. package/dist/tools/GrepTool/GrepTool.js +16 -5
  261. package/dist/tools/GrepTool/GrepTool.js.map +2 -2
  262. package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js.map +2 -2
  263. package/dist/tools/MCPSearchTool/MCPSearchTool.js +172 -0
  264. package/dist/tools/MCPSearchTool/MCPSearchTool.js.map +7 -0
  265. package/dist/tools/MCPSearchTool/prompt.js +77 -0
  266. package/dist/tools/MCPSearchTool/prompt.js.map +7 -0
  267. package/dist/tools/MultiEditTool/prompt.js +4 -7
  268. package/dist/tools/MultiEditTool/prompt.js.map +2 -2
  269. package/dist/tools/PlanModeTool/EnterPlanModeTool.js +12 -8
  270. package/dist/tools/PlanModeTool/EnterPlanModeTool.js.map +2 -2
  271. package/dist/tools/PlanModeTool/ExitPlanModeTool.js +54 -1
  272. package/dist/tools/PlanModeTool/ExitPlanModeTool.js.map +2 -2
  273. package/dist/tools/PlanModeTool/prompt.js +23 -74
  274. package/dist/tools/PlanModeTool/prompt.js.map +2 -2
  275. package/dist/tools/SendMessageTool/SendMessageTool.js +341 -0
  276. package/dist/tools/SendMessageTool/SendMessageTool.js.map +7 -0
  277. package/dist/tools/SendMessageTool/prompt.js +44 -0
  278. package/dist/tools/SendMessageTool/prompt.js.map +7 -0
  279. package/dist/tools/TaskCreateTool/prompt.js +15 -4
  280. package/dist/tools/TaskCreateTool/prompt.js.map +2 -2
  281. package/dist/tools/TaskListTool/prompt.js +18 -3
  282. package/dist/tools/TaskListTool/prompt.js.map +2 -2
  283. package/dist/tools/TaskOutputTool/prompt.js +4 -3
  284. package/dist/tools/TaskOutputTool/prompt.js.map +2 -2
  285. package/dist/tools/TaskTool/TaskTool.js +762 -98
  286. package/dist/tools/TaskTool/TaskTool.js.map +3 -3
  287. package/dist/tools/TaskTool/constants.js +8 -2
  288. package/dist/tools/TaskTool/constants.js.map +2 -2
  289. package/dist/tools/TaskTool/prompt.js +74 -70
  290. package/dist/tools/TaskTool/prompt.js.map +2 -2
  291. package/dist/tools/TaskUpdateTool/TaskUpdateTool.js +15 -1
  292. package/dist/tools/TaskUpdateTool/TaskUpdateTool.js.map +2 -2
  293. package/dist/tools/TeamCreateTool/TeamCreateTool.js +129 -0
  294. package/dist/tools/TeamCreateTool/TeamCreateTool.js.map +7 -0
  295. package/dist/tools/TeamCreateTool/prompt.js +58 -0
  296. package/dist/tools/TeamCreateTool/prompt.js.map +7 -0
  297. package/dist/tools/TeamDeleteTool/TeamDeleteTool.js +151 -0
  298. package/dist/tools/TeamDeleteTool/TeamDeleteTool.js.map +7 -0
  299. package/dist/tools/TeamDeleteTool/prompt.js +16 -0
  300. package/dist/tools/TeamDeleteTool/prompt.js.map +7 -0
  301. package/dist/tools/URLFetcherTool/URLFetcherTool.js +106 -15
  302. package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +2 -2
  303. package/dist/tools/URLFetcherTool/prompt.js +3 -2
  304. package/dist/tools/URLFetcherTool/prompt.js.map +2 -2
  305. package/dist/tools/WebSearchTool/WebSearchTool.js +2 -1
  306. package/dist/tools/WebSearchTool/WebSearchTool.js.map +2 -2
  307. package/dist/tools/WebSearchTool/prompt.js +5 -4
  308. package/dist/tools/WebSearchTool/prompt.js.map +2 -2
  309. package/dist/tools.js +100 -20
  310. package/dist/tools.js.map +2 -2
  311. package/dist/types/PermissionMode.js +35 -6
  312. package/dist/types/PermissionMode.js.map +2 -2
  313. package/dist/types/hooks.js +2 -0
  314. package/dist/types/hooks.js.map +2 -2
  315. package/dist/types/plugin.js +2 -0
  316. package/dist/types/plugin.js.map +3 -3
  317. package/dist/utils/CircuitBreaker.js +15 -9
  318. package/dist/utils/CircuitBreaker.js.map +2 -2
  319. package/dist/utils/agentLoader.js +249 -112
  320. package/dist/utils/agentLoader.js.map +2 -2
  321. package/dist/utils/animationManager.js +40 -3
  322. package/dist/utils/animationManager.js.map +2 -2
  323. package/dist/utils/ask.js +7 -6
  324. package/dist/utils/ask.js.map +2 -2
  325. package/dist/utils/atomicWrite.js +23 -0
  326. package/dist/utils/atomicWrite.js.map +7 -0
  327. package/dist/utils/autoCompactCore.js +73 -56
  328. package/dist/utils/autoCompactCore.js.map +2 -2
  329. package/dist/utils/autoMemoryPaths.js +89 -0
  330. package/dist/utils/autoMemoryPaths.js.map +7 -0
  331. package/dist/utils/config.js +63 -38
  332. package/dist/utils/config.js.map +2 -2
  333. package/dist/utils/configSchema.js +13 -8
  334. package/dist/utils/configSchema.js.map +2 -2
  335. package/dist/utils/credentials/index.js +14 -0
  336. package/dist/utils/credentials/index.js.map +2 -2
  337. package/dist/utils/dualPath.js +24 -0
  338. package/dist/utils/dualPath.js.map +7 -0
  339. package/dist/utils/exit.js +66 -7
  340. package/dist/utils/exit.js.map +2 -2
  341. package/dist/utils/externalEditor.js +155 -0
  342. package/dist/utils/externalEditor.js.map +7 -0
  343. package/dist/utils/fileLock.js +67 -0
  344. package/dist/utils/fileLock.js.map +7 -0
  345. package/dist/utils/format.js +24 -14
  346. package/dist/utils/format.js.map +2 -2
  347. package/dist/utils/globalErrorHandler.js +5 -96
  348. package/dist/utils/globalErrorHandler.js.map +3 -3
  349. package/dist/utils/groupHandlers/parallelTasksHandler.js +5 -3
  350. package/dist/utils/groupHandlers/parallelTasksHandler.js.map +2 -2
  351. package/dist/utils/groupHandlers/taskHandler.js +2 -2
  352. package/dist/utils/groupHandlers/taskHandler.js.map +2 -2
  353. package/dist/utils/hookManager.js +64 -6
  354. package/dist/utils/hookManager.js.map +2 -2
  355. package/dist/utils/log.js +6 -2
  356. package/dist/utils/log.js.map +2 -2
  357. package/dist/utils/markdown.js +237 -19
  358. package/dist/utils/markdown.js.map +2 -2
  359. package/dist/utils/messageContextManager.js +18 -5
  360. package/dist/utils/messageContextManager.js.map +2 -2
  361. package/dist/utils/messageGroupManager.js +1 -1
  362. package/dist/utils/messageGroupManager.js.map +2 -2
  363. package/dist/utils/messages.js +104 -46
  364. package/dist/utils/messages.js.map +2 -2
  365. package/dist/utils/model.js +2 -2
  366. package/dist/utils/model.js.map +2 -2
  367. package/dist/utils/pasteCache.js +8 -4
  368. package/dist/utils/pasteCache.js.map +2 -2
  369. package/dist/utils/pluginLoader.js +18 -0
  370. package/dist/utils/pluginLoader.js.map +2 -2
  371. package/dist/utils/secureKeyStorage.js +36 -7
  372. package/dist/utils/secureKeyStorage.js.map +2 -2
  373. package/dist/utils/simpleMode.js +7 -0
  374. package/dist/utils/simpleMode.js.map +7 -0
  375. package/dist/utils/streamingState.js +11 -1
  376. package/dist/utils/streamingState.js.map +2 -2
  377. package/dist/utils/taskDisplayUtils.js +2 -1
  378. package/dist/utils/taskDisplayUtils.js.map +2 -2
  379. package/dist/utils/teamConfig.js +2 -2
  380. package/dist/utils/teamConfig.js.map +2 -2
  381. package/dist/utils/thinking.js +6 -2
  382. package/dist/utils/thinking.js.map +3 -3
  383. package/dist/utils/tokenProgress.js +55 -0
  384. package/dist/utils/tokenProgress.js.map +7 -0
  385. package/dist/utils/toolRiskClassification.js +26 -17
  386. package/dist/utils/toolRiskClassification.js.map +2 -2
  387. package/dist/utils/tooling/toolError.js +12 -0
  388. package/dist/utils/tooling/toolError.js.map +7 -0
  389. package/dist/version.js +2 -2
  390. package/dist/version.js.map +1 -1
  391. package/package.json +10 -8
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/services/mcpClient.ts"],
4
- "sourcesContent": ["import { zipObject } from 'lodash-es'\nimport { debug as debugLogger } from '@utils/debugLogger'\nimport {\n getCurrentProjectConfig,\n McpServerConfig,\n saveCurrentProjectConfig,\n getGlobalConfig,\n saveGlobalConfig,\n getMcprcConfig,\n addMcprcServerForTesting,\n removeMcprcServerForTesting,\n} from '@utils/config'\nimport { existsSync, readFileSync, writeFileSync } from 'fs'\nimport { join } from 'path'\nimport { getCwd } from '@utils/state'\nimport { safeParseJSON } from '@utils/json'\nimport {\n ImageBlockParam,\n MessageParam,\n ToolResultBlockParam,\n} from '@anthropic-ai/sdk/resources/index.mjs'\nimport { Client } from '@modelcontextprotocol/sdk/client/index.js'\nimport { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'\nimport { SSEClientTransport } from '@modelcontextprotocol/sdk/client/sse.js'\nimport {\n CallToolResultSchema,\n ClientRequest,\n ListPromptsResult,\n ListPromptsResultSchema,\n ListToolsResult,\n ListToolsResultSchema,\n ListResourcesResult,\n ListResourcesResultSchema,\n ReadResourceResult,\n ReadResourceResultSchema,\n Result,\n ResultSchema,\n ToolListChangedNotificationSchema,\n} from '@modelcontextprotocol/sdk/types.js'\nimport { memoize, pickBy } from 'lodash-es'\nimport type { Tool } from '@tool'\nimport { MCPTool } from '@tools/MCPTool/MCPTool'\nimport { logMCPError } from '@utils/log'\nimport { ensureProxyBypass } from '@utils/envConfig'\nimport { Command } from '@commands'\nimport { PRODUCT_COMMAND } from '@constants/product'\nimport { CircuitBreakerRegistry, CircuitOpenError } from '@utils/CircuitBreaker'\nimport { emitReminderEvent } from '@services/systemReminder'\n\ntype McpName = string\n\n/**\n * Circuit breaker registry for MCP server connections\n * Each server gets its own circuit breaker to prevent cascading failures\n */\n/**\n * Env var prefixes/names to strip when spawning MCP server processes.\n * Prevents leaking API keys and secrets to third-party servers.\n * Explicit per-server `env` values in config still override (intentional).\n */\nconst SENSITIVE_ENV_PREFIXES = [\n 'ANTHROPIC_',\n 'OPENAI_',\n 'AWS_SECRET',\n 'GITHUB_TOKEN',\n 'GH_TOKEN',\n 'MINTO_API',\n]\n\nfunction filterSensitiveEnv(env: NodeJS.ProcessEnv): Record<string, string> {\n return Object.fromEntries(\n Object.entries(env)\n .filter(\n ([k]) => !SENSITIVE_ENV_PREFIXES.some(p => k.startsWith(p) || k === p),\n )\n .filter((e): e is [string, string] => e[1] !== undefined),\n )\n}\n\nconst mcpCircuitBreakers = new CircuitBreakerRegistry({\n failureThreshold: 5,\n successThreshold: 2,\n openTimeout: 30000, // 30 seconds before trying again\n})\n\nexport function parseEnvVars(\n rawEnvArgs: string[] | undefined,\n): Record<string, string> {\n const parsedEnv: Record<string, string> = {}\n\n // Parse individual env vars\n if (rawEnvArgs) {\n for (const envStr of rawEnvArgs) {\n const [key, ...valueParts] = envStr.split('=')\n if (!key || valueParts.length === 0) {\n throw new Error(\n `Invalid environment variable format: ${envStr}, environment variables should be added as: -e KEY1=value1 -e KEY2=value2`,\n )\n }\n parsedEnv[key] = valueParts.join('=')\n }\n }\n return parsedEnv\n}\n\nconst VALID_SCOPES = ['project', 'global', 'mcprc'] as const\ntype ConfigScope = (typeof VALID_SCOPES)[number]\nconst EXTERNAL_SCOPES = ['project', 'global'] as ConfigScope[]\n\nexport function ensureConfigScope(scope?: string): ConfigScope {\n if (!scope) return 'project'\n\n const scopesToCheck =\n process.env.USER_TYPE === 'external' ? EXTERNAL_SCOPES : VALID_SCOPES\n\n if (!scopesToCheck.includes(scope as ConfigScope)) {\n throw new Error(\n `Invalid scope: ${scope}. Must be one of: ${scopesToCheck.join(', ')}`,\n )\n }\n\n return scope as ConfigScope\n}\n\nexport function addMcpServer(\n name: McpName,\n server: McpServerConfig,\n scope: ConfigScope = 'project',\n): void {\n if (scope === 'mcprc') {\n if (process.env.NODE_ENV === 'test') {\n addMcprcServerForTesting(name, server)\n } else {\n const mcprcPath = join(getCwd(), '.mcprc')\n let mcprcConfig: Record<string, McpServerConfig> = {}\n\n // Read existing config if present\n if (existsSync(mcprcPath)) {\n try {\n const mcprcContent = readFileSync(mcprcPath, 'utf-8')\n const existingConfig = safeParseJSON(mcprcContent)\n if (existingConfig && typeof existingConfig === 'object') {\n mcprcConfig = existingConfig as Record<string, McpServerConfig>\n }\n } catch (e) {\n debugLogger.warn(\n 'MCP_CONFIG',\n `Failed to read/parse .mcprc: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }\n\n // Add the server\n mcprcConfig[name] = server\n\n // Write back to .mcprc\n try {\n writeFileSync(mcprcPath, JSON.stringify(mcprcConfig, null, 2), 'utf-8')\n } catch (error) {\n throw new Error(`Failed to write to .mcprc: ${error}`)\n }\n }\n } else if (scope === 'global') {\n const config = getGlobalConfig()\n if (!config.mcpServers) {\n config.mcpServers = {}\n }\n config.mcpServers[name] = server\n saveGlobalConfig(config)\n } else {\n const config = getCurrentProjectConfig()\n if (!config.mcpServers) {\n config.mcpServers = {}\n }\n config.mcpServers[name] = server\n saveCurrentProjectConfig(config)\n }\n}\n\nexport function removeMcpServer(\n name: McpName,\n scope: ConfigScope = 'project',\n): void {\n if (scope === 'mcprc') {\n if (process.env.NODE_ENV === 'test') {\n removeMcprcServerForTesting(name)\n } else {\n const mcprcPath = join(getCwd(), '.mcprc')\n if (!existsSync(mcprcPath)) {\n throw new Error('No .mcprc file found in this directory')\n }\n\n try {\n const mcprcContent = readFileSync(mcprcPath, 'utf-8')\n const mcprcConfig = safeParseJSON(mcprcContent) as Record<\n string,\n McpServerConfig\n > | null\n\n if (\n !mcprcConfig ||\n typeof mcprcConfig !== 'object' ||\n !mcprcConfig[name]\n ) {\n throw new Error(`No MCP server found with name: ${name} in .mcprc`)\n }\n\n delete mcprcConfig[name]\n writeFileSync(mcprcPath, JSON.stringify(mcprcConfig, null, 2), 'utf-8')\n } catch (error) {\n if (error instanceof Error) {\n throw error\n }\n throw new Error(`Failed to remove from .mcprc: ${error}`)\n }\n }\n } else if (scope === 'global') {\n const config = getGlobalConfig()\n if (!config.mcpServers?.[name]) {\n throw new Error(`No global MCP server found with name: ${name}`)\n }\n delete config.mcpServers[name]\n saveGlobalConfig(config)\n } else {\n const config = getCurrentProjectConfig()\n if (!config.mcpServers?.[name]) {\n throw new Error(`No local MCP server found with name: ${name}`)\n }\n delete config.mcpServers[name]\n saveCurrentProjectConfig(config)\n }\n}\n\nexport function listMCPServers(): Record<string, McpServerConfig> {\n const globalConfig = getGlobalConfig()\n const mcprcConfig = getMcprcConfig()\n const projectConfig = getCurrentProjectConfig()\n return {\n ...(globalConfig.mcpServers ?? {}),\n ...(mcprcConfig ?? {}), // mcprc configs override global ones\n ...(projectConfig.mcpServers ?? {}), // Project configs override mcprc ones\n }\n}\n\nexport type ScopedMcpServerConfig = McpServerConfig & {\n scope: ConfigScope\n}\n\nexport function getMcpServer(name: McpName): ScopedMcpServerConfig | undefined {\n const projectConfig = getCurrentProjectConfig()\n const mcprcConfig = getMcprcConfig()\n const globalConfig = getGlobalConfig()\n\n // Check each scope in order of precedence\n if (projectConfig.mcpServers?.[name]) {\n return { ...projectConfig.mcpServers[name], scope: 'project' }\n }\n\n if (mcprcConfig?.[name]) {\n return { ...mcprcConfig[name], scope: 'mcprc' }\n }\n\n if (globalConfig.mcpServers?.[name]) {\n return { ...globalConfig.mcpServers[name], scope: 'global' }\n }\n\n return undefined\n}\n\nasync function connectToServer(\n name: string,\n serverRef: McpServerConfig,\n): Promise<Client> {\n let transport\n\n if (serverRef.type === 'http') {\n // Streamable HTTP transport (CC spec)\n const { StreamableHTTPClientTransport } = await import(\n '@modelcontextprotocol/sdk/client/streamableHttp.js'\n )\n transport = new StreamableHTTPClientTransport(\n new URL((serverRef as any).url),\n {\n // Pass headers but never log sensitive values\n requestInit: (serverRef as any).headers\n ? { headers: (serverRef as any).headers }\n : undefined,\n },\n )\n } else if (serverRef.type === 'sse') {\n transport = new SSEClientTransport(new URL(serverRef.url))\n } else {\n transport = new StdioClientTransport({\n command: serverRef.command,\n args: serverRef.args,\n env: ensureProxyBypass({\n ...filterSensitiveEnv(process.env),\n ...serverRef.env,\n } as Record<string, string>),\n stderr: 'pipe', // prevents error output from the MCP server from printing to the UI\n })\n }\n\n const client = new Client(\n {\n name: PRODUCT_COMMAND,\n version: '0.1.0',\n },\n {\n capabilities: {},\n },\n )\n\n // Add a timeout to connection attempts to prevent tests from hanging indefinitely\n const connectionTimeout = poolConfig.connectionTimeout\n const connectPromise = client.connect(transport)\n const timeoutPromise = new Promise<never>((_, reject) => {\n const timeoutId = setTimeout(() => {\n reject(\n new Error(\n `Connection to MCP server \"${name}\" timed out after ${connectionTimeout}ms`,\n ),\n )\n }, connectionTimeout)\n\n // Clean up timeout if connect resolves or rejects\n connectPromise.then(\n () => clearTimeout(timeoutId),\n () => clearTimeout(timeoutId),\n )\n })\n\n await Promise.race([connectPromise, timeoutPromise])\n\n if (serverRef.type === 'stdio') {\n ;(transport as StdioClientTransport).stderr?.on('data', (data: Buffer) => {\n const errorText = data.toString().trim()\n if (errorText) {\n logMCPError(name, `Server stderr: ${errorText}`)\n }\n })\n }\n\n // Listen for tools/list_changed notification to dynamically update tool lists\n client.setNotificationHandler(ToolListChangedNotificationSchema, async () => {\n // Clear cached tools so next getMCPTools() fetches fresh list\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n })\n\n return client\n}\n\ntype ConnectedClient = {\n client: Client\n name: string\n type: 'connected'\n}\ntype FailedClient = {\n name: string\n type: 'failed'\n}\nexport type WrappedClient = ConnectedClient | FailedClient\n\nexport function getMcprcServerStatus(\n serverName: string,\n): 'approved' | 'rejected' | 'pending' {\n const config = getCurrentProjectConfig()\n if (config.approvedMcprcServers?.includes(serverName)) {\n return 'approved'\n }\n if (config.rejectedMcprcServers?.includes(serverName)) {\n return 'rejected'\n }\n return 'pending'\n}\n\n// Store connected clients for cleanup on shutdown\nlet connectedClients: WrappedClient[] = []\n\n// Connection pool configuration\nexport interface McpConnectionPoolConfig {\n /** Maximum concurrent connections (default: 10) */\n maxConnections: number\n /** Connection timeout in ms (default: 10000) */\n connectionTimeout: number\n /** Health check interval in ms (default: 30000, 0 to disable) */\n healthCheckInterval: number\n /** Retry failed connections (default: true) */\n retryFailed: boolean\n /** Max retry attempts (default: 3) */\n maxRetries: number\n /** Maximum output size from MCP tool calls in characters (default: 25000) */\n maxOutputChars: number\n}\n\nconst DEFAULT_POOL_CONFIG: McpConnectionPoolConfig = {\n maxConnections: 10,\n connectionTimeout: Number(process.env.MCP_TIMEOUT) || 10000,\n healthCheckInterval: 30000,\n retryFailed: true,\n maxRetries: 2,\n maxOutputChars: Number(process.env.MAX_MCP_OUTPUT_CHARS) || 25000,\n}\n\nlet poolConfig: McpConnectionPoolConfig = { ...DEFAULT_POOL_CONFIG }\nlet healthCheckTimer: ReturnType<typeof setInterval> | null = null\n\n// Connection lock mechanism to prevent concurrent refreshMCPConnections calls\nlet connectionLock: Promise<void> | null = null\nlet pendingRefresh = false\n\n/**\n * Acquires the connection lock for exclusive access to MCP connection refresh\n * Waits until any existing lock is released before creating a new one\n */\nasync function acquireConnectionLock(): Promise<() => void> {\n while (connectionLock) {\n await connectionLock\n }\n\n let release: () => void\n connectionLock = new Promise(resolve => {\n release = () => {\n connectionLock = null\n resolve()\n }\n })\n\n return release!\n}\n\n/**\n * Configure MCP connection pool settings\n */\nexport function configureMcpPool(\n config: Partial<McpConnectionPoolConfig>,\n): void {\n poolConfig = { ...poolConfig, ...config }\n\n // Restart health check with new interval if changed\n if (config.healthCheckInterval !== undefined) {\n stopHealthCheck()\n if (poolConfig.healthCheckInterval > 0) {\n startHealthCheck()\n }\n }\n}\n\n/**\n * Get current pool configuration\n */\nexport function getMcpPoolConfig(): McpConnectionPoolConfig {\n return { ...poolConfig }\n}\n\n/**\n * Connection health status\n */\nexport interface McpConnectionHealth {\n name: string\n status: 'healthy' | 'unhealthy' | 'unknown'\n lastCheck: number\n latencyMs?: number\n error?: string\n}\n\n// Track connection health\nconst connectionHealth: Map<string, McpConnectionHealth> = new Map()\n\n/**\n * Check health of a single MCP connection\n */\nasync function checkConnectionHealth(\n client: ConnectedClient,\n): Promise<McpConnectionHealth> {\n const startTime = Date.now()\n try {\n // Try to get server capabilities as a health check\n await client.client.getServerCapabilities()\n const latencyMs = Date.now() - startTime\n\n return {\n name: client.name,\n status: 'healthy',\n lastCheck: Date.now(),\n latencyMs,\n }\n } catch (error) {\n return {\n name: client.name,\n status: 'unhealthy',\n lastCheck: Date.now(),\n error: error instanceof Error ? error.message : String(error),\n }\n }\n}\n\n/**\n * Run health check on all connections\n */\nexport async function runHealthCheck(): Promise<McpConnectionHealth[]> {\n const clients = await getClients()\n const healthResults: McpConnectionHealth[] = []\n\n for (const client of clients) {\n if (client.type === 'connected') {\n const health = await checkConnectionHealth(client)\n connectionHealth.set(client.name, health)\n healthResults.push(health)\n } else {\n const health: McpConnectionHealth = {\n name: client.name,\n status: 'unhealthy',\n lastCheck: Date.now(),\n error: 'Connection failed',\n }\n connectionHealth.set(client.name, health)\n healthResults.push(health)\n }\n }\n\n return healthResults\n}\n\n/**\n * Get cached health status for all connections\n */\nexport function getConnectionHealth(): McpConnectionHealth[] {\n return Array.from(connectionHealth.values())\n}\n\n/**\n * Get health status for a specific server\n */\nexport function getServerHealth(\n serverName: string,\n): McpConnectionHealth | null {\n return connectionHealth.get(serverName) || null\n}\n\n/**\n * Get circuit breaker for an MCP server\n * Used to check breaker status and manually control it\n */\nexport function getServerCircuitBreaker(serverName: string) {\n return mcpCircuitBreakers.getOrCreate(serverName)\n}\n\n/**\n * Get circuit breaker stats for all servers\n */\nexport function getAllCircuitBreakerStats() {\n const stats: Record<string, any> = {}\n for (const [name, breaker] of mcpCircuitBreakers.getAll().entries()) {\n stats[name] = breaker.getStats()\n }\n return stats\n}\n\n/**\n * Start periodic health checks\n */\nexport function startHealthCheck(): void {\n if (healthCheckTimer) return\n if (poolConfig.healthCheckInterval <= 0) return\n\n healthCheckTimer = setInterval(async () => {\n try {\n await runHealthCheck()\n } catch (e) {\n debugLogger.warn(\n 'MCP_HEALTH',\n `Health check failed: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }, poolConfig.healthCheckInterval)\n}\n\n/**\n * Stop periodic health checks\n */\nexport function stopHealthCheck(): void {\n if (healthCheckTimer) {\n clearInterval(healthCheckTimer)\n healthCheckTimer = null\n }\n}\n\n/**\n * Get connection pool statistics\n */\nexport function getPoolStats(): {\n totalConnections: number\n connectedCount: number\n failedCount: number\n healthyCount: number\n unhealthyCount: number\n} {\n const health = getConnectionHealth()\n return {\n totalConnections: connectedClients.length,\n connectedCount: connectedClients.filter(c => c.type === 'connected').length,\n failedCount: connectedClients.filter(c => c.type === 'failed').length,\n healthyCount: health.filter(h => h.status === 'healthy').length,\n unhealthyCount: health.filter(h => h.status === 'unhealthy').length,\n }\n}\n\n/**\n * Connect to an MCP server with retry and exponential backoff.\n * Uses poolConfig.retryFailed and poolConfig.maxRetries settings.\n */\nasync function connectWithRetry(\n name: string,\n serverRef: McpServerConfig,\n): Promise<Client> {\n const { retryFailed, maxRetries } = poolConfig\n let lastError: Error | undefined\n\n const attempts = retryFailed ? maxRetries : 0\n for (let attempt = 0; attempt <= attempts; attempt++) {\n try {\n return await connectToServer(name, serverRef)\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n if (attempt < attempts && retryFailed) {\n const delay = 1000 * Math.pow(2, attempt) // 1s, 2s\n logMCPError(\n name,\n `Connection attempt ${attempt + 1} failed, retrying in ${delay}ms...`,\n )\n await new Promise(r => setTimeout(r, delay))\n }\n }\n }\n throw lastError!\n}\n\nexport const getClients = memoize(async (): Promise<WrappedClient[]> => {\n // TODO: This is a temporary fix for a hang during npm run verify in CI.\n // We need to investigate why MCP client connections hang in CI verify but not in CI tests.\n if (process.env.CI && process.env.NODE_ENV !== 'test') {\n return []\n }\n\n const globalServers = getGlobalConfig().mcpServers ?? {}\n const mcprcServers = getMcprcConfig()\n const projectServers = getCurrentProjectConfig().mcpServers ?? {}\n\n // Filter mcprc servers to only include approved ones\n const approvedMcprcServers = pickBy(\n mcprcServers,\n (_, name) => getMcprcServerStatus(name) === 'approved',\n )\n\n const allServers = {\n ...globalServers,\n ...approvedMcprcServers, // Approved .mcprc servers override global ones\n ...projectServers, // Project servers take highest precedence\n }\n\n // Filter out disabled servers\n const enabledServers = pickBy(\n allServers,\n server => server.enabled !== false, // Default to enabled if not specified\n )\n\n const clients = await Promise.all(\n Object.entries(enabledServers).map(async ([name, serverRef]) => {\n try {\n const client = await connectWithRetry(\n name,\n serverRef as McpServerConfig,\n )\n return { name, client, type: 'connected' as const }\n } catch (error) {\n logMCPError(\n name,\n `Connection failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n return { name, type: 'failed' as const }\n }\n }),\n )\n\n // Store reference for shutdown cleanup\n connectedClients = clients\n return clients\n})\n\nasync function requestAll<\n ResultT extends Result,\n ResultSchemaT extends typeof ResultSchema,\n>(\n req: ClientRequest,\n resultSchema: ResultSchemaT,\n requiredCapability: string,\n): Promise<{ client: ConnectedClient; result: ResultT }[]> {\n const clients = await getClients()\n const results = await Promise.allSettled(\n clients.map(async client => {\n if (client.type === 'failed') return null\n\n try {\n const capabilities = await client.client.getServerCapabilities()\n if (!capabilities?.[requiredCapability]) {\n return null\n }\n return {\n client,\n result: (await client.client.request(req, resultSchema)) as ResultT,\n }\n } catch (error) {\n if (client.type === 'connected') {\n logMCPError(\n client.name,\n `Failed to request '${req.method}': ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n return null\n }\n }),\n )\n return results\n .filter(\n (\n result,\n ): result is PromiseFulfilledResult<{\n client: ConnectedClient\n result: ResultT\n } | null> => result.status === 'fulfilled',\n )\n .map(result => result.value)\n .filter(\n (result): result is { client: ConnectedClient; result: ResultT } =>\n result !== null,\n )\n}\n\n/**\n * Check if lazy/deferred MCP tool loading is enabled.\n * When enabled, inputSchema is loaded on-demand instead of upfront.\n */\nfunction isLazyMCPEnabled(): boolean {\n return !!(\n process.env.MINTO_ENABLE_TOOL_SEARCH || process.env.ENABLE_TOOL_SEARCH\n )\n}\n\n/**\n * Cache for lazily-loaded MCP tool schemas\n */\nconst lazySchemaCache = new Map<string, Tool['inputJSONSchema']>()\n\n/**\n * Lazily fetch the full input schema for an MCP tool\n */\nasync function getLazyInputSchema(\n client: ConnectedClient,\n toolName: string,\n fullName: string,\n): Promise<Tool['inputJSONSchema']> {\n if (lazySchemaCache.has(fullName)) {\n return lazySchemaCache.get(fullName)\n }\n\n try {\n const result = await client.client.request(\n { method: 'tools/list' },\n ListToolsResultSchema,\n )\n for (const tool of result.tools) {\n const key = 'mcp__' + client.name + '__' + tool.name\n lazySchemaCache.set(key, tool.inputSchema as Tool['inputJSONSchema'])\n }\n return lazySchemaCache.get(fullName)\n } catch {\n return undefined\n }\n}\n\nexport const getMCPTools = memoize(async (): Promise<Tool[]> => {\n const toolsList = await requestAll<\n ListToolsResult,\n typeof ListToolsResultSchema\n >(\n {\n method: 'tools/list',\n },\n ListToolsResultSchema,\n 'tools',\n )\n\n const lazyMode = isLazyMCPEnabled()\n\n return toolsList.flatMap(({ client, result: { tools } }) => {\n // Validate tool structure before processing\n const validTools = tools.filter(\n (tool): tool is typeof tool & { name: string } => {\n if (!tool || typeof tool.name !== 'string' || tool.name.length === 0) {\n logMCPError(\n `Invalid MCP tool from ${client.name}: missing or invalid name`,\n new Error('Tool validation failed'),\n )\n return false\n }\n return true\n },\n )\n\n return validTools.map((tool): Tool => {\n const fullName = 'mcp__' + client.name + '__' + tool.name\n\n // In lazy mode, defer schema loading until actually called\n if (lazyMode) {\n lazySchemaCache.set(\n fullName,\n tool.inputSchema as Tool['inputJSONSchema'],\n )\n }\n\n return {\n ...MCPTool,\n name: fullName,\n async description() {\n return tool.description ?? ''\n },\n async prompt() {\n return tool.description ?? ''\n },\n // In lazy mode, schema starts as undefined and is loaded on-demand\n inputJSONSchema: lazyMode\n ? undefined\n : (tool.inputSchema as Tool['inputJSONSchema']),\n async validateInput(input, context) {\n // MCP tools handle their own validation through their schemas\n return { result: true }\n },\n async *call(args: Record<string, unknown>, context) {\n // If in lazy mode and schema wasn't loaded yet, load it now\n if (lazyMode && !lazySchemaCache.has(fullName)) {\n await getLazyInputSchema(client, tool.name, fullName)\n }\n const data = await callMCPTool({ client, tool: tool.name, args })\n yield {\n type: 'result' as const,\n data,\n resultForAssistant: data,\n }\n },\n userFacingName() {\n return `${client.name}:${tool.name} (MCP)`\n },\n }\n })\n })\n})\n\n/**\n * Wraps a promise with a timeout.\n * If the promise doesn't resolve/reject within the timeout, rejects with a TimeoutError.\n */\nasync function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n operationName: string,\n): Promise<T> {\n let timeoutId: ReturnType<typeof setTimeout> | undefined\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(\n new Error(\n `MCP operation \"${operationName}\" timed out after ${timeoutMs}ms`,\n ),\n )\n }, timeoutMs)\n })\n\n try {\n const result = await Promise.race([promise, timeoutPromise])\n if (timeoutId) clearTimeout(timeoutId)\n return result\n } catch (error) {\n if (timeoutId) clearTimeout(timeoutId)\n throw error\n }\n}\n\n/**\n * Checks if an error is a timeout-related error.\n */\nfunction isTimeoutError(error: unknown): boolean {\n if (error instanceof Error) {\n return (\n error.name === 'TimeoutError' ||\n error.message.includes('timed out') ||\n error.message.includes('timeout')\n )\n }\n if (error instanceof DOMException) {\n return error.name === 'TimeoutError'\n }\n return false\n}\n\n// Import timing constants - use dynamic value or fallback\nconst MCP_TOOL_TIMEOUT_MS = 120000 // 2 minutes\nconst MCP_TOOL_MAX_RETRIES = 2\n\n/**\n * Truncate MCP tool output to prevent token overflow\n * Uses configurable maxOutputChars from pool config\n */\nfunction truncateMcpOutput(text: string): string {\n const limit = poolConfig.maxOutputChars\n if (text.length <= limit) return text\n return (\n text.slice(0, limit) +\n `\\n\\n[Output truncated: ${text.length} chars exceeded limit of ${limit}]`\n )\n}\n\nasync function callMCPTool({\n client: { client, name },\n tool,\n args,\n}: {\n client: ConnectedClient\n tool: string\n args: Record<string, unknown>\n}): Promise<ToolResultBlockParam['content']> {\n const breaker = mcpCircuitBreakers.getOrCreate(name)\n let lastError: Error | undefined\n const maxRetries = MCP_TOOL_MAX_RETRIES\n\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\n try {\n // Check circuit breaker before attempting the call\n const result = await breaker.executeWithTimeout(async () => {\n const callPromise = client.callTool(\n {\n name: tool,\n arguments: args,\n },\n CallToolResultSchema,\n )\n\n // Wrap with timeout protection\n return await withTimeout(\n callPromise,\n MCP_TOOL_TIMEOUT_MS,\n `${name}:${tool}`,\n )\n }, MCP_TOOL_TIMEOUT_MS)\n\n if ('isError' in result && result.isError) {\n const errorMessage = `Error calling tool ${tool}: ${result.error}`\n logMCPError(name, errorMessage)\n throw Error(errorMessage)\n }\n\n // Handle toolResult-type response\n if ('toolResult' in result) {\n const text = String(result.toolResult)\n return truncateMcpOutput(text)\n }\n\n // Handle content array response\n if ('content' in result && Array.isArray(result.content)) {\n return result.content.map(item => {\n if (item.type === 'image') {\n return {\n type: 'image',\n source: {\n type: 'base64',\n data: String(item.data),\n media_type:\n item.mimeType as ImageBlockParam.Source['media_type'],\n },\n }\n }\n return item\n })\n }\n\n throw Error(`Unexpected response format from tool ${tool}`)\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n\n // Circuit breaker is open - fail fast without retries\n if (lastError instanceof CircuitOpenError) {\n logMCPError(\n name,\n `Circuit breaker is open for server \"${name}\". ${lastError.message}`,\n )\n throw lastError\n }\n\n // Only retry on timeout errors, not on other errors\n if (isTimeoutError(error) && attempt < maxRetries) {\n logMCPError(\n name,\n `Tool \"${tool}\" timed out (attempt ${attempt + 1}/${maxRetries + 1}), retrying...`,\n )\n // Exponential backoff: 1s, 2s, 4s...\n await new Promise(resolve =>\n setTimeout(resolve, 1000 * Math.pow(2, attempt)),\n )\n continue\n }\n\n // Log and throw for non-retryable errors or after all retries exhausted\n if (isTimeoutError(error)) {\n logMCPError(\n name,\n `Tool \"${tool}\" failed after ${maxRetries + 1} attempts due to timeout`,\n )\n }\n throw lastError\n }\n }\n\n // This should never be reached, but TypeScript needs it\n throw lastError || new Error(`MCP tool ${tool} failed with unknown error`)\n}\n\nexport const getMCPCommands = memoize(async (): Promise<Command[]> => {\n const results = await requestAll<\n ListPromptsResult,\n typeof ListPromptsResultSchema\n >(\n {\n method: 'prompts/list',\n },\n ListPromptsResultSchema,\n 'prompts',\n )\n\n return results.flatMap(({ client, result }) =>\n result.prompts?.map(_ => {\n const argNames = Object.values(_.arguments ?? {}).map(k => k.name)\n return {\n type: 'prompt',\n name: 'mcp__' + client.name + '__' + _.name,\n description: _.description ?? '',\n isEnabled: true,\n isHidden: false,\n progressMessage: 'running',\n userFacingName() {\n return `${client.name}:${_.name} (MCP)`\n },\n argNames,\n async getPromptForCommand(args: string) {\n const argsArray = args.split(' ')\n return await runCommand(\n { name: _.name, client },\n zipObject(argNames, argsArray),\n )\n },\n }\n }),\n )\n})\n\nexport async function runCommand(\n { name, client }: { name: string; client: ConnectedClient },\n args: Record<string, string>,\n): Promise<MessageParam[]> {\n try {\n const result = await client.client.getPrompt({ name, arguments: args })\n return result.messages.map(\n (message): MessageParam => ({\n role: message.role,\n content: [\n message.content.type === 'text'\n ? {\n type: 'text',\n text: message.content.text,\n }\n : message.content.type === 'resource'\n ? {\n type: 'text',\n text:\n typeof (message.content as any).resource?.text === 'string'\n ? (message.content as any).resource.text\n : `[MCP Resource: ${(message.content as any).resource?.uri ?? 'unknown'}]`,\n }\n : {\n type: 'image',\n source: {\n data: String(message.content.data),\n media_type: message.content\n .mimeType as ImageBlockParam.Source['media_type'],\n type: 'base64',\n },\n },\n ],\n }),\n )\n } catch (error) {\n logMCPError(\n client.name,\n `Error running command '${name}': ${error instanceof Error ? error.message : String(error)}`,\n )\n throw error\n }\n}\n\n/**\n * Close a list of MCP clients in the background (fire-and-forget).\n * Does not block the caller \u2014 errors are silently ignored.\n */\nfunction closeClientsInBackground(clients: WrappedClient[]): void {\n const CLOSE_TIMEOUT_MS = 5000\n const closePromises = clients.map(async wrappedClient => {\n if (wrappedClient.type === 'connected') {\n try {\n await wrappedClient.client.close()\n } catch (e) {\n debugLogger.warn(\n 'MCP_CLOSE',\n `Failed to close MCP client: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }\n })\n\n // Race against timeout so orphaned closes don't linger forever\n Promise.race([\n Promise.allSettled(closePromises),\n new Promise(resolve => setTimeout(resolve, CLOSE_TIMEOUT_MS)),\n ]).catch(() => {})\n}\n\n/**\n * Refreshes MCP connections by clearing caches and allowing fresh\n * reconnections on next getClients() call.\n *\n * Old connections are closed in the background AFTER cache clear,\n * so new connections can be established independently without waiting\n * for old ones to shut down (which could fail or be slow).\n *\n * Uses an async lock to prevent concurrent execution and race conditions.\n * Also resets circuit breakers to allow reconnection attempts.\n */\nexport async function refreshMCPConnections(): Promise<void> {\n // Early return if refresh is already pending to avoid request accumulation\n if (pendingRefresh) {\n return\n }\n\n pendingRefresh = true\n\n const release = await acquireConnectionLock()\n\n try {\n // Snapshot old connections for background cleanup\n const oldClients = connectedClients\n connectedClients = []\n\n // Clear memoization caches so next getClients() reconnects fresh\n if (typeof getClients.cache?.clear === 'function') {\n getClients.cache.clear()\n }\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n if (typeof getMCPCommands.cache?.clear === 'function') {\n getMCPCommands.cache.clear()\n }\n\n // Reset circuit breakers to allow reconnection attempts\n mcpCircuitBreakers.resetAll()\n\n // Also invalidate the main tools cache since MCP tools are part of it\n // This is imported lazily to avoid circular dependencies\n try {\n const { invalidateToolsCache } = await import('../tools')\n invalidateToolsCache()\n } catch {\n // Ignore errors - tools.ts may not be loaded yet\n }\n\n // Close old connections in background \u2014 don't block the refresh.\n // New connections (via getClients()) will be created independently.\n if (oldClients.length > 0) {\n closeClientsInBackground(oldClients)\n }\n } finally {\n pendingRefresh = false\n release()\n }\n}\n\n/**\n * Shuts down all MCP client connections gracefully.\n * Should be called when the application is exiting.\n * Unlike refresh, shutdown WAITS for connections to close.\n */\nexport async function shutdownMCPClients(): Promise<void> {\n if (connectedClients.length === 0) return\n\n const oldClients = connectedClients\n connectedClients = []\n\n const SHUTDOWN_TIMEOUT_MS = 3000\n const closePromises = oldClients.map(async wrappedClient => {\n if (wrappedClient.type === 'connected') {\n try {\n await wrappedClient.client.close()\n } catch {\n // Ignore errors during shutdown\n }\n }\n })\n\n try {\n await Promise.race([\n Promise.allSettled(closePromises),\n new Promise(resolve => setTimeout(resolve, SHUTDOWN_TIMEOUT_MS)),\n ])\n } catch {\n // Ignore timeout errors\n }\n\n // Clear caches so no stale state remains\n if (typeof getClients.cache?.clear === 'function') {\n getClients.cache.clear()\n }\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n if (typeof getMCPCommands.cache?.clear === 'function') {\n getMCPCommands.cache.clear()\n }\n}\n\n/**\n * MCP Resource type for listing\n */\nexport type McpResource = {\n uri: string\n name: string\n description?: string\n mimeType?: string\n serverName: string\n}\n\n/**\n * List all resources from MCP servers\n * @param serverFilter Optional server name to filter resources from a specific server\n */\nexport async function listMCPResources(\n serverFilter?: string,\n): Promise<McpResource[]> {\n const results = await requestAll<\n ListResourcesResult,\n typeof ListResourcesResultSchema\n >(\n {\n method: 'resources/list',\n },\n ListResourcesResultSchema,\n 'resources',\n )\n\n return results\n .filter(({ client }) => !serverFilter || client.name === serverFilter)\n .flatMap(({ client, result }) =>\n (result.resources || []).map(resource => ({\n uri: resource.uri,\n name: resource.name,\n description: resource.description,\n mimeType: resource.mimeType,\n serverName: client.name,\n })),\n )\n}\n\n/**\n * MCP Resource content type\n */\nexport type McpResourceContent = {\n uri: string\n mimeType?: string\n text?: string\n blob?: string // Base64 encoded binary data\n}\n\n/**\n * Read a specific resource from an MCP server\n * @param uri The resource URI to read\n * @param serverName Optional server name to target a specific server\n */\nexport async function readMCPResource(\n uri: string,\n serverName?: string,\n): Promise<McpResourceContent | null> {\n const clients = await getClients()\n\n // Find the appropriate client\n const targetClients = serverName\n ? (clients.filter(\n c => c.type === 'connected' && c.name === serverName,\n ) as ConnectedClient[])\n : (clients.filter(c => c.type === 'connected') as ConnectedClient[])\n\n for (const client of targetClients) {\n try {\n // Check if this server has resources capability\n const capabilities = await client.client.getServerCapabilities()\n if (!capabilities?.resources) {\n continue\n }\n\n // Try to read the resource\n const result = (await client.client.request(\n {\n method: 'resources/read',\n params: { uri },\n },\n ReadResourceResultSchema,\n )) as ReadResourceResult\n\n if (result.contents && result.contents.length > 0) {\n const content = result.contents[0]\n return {\n uri: content.uri,\n mimeType: content.mimeType,\n text: 'text' in content ? String(content.text) : undefined,\n blob: 'blob' in content ? String(content.blob) : undefined,\n }\n } else {\n emitReminderEvent('mcp:resource_no_content', { resourceUri: uri })\n }\n } catch (error) {\n // Log but continue trying other servers\n logMCPError(\n client.name,\n `Failed to read resource '${uri}': ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n }\n\n return null\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { zipObject } from 'lodash-es'\nimport { debug as debugLogger } from '@utils/debugLogger'\nimport {\n getCurrentProjectConfig,\n McpServerConfig,\n saveCurrentProjectConfig,\n getGlobalConfig,\n getMutableGlobalConfig,\n saveGlobalConfig,\n getMcprcConfig,\n addMcprcServerForTesting,\n removeMcprcServerForTesting,\n} from '@utils/config'\nimport { existsSync, readFileSync, writeFileSync } from 'fs'\nimport { join } from 'path'\nimport { getCwd } from '@utils/state'\nimport { safeParseJSON } from '@utils/json'\nimport {\n ImageBlockParam,\n MessageParam,\n ToolResultBlockParam,\n} from '@anthropic-ai/sdk/resources/index.mjs'\nimport { Client } from '@modelcontextprotocol/sdk/client/index.js'\nimport { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'\nimport { SSEClientTransport } from '@modelcontextprotocol/sdk/client/sse.js'\nimport {\n CallToolResultSchema,\n ClientRequest,\n ListPromptsResult,\n ListPromptsResultSchema,\n ListToolsResult,\n ListToolsResultSchema,\n ListResourcesResult,\n ListResourcesResultSchema,\n ReadResourceResult,\n ReadResourceResultSchema,\n Result,\n ResultSchema,\n ToolListChangedNotificationSchema,\n} from '@modelcontextprotocol/sdk/types.js'\nimport { memoize, pickBy } from 'lodash-es'\nimport type { Tool } from '@tool'\nimport { MCPTool } from '@tools/MCPTool/MCPTool'\nimport { logMCPError } from '@utils/log'\nimport { ensureProxyBypass } from '@utils/envConfig'\nimport { Command } from '@commands'\nimport { PRODUCT_COMMAND } from '@constants/product'\nimport { CircuitBreakerRegistry, CircuitOpenError } from '@utils/CircuitBreaker'\nimport { emitReminderEvent } from '@services/systemReminder'\n\ntype McpName = string\n\n/**\n * Circuit breaker registry for MCP server connections\n * Each server gets its own circuit breaker to prevent cascading failures\n */\n/**\n * Env var prefixes/names to strip when spawning MCP server processes.\n * Prevents leaking API keys and secrets to third-party servers.\n * Explicit per-server `env` values in config still override (intentional).\n */\nconst SENSITIVE_ENV_PREFIXES = [\n 'ANTHROPIC_',\n 'OPENAI_',\n 'AWS_SECRET',\n 'AWS_ACCESS_KEY',\n 'AWS_SESSION_TOKEN',\n 'AZURE_',\n 'GOOGLE_API_KEY',\n 'GOOGLE_APPLICATION_CREDENTIALS',\n 'HF_TOKEN',\n 'REPLICATE_API_TOKEN',\n 'HUGGINGFACE_',\n 'COHERE_',\n 'MISTRAL_',\n 'GROQ_',\n 'TOGETHER_',\n 'FIREWORKS_',\n 'PERPLEXITY_',\n 'DEEPSEEK_',\n 'GITHUB_TOKEN',\n 'GH_TOKEN',\n 'MINTO_API',\n]\n\n/**\n * Sensitive env var suffix patterns.\n * Any env var ending with one of these suffixes is filtered out,\n * UNLESS it is in the allowlist of safe system vars.\n */\nconst SENSITIVE_ENV_SUFFIXES = ['_SECRET', '_PASSWORD', '_TOKEN', '_KEY']\n\n/**\n * System env vars (or prefixes) that should NEVER be filtered,\n * even if they match a sensitive suffix pattern.\n */\nconst SAFE_SYSTEM_ENV_VARS = new Set([\n 'HOME',\n 'PATH',\n 'TERM',\n 'SHELL',\n 'USER',\n 'LANG',\n 'EDITOR',\n 'DISPLAY',\n 'SSH_AUTH_SOCK',\n 'GPG_TTY',\n])\n\nfunction isSafeSystemVar(key: string): boolean {\n return SAFE_SYSTEM_ENV_VARS.has(key) || key.startsWith('XDG_')\n}\n\nexport function filterSensitiveEnv(\n env: NodeJS.ProcessEnv,\n): Record<string, string> {\n return Object.fromEntries(\n Object.entries(env)\n .filter(([k]) => {\n // Always keep safe system vars\n if (isSafeSystemVar(k)) return true\n // Block prefix matches\n if (SENSITIVE_ENV_PREFIXES.some(p => k.startsWith(p) || k === p))\n return false\n // Block suffix matches\n if (SENSITIVE_ENV_SUFFIXES.some(s => k.endsWith(s))) return false\n return true\n })\n .filter((e): e is [string, string] => e[1] !== undefined),\n )\n}\n\nconst mcpCircuitBreakers = new CircuitBreakerRegistry({\n failureThreshold: 5,\n successThreshold: 2,\n openTimeout: 30000, // 30 seconds before trying again\n})\n\nexport function parseEnvVars(\n rawEnvArgs: string[] | undefined,\n): Record<string, string> {\n const parsedEnv: Record<string, string> = {}\n\n // Parse individual env vars\n if (rawEnvArgs) {\n for (const envStr of rawEnvArgs) {\n const [key, ...valueParts] = envStr.split('=')\n if (!key || valueParts.length === 0) {\n throw new Error(\n `Invalid environment variable format: ${envStr}, environment variables should be added as: -e KEY1=value1 -e KEY2=value2`,\n )\n }\n parsedEnv[key] = valueParts.join('=')\n }\n }\n return parsedEnv\n}\n\nconst VALID_SCOPES = ['project', 'global', 'mcprc'] as const\ntype ConfigScope = (typeof VALID_SCOPES)[number]\nconst EXTERNAL_SCOPES = ['project', 'global'] as ConfigScope[]\n\nexport function ensureConfigScope(scope?: string): ConfigScope {\n if (!scope) return 'project'\n\n const scopesToCheck =\n process.env.USER_TYPE === 'external' ? EXTERNAL_SCOPES : VALID_SCOPES\n\n if (!scopesToCheck.includes(scope as ConfigScope)) {\n throw new Error(\n `Invalid scope: ${scope}. Must be one of: ${scopesToCheck.join(', ')}`,\n )\n }\n\n return scope as ConfigScope\n}\n\nexport function addMcpServer(\n name: McpName,\n server: McpServerConfig,\n scope: ConfigScope = 'project',\n): void {\n if (scope === 'mcprc') {\n if (process.env.NODE_ENV === 'test') {\n addMcprcServerForTesting(name, server)\n } else {\n const mcprcPath = join(getCwd(), '.mcprc')\n let mcprcConfig: Record<string, McpServerConfig> = {}\n\n // Read existing config if present\n if (existsSync(mcprcPath)) {\n try {\n const mcprcContent = readFileSync(mcprcPath, 'utf-8')\n const existingConfig = safeParseJSON(mcprcContent)\n if (existingConfig && typeof existingConfig === 'object') {\n mcprcConfig = existingConfig as Record<string, McpServerConfig>\n }\n } catch (e) {\n debugLogger.warn(\n 'MCP_CONFIG',\n `Failed to read/parse .mcprc: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }\n\n // Add the server\n mcprcConfig[name] = server\n\n // Write back to .mcprc\n try {\n writeFileSync(mcprcPath, JSON.stringify(mcprcConfig, null, 2), 'utf-8')\n } catch (error) {\n throw new Error(`Failed to write to .mcprc: ${error}`)\n }\n }\n } else if (scope === 'global') {\n const config = getMutableGlobalConfig()\n if (!config.mcpServers) {\n config.mcpServers = {}\n }\n config.mcpServers[name] = server\n saveGlobalConfig(config)\n } else {\n const config = getCurrentProjectConfig()\n if (!config.mcpServers) {\n config.mcpServers = {}\n }\n config.mcpServers[name] = server\n saveCurrentProjectConfig(config)\n }\n}\n\nexport function removeMcpServer(\n name: McpName,\n scope: ConfigScope = 'project',\n): void {\n if (scope === 'mcprc') {\n if (process.env.NODE_ENV === 'test') {\n removeMcprcServerForTesting(name)\n } else {\n const mcprcPath = join(getCwd(), '.mcprc')\n if (!existsSync(mcprcPath)) {\n throw new Error('No .mcprc file found in this directory')\n }\n\n try {\n const mcprcContent = readFileSync(mcprcPath, 'utf-8')\n const mcprcConfig = safeParseJSON(mcprcContent) as Record<\n string,\n McpServerConfig\n > | null\n\n if (\n !mcprcConfig ||\n typeof mcprcConfig !== 'object' ||\n !mcprcConfig[name]\n ) {\n throw new Error(`No MCP server found with name: ${name} in .mcprc`)\n }\n\n delete mcprcConfig[name]\n writeFileSync(mcprcPath, JSON.stringify(mcprcConfig, null, 2), 'utf-8')\n } catch (error) {\n if (error instanceof Error) {\n throw error\n }\n throw new Error(`Failed to remove from .mcprc: ${error}`)\n }\n }\n } else if (scope === 'global') {\n const config = getGlobalConfig()\n if (!config.mcpServers?.[name]) {\n throw new Error(`No global MCP server found with name: ${name}`)\n }\n delete config.mcpServers[name]\n saveGlobalConfig(config)\n } else {\n const config = getCurrentProjectConfig()\n if (!config.mcpServers?.[name]) {\n throw new Error(`No local MCP server found with name: ${name}`)\n }\n delete config.mcpServers[name]\n saveCurrentProjectConfig(config)\n }\n}\n\nexport function listMCPServers(): Record<string, McpServerConfig> {\n const globalConfig = getGlobalConfig()\n const mcprcConfig = getMcprcConfig()\n const projectConfig = getCurrentProjectConfig()\n return {\n ...(globalConfig.mcpServers ?? {}),\n ...(mcprcConfig ?? {}), // mcprc configs override global ones\n ...(projectConfig.mcpServers ?? {}), // Project configs override mcprc ones\n }\n}\n\nexport type ScopedMcpServerConfig = McpServerConfig & {\n scope: ConfigScope\n}\n\nexport function getMcpServer(name: McpName): ScopedMcpServerConfig | undefined {\n const projectConfig = getCurrentProjectConfig()\n const mcprcConfig = getMcprcConfig()\n const globalConfig = getGlobalConfig()\n\n // Check each scope in order of precedence\n if (projectConfig.mcpServers?.[name]) {\n return { ...projectConfig.mcpServers[name], scope: 'project' }\n }\n\n if (mcprcConfig?.[name]) {\n return { ...mcprcConfig[name], scope: 'mcprc' }\n }\n\n if (globalConfig.mcpServers?.[name]) {\n return { ...globalConfig.mcpServers[name], scope: 'global' }\n }\n\n return undefined\n}\n\nasync function connectToServer(\n name: string,\n serverRef: McpServerConfig,\n): Promise<Client> {\n let transport\n\n if (serverRef.type === 'http') {\n // Streamable HTTP transport (CC spec)\n const { StreamableHTTPClientTransport } = await import(\n '@modelcontextprotocol/sdk/client/streamableHttp.js'\n )\n transport = new StreamableHTTPClientTransport(\n new URL((serverRef as any).url),\n {\n // Pass headers but never log sensitive values\n requestInit: (serverRef as any).headers\n ? { headers: (serverRef as any).headers }\n : undefined,\n },\n )\n } else if (serverRef.type === 'sse') {\n transport = new SSEClientTransport(new URL(serverRef.url))\n } else {\n transport = new StdioClientTransport({\n command: serverRef.command,\n args: serverRef.args,\n env: ensureProxyBypass({\n ...filterSensitiveEnv(process.env),\n ...serverRef.env,\n } as Record<string, string>),\n stderr: 'pipe', // prevents error output from the MCP server from printing to the UI\n })\n }\n\n const client = new Client(\n {\n name: PRODUCT_COMMAND,\n version: '0.1.0',\n },\n {\n capabilities: {},\n },\n )\n\n // Add a timeout to connection attempts to prevent tests from hanging indefinitely\n const connectionTimeout = poolConfig.connectionTimeout\n const connectPromise = client.connect(transport)\n const timeoutPromise = new Promise<never>((_, reject) => {\n const timeoutId = setTimeout(() => {\n reject(\n new Error(\n `Connection to MCP server \"${name}\" timed out after ${connectionTimeout}ms`,\n ),\n )\n }, connectionTimeout)\n\n // Clean up timeout if connect resolves or rejects\n connectPromise.then(\n () => clearTimeout(timeoutId),\n () => clearTimeout(timeoutId),\n )\n })\n\n await Promise.race([connectPromise, timeoutPromise])\n\n if (serverRef.type === 'stdio') {\n // Track child process PID for synchronous cleanup on exit.\n // StdioClientTransport stores the child process in _process (private).\n const childProc = (transport as any)._process\n if (childProc?.pid) {\n const { registerChildPid, unregisterChildPid } = await import(\n '@utils/exit'\n )\n registerChildPid(childProc.pid)\n // Unregister when the child process exits naturally\n childProc.on?.('exit', () => unregisterChildPid(childProc.pid))\n }\n\n ;(transport as StdioClientTransport).stderr?.on('data', (data: Buffer) => {\n const errorText = data.toString().trim()\n if (errorText) {\n logMCPError(name, `Server stderr: ${errorText}`)\n }\n })\n }\n\n // Listen for tools/list_changed notification to dynamically update tool lists\n client.setNotificationHandler(ToolListChangedNotificationSchema, async () => {\n // Clear cached tools so next getMCPTools() fetches fresh list\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n })\n\n return client\n}\n\ntype ConnectedClient = {\n client: Client\n name: string\n type: 'connected'\n}\ntype FailedClient = {\n name: string\n type: 'failed'\n}\nexport type WrappedClient = ConnectedClient | FailedClient\n\nexport function getMcprcServerStatus(\n serverName: string,\n): 'approved' | 'rejected' | 'pending' {\n const config = getCurrentProjectConfig()\n if (config.approvedMcprcServers?.includes(serverName)) {\n return 'approved'\n }\n if (config.rejectedMcprcServers?.includes(serverName)) {\n return 'rejected'\n }\n return 'pending'\n}\n\n// Store connected clients for cleanup on shutdown\nlet connectedClients: WrappedClient[] = []\n\n// Connection pool configuration\nexport interface McpConnectionPoolConfig {\n /** Maximum concurrent connections (default: 10) */\n maxConnections: number\n /** Connection timeout in ms (default: 10000) */\n connectionTimeout: number\n /** Health check interval in ms (default: 30000, 0 to disable) */\n healthCheckInterval: number\n /** Retry failed connections (default: true) */\n retryFailed: boolean\n /** Max retry attempts (default: 3) */\n maxRetries: number\n /** Maximum output size from MCP tool calls in characters (default: 25000) */\n maxOutputChars: number\n}\n\nconst DEFAULT_POOL_CONFIG: McpConnectionPoolConfig = {\n maxConnections: 10,\n connectionTimeout: Number(process.env.MCP_TIMEOUT) || 10000,\n healthCheckInterval: 30000,\n retryFailed: true,\n maxRetries: 2,\n maxOutputChars: Number(process.env.MAX_MCP_OUTPUT_CHARS) || 25000,\n}\n\nlet poolConfig: McpConnectionPoolConfig = { ...DEFAULT_POOL_CONFIG }\nlet healthCheckTimer: ReturnType<typeof setInterval> | null = null\n\n// Connection lock mechanism to prevent concurrent refreshMCPConnections calls\nlet connectionLock: Promise<void> | null = null\nlet pendingRefresh = false\n\n/**\n * Acquires the connection lock for exclusive access to MCP connection refresh\n * Waits until any existing lock is released before creating a new one\n */\nasync function acquireConnectionLock(): Promise<() => void> {\n while (connectionLock) {\n await connectionLock\n }\n\n let release: () => void\n connectionLock = new Promise(resolve => {\n release = () => {\n connectionLock = null\n resolve()\n }\n })\n\n return release!\n}\n\n/**\n * Configure MCP connection pool settings\n */\nexport function configureMcpPool(\n config: Partial<McpConnectionPoolConfig>,\n): void {\n poolConfig = { ...poolConfig, ...config }\n\n // Restart health check with new interval if changed\n if (config.healthCheckInterval !== undefined) {\n stopHealthCheck()\n if (poolConfig.healthCheckInterval > 0) {\n startHealthCheck()\n }\n }\n}\n\n/**\n * Get current pool configuration\n */\nexport function getMcpPoolConfig(): McpConnectionPoolConfig {\n return { ...poolConfig }\n}\n\n/**\n * Connection health status\n */\nexport interface McpConnectionHealth {\n name: string\n status: 'healthy' | 'unhealthy' | 'unknown'\n lastCheck: number\n latencyMs?: number\n error?: string\n}\n\n// Track connection health\nconst connectionHealth: Map<string, McpConnectionHealth> = new Map()\n\n/**\n * Check health of a single MCP connection\n */\nasync function checkConnectionHealth(\n client: ConnectedClient,\n): Promise<McpConnectionHealth> {\n const startTime = Date.now()\n try {\n // Try to get server capabilities as a health check\n await client.client.getServerCapabilities()\n const latencyMs = Date.now() - startTime\n\n return {\n name: client.name,\n status: 'healthy',\n lastCheck: Date.now(),\n latencyMs,\n }\n } catch (error) {\n return {\n name: client.name,\n status: 'unhealthy',\n lastCheck: Date.now(),\n error: error instanceof Error ? error.message : String(error),\n }\n }\n}\n\n/**\n * Run health check on all connections\n */\nexport async function runHealthCheck(): Promise<McpConnectionHealth[]> {\n const clients = await getClients()\n const healthResults: McpConnectionHealth[] = []\n\n for (const client of clients) {\n if (client.type === 'connected') {\n const health = await checkConnectionHealth(client)\n connectionHealth.set(client.name, health)\n healthResults.push(health)\n } else {\n const health: McpConnectionHealth = {\n name: client.name,\n status: 'unhealthy',\n lastCheck: Date.now(),\n error: 'Connection failed',\n }\n connectionHealth.set(client.name, health)\n healthResults.push(health)\n }\n }\n\n return healthResults\n}\n\n/**\n * Get cached health status for all connections\n */\nexport function getConnectionHealth(): McpConnectionHealth[] {\n return Array.from(connectionHealth.values())\n}\n\n/**\n * Get health status for a specific server\n */\nexport function getServerHealth(\n serverName: string,\n): McpConnectionHealth | null {\n return connectionHealth.get(serverName) || null\n}\n\n/**\n * Get circuit breaker for an MCP server\n * Used to check breaker status and manually control it\n */\nexport function getServerCircuitBreaker(serverName: string) {\n return mcpCircuitBreakers.getOrCreate(serverName)\n}\n\n/**\n * Get circuit breaker stats for all servers\n */\nexport function getAllCircuitBreakerStats() {\n const stats: Record<string, any> = {}\n for (const [name, breaker] of mcpCircuitBreakers.getAll().entries()) {\n stats[name] = breaker.getStats()\n }\n return stats\n}\n\n/**\n * Start periodic health checks\n */\nexport function startHealthCheck(): void {\n if (healthCheckTimer) return\n if (poolConfig.healthCheckInterval <= 0) return\n\n healthCheckTimer = setInterval(async () => {\n try {\n await runHealthCheck()\n } catch (e) {\n debugLogger.warn(\n 'MCP_HEALTH',\n `Health check failed: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }, poolConfig.healthCheckInterval)\n}\n\n/**\n * Stop periodic health checks\n */\nexport function stopHealthCheck(): void {\n if (healthCheckTimer) {\n clearInterval(healthCheckTimer)\n healthCheckTimer = null\n }\n}\n\n/**\n * Get connection pool statistics\n */\nexport function getPoolStats(): {\n totalConnections: number\n connectedCount: number\n failedCount: number\n healthyCount: number\n unhealthyCount: number\n} {\n const health = getConnectionHealth()\n return {\n totalConnections: connectedClients.length,\n connectedCount: connectedClients.filter(c => c.type === 'connected').length,\n failedCount: connectedClients.filter(c => c.type === 'failed').length,\n healthyCount: health.filter(h => h.status === 'healthy').length,\n unhealthyCount: health.filter(h => h.status === 'unhealthy').length,\n }\n}\n\n/**\n * Connect to an MCP server with retry and exponential backoff.\n * Uses poolConfig.retryFailed and poolConfig.maxRetries settings.\n */\nasync function connectWithRetry(\n name: string,\n serverRef: McpServerConfig,\n): Promise<Client> {\n const { retryFailed, maxRetries } = poolConfig\n let lastError: Error | undefined\n\n const attempts = retryFailed ? maxRetries : 0\n for (let attempt = 0; attempt <= attempts; attempt++) {\n try {\n return await connectToServer(name, serverRef)\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n if (attempt < attempts && retryFailed) {\n const delay = 1000 * Math.pow(2, attempt) // 1s, 2s\n logMCPError(\n name,\n `Connection attempt ${attempt + 1} failed, retrying in ${delay}ms...`,\n )\n await new Promise(r => setTimeout(r, delay))\n }\n }\n }\n throw lastError!\n}\n\nexport const getClients = memoize(async (): Promise<WrappedClient[]> => {\n // TODO: This is a temporary fix for a hang during npm run verify in CI.\n // We need to investigate why MCP client connections hang in CI verify but not in CI tests.\n if (process.env.CI && process.env.NODE_ENV !== 'test') {\n return []\n }\n\n const globalServers = getGlobalConfig().mcpServers ?? {}\n const mcprcServers = getMcprcConfig()\n const projectServers = getCurrentProjectConfig().mcpServers ?? {}\n\n // Filter mcprc servers to only include approved ones\n const approvedMcprcServers = pickBy(\n mcprcServers,\n (_, name) => getMcprcServerStatus(name) === 'approved',\n )\n\n const allServers = {\n ...globalServers,\n ...approvedMcprcServers, // Approved .mcprc servers override global ones\n ...projectServers, // Project servers take highest precedence\n }\n\n // Filter out disabled servers\n const enabledServers = pickBy(\n allServers,\n server => server.enabled !== false, // Default to enabled if not specified\n )\n\n const clients = await Promise.all(\n Object.entries(enabledServers).map(async ([name, serverRef]) => {\n try {\n const client = await connectWithRetry(\n name,\n serverRef as McpServerConfig,\n )\n return { name, client, type: 'connected' as const }\n } catch (error) {\n logMCPError(\n name,\n `Connection failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n return { name, type: 'failed' as const }\n }\n }),\n )\n\n // Store reference for shutdown cleanup\n connectedClients = clients\n return clients\n})\n\nasync function requestAll<\n ResultT extends Result,\n ResultSchemaT extends typeof ResultSchema,\n>(\n req: ClientRequest,\n resultSchema: ResultSchemaT,\n requiredCapability: string,\n): Promise<{ client: ConnectedClient; result: ResultT }[]> {\n const clients = await getClients()\n const results = await Promise.allSettled(\n clients.map(async client => {\n if (client.type === 'failed') return null\n\n try {\n const capabilities = await client.client.getServerCapabilities()\n if (!capabilities?.[requiredCapability]) {\n return null\n }\n return {\n client,\n result: (await client.client.request(req, resultSchema)) as ResultT,\n }\n } catch (error) {\n if (client.type === 'connected') {\n logMCPError(\n client.name,\n `Failed to request '${req.method}': ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n return null\n }\n }),\n )\n return results\n .filter(\n (\n result,\n ): result is PromiseFulfilledResult<{\n client: ConnectedClient\n result: ResultT\n } | null> => result.status === 'fulfilled',\n )\n .map(result => result.value)\n .filter(\n (result): result is { client: ConnectedClient; result: ResultT } =>\n result !== null,\n )\n}\n\n/**\n * Check if lazy/deferred MCP tool loading is enabled.\n * When enabled, inputSchema is loaded on-demand instead of upfront.\n */\nfunction isLazyMCPEnabled(): boolean {\n return !!(\n process.env.MINTO_ENABLE_TOOL_SEARCH || process.env.ENABLE_TOOL_SEARCH\n )\n}\n\n/**\n * Cache for lazily-loaded MCP tool schemas\n */\nconst lazySchemaCache = new Map<string, Tool['inputJSONSchema']>()\n\n// \u2500\u2500 Deferred MCP Tool Registry \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n// When tool search mode is active, tools that are not yet activated are\n// stored here with minimal metadata (name + description + server).\n// The ToolSearch tool queries this registry and calls activateMCPTool()\n// to promote deferred tools into the active tool set.\n\nexport interface DeferredMCPToolMeta {\n name: string\n description: string\n serverName: string\n}\n\nconst deferredTools = new Map<string, DeferredMCPToolMeta>()\nconst activatedToolNames = new Set<string>()\n\n/**\n * Get the current set of deferred (not yet activated) MCP tools.\n * Used by ToolSearch to search through available tools.\n */\nexport function getDeferredMCPTools(): Map<string, DeferredMCPToolMeta> {\n return deferredTools\n}\n\n/**\n * Activate a deferred MCP tool by its full name (e.g. \"mcp__server__tool\").\n * Removes it from the deferred registry and adds to the activated set.\n * Returns true if the tool was newly activated, false if already active.\n *\n * Note: The tool won't appear in the tool list until getTools() is\n * re-invoked (cache invalidated). The tools.ts layer handles this.\n */\nexport function activateMCPTool(toolFullName: string): boolean {\n if (activatedToolNames.has(toolFullName)) {\n return false\n }\n activatedToolNames.add(toolFullName)\n deferredTools.delete(toolFullName)\n\n // Invalidate tools cache so the newly activated tool gets included\n try {\n // Dynamic import to avoid circular dependency\n import('../tools')\n .then(({ invalidateToolsCache }) => {\n invalidateToolsCache()\n })\n .catch(() => {})\n } catch {\n // Ignore - tools.ts may not be loaded\n }\n\n return true\n}\n\n/**\n * Check whether a tool has been activated (or was never deferred).\n */\nexport function isMCPToolActivated(toolFullName: string): boolean {\n return activatedToolNames.has(toolFullName)\n}\n\n/**\n * Clear all deferred/activated state. Called during MCP refresh.\n */\nexport function clearDeferredState(): void {\n deferredTools.clear()\n activatedToolNames.clear()\n}\n\n/**\n * Lazily fetch the full input schema for an MCP tool\n */\nasync function getLazyInputSchema(\n client: ConnectedClient,\n toolName: string,\n fullName: string,\n): Promise<Tool['inputJSONSchema']> {\n if (lazySchemaCache.has(fullName)) {\n return lazySchemaCache.get(fullName)\n }\n\n try {\n const result = await client.client.request(\n { method: 'tools/list' },\n ListToolsResultSchema,\n )\n for (const tool of result.tools) {\n const key = 'mcp__' + client.name + '__' + tool.name\n lazySchemaCache.set(key, tool.inputSchema as Tool['inputJSONSchema'])\n }\n return lazySchemaCache.get(fullName)\n } catch {\n return undefined\n }\n}\n\nexport const getMCPTools = memoize(async (): Promise<Tool[]> => {\n const toolsList = await requestAll<\n ListToolsResult,\n typeof ListToolsResultSchema\n >(\n {\n method: 'tools/list',\n },\n ListToolsResultSchema,\n 'tools',\n )\n\n const lazyMode = isLazyMCPEnabled()\n\n return toolsList.flatMap(({ client, result: { tools } }) => {\n // Validate tool structure before processing\n const validTools = tools.filter(\n (tool): tool is typeof tool & { name: string } => {\n if (!tool || typeof tool.name !== 'string' || tool.name.length === 0) {\n logMCPError(\n `Invalid MCP tool from ${client.name}: missing or invalid name`,\n new Error('Tool validation failed'),\n )\n return false\n }\n return true\n },\n )\n\n return validTools.map((tool): Tool => {\n const fullName = 'mcp__' + client.name + '__' + tool.name\n\n // In lazy mode, defer schema loading until actually called\n if (lazyMode) {\n lazySchemaCache.set(\n fullName,\n tool.inputSchema as Tool['inputJSONSchema'],\n )\n }\n\n return {\n ...MCPTool,\n name: fullName,\n async description() {\n return tool.description ?? ''\n },\n async prompt() {\n return tool.description ?? ''\n },\n // In lazy mode, schema starts as undefined and is loaded on-demand\n inputJSONSchema: lazyMode\n ? undefined\n : (tool.inputSchema as Tool['inputJSONSchema']),\n async validateInput(input, context) {\n // MCP tools handle their own validation through their schemas\n return { result: true }\n },\n async *call(args: Record<string, unknown>, context) {\n // If in lazy mode and schema wasn't loaded yet, load it now\n if (lazyMode && !lazySchemaCache.has(fullName)) {\n await getLazyInputSchema(client, tool.name, fullName)\n }\n const data = await callMCPTool({ client, tool: tool.name, args })\n yield {\n type: 'result' as const,\n data,\n resultForAssistant: data,\n }\n },\n userFacingName() {\n return `${client.name}:${tool.name} (MCP)`\n },\n }\n })\n })\n})\n\n/**\n * Wraps a promise with a timeout.\n * If the promise doesn't resolve/reject within the timeout, rejects with a TimeoutError.\n */\nasync function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n operationName: string,\n): Promise<T> {\n let timeoutId: ReturnType<typeof setTimeout> | undefined\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(\n new Error(\n `MCP operation \"${operationName}\" timed out after ${timeoutMs}ms`,\n ),\n )\n }, timeoutMs)\n })\n\n try {\n const result = await Promise.race([promise, timeoutPromise])\n if (timeoutId) clearTimeout(timeoutId)\n return result\n } catch (error) {\n if (timeoutId) clearTimeout(timeoutId)\n throw error\n }\n}\n\n/**\n * Checks if an error is a timeout-related error.\n */\nfunction isTimeoutError(error: unknown): boolean {\n if (error instanceof Error) {\n return (\n error.name === 'TimeoutError' ||\n error.message.includes('timed out') ||\n error.message.includes('timeout')\n )\n }\n if (error instanceof DOMException) {\n return error.name === 'TimeoutError'\n }\n return false\n}\n\n// Import timing constants - use dynamic value or fallback\nconst MCP_TOOL_TIMEOUT_MS = 120000 // 2 minutes\nconst MCP_TOOL_MAX_RETRIES = 2\n\n/**\n * Truncate MCP tool output to prevent token overflow\n * Uses configurable maxOutputChars from pool config\n */\nfunction truncateMcpOutput(text: string): string {\n const limit = poolConfig.maxOutputChars\n if (text.length <= limit) return text\n return (\n text.slice(0, limit) +\n `\\n\\n[Output truncated: ${text.length} chars exceeded limit of ${limit}]`\n )\n}\n\nasync function callMCPTool({\n client: { client, name },\n tool,\n args,\n}: {\n client: ConnectedClient\n tool: string\n args: Record<string, unknown>\n}): Promise<ToolResultBlockParam['content']> {\n const breaker = mcpCircuitBreakers.getOrCreate(name)\n let lastError: Error | undefined\n const maxRetries = MCP_TOOL_MAX_RETRIES\n\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\n try {\n // Check circuit breaker before attempting the call\n const result = await breaker.executeWithTimeout(async () => {\n const callPromise = client.callTool(\n {\n name: tool,\n arguments: args,\n },\n CallToolResultSchema,\n )\n\n // Wrap with timeout protection\n return await withTimeout(\n callPromise,\n MCP_TOOL_TIMEOUT_MS,\n `${name}:${tool}`,\n )\n }, MCP_TOOL_TIMEOUT_MS)\n\n if ('isError' in result && result.isError) {\n const errorMessage = `Error calling tool ${tool}: ${result.error}`\n logMCPError(name, errorMessage)\n throw Error(errorMessage)\n }\n\n // Handle toolResult-type response\n if ('toolResult' in result) {\n const text = String(result.toolResult)\n return truncateMcpOutput(text)\n }\n\n // Handle content array response\n if ('content' in result && Array.isArray(result.content)) {\n return result.content.map(item => {\n if (item.type === 'image') {\n return {\n type: 'image',\n source: {\n type: 'base64',\n data: String(item.data),\n media_type:\n item.mimeType as ImageBlockParam.Source['media_type'],\n },\n }\n }\n return item\n })\n }\n\n throw Error(`Unexpected response format from tool ${tool}`)\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n\n // Circuit breaker is open - fail fast without retries\n if (lastError instanceof CircuitOpenError) {\n logMCPError(\n name,\n `Circuit breaker is open for server \"${name}\". ${lastError.message}`,\n )\n throw lastError\n }\n\n // Only retry on timeout errors, not on other errors\n if (isTimeoutError(error) && attempt < maxRetries) {\n logMCPError(\n name,\n `Tool \"${tool}\" timed out (attempt ${attempt + 1}/${maxRetries + 1}), retrying...`,\n )\n // Exponential backoff: 1s, 2s, 4s...\n await new Promise(resolve =>\n setTimeout(resolve, 1000 * Math.pow(2, attempt)),\n )\n continue\n }\n\n // Log and throw for non-retryable errors or after all retries exhausted\n if (isTimeoutError(error)) {\n logMCPError(\n name,\n `Tool \"${tool}\" failed after ${maxRetries + 1} attempts due to timeout`,\n )\n }\n throw lastError\n }\n }\n\n // This should never be reached, but TypeScript needs it\n throw lastError || new Error(`MCP tool ${tool} failed with unknown error`)\n}\n\nexport const getMCPCommands = memoize(async (): Promise<Command[]> => {\n const results = await requestAll<\n ListPromptsResult,\n typeof ListPromptsResultSchema\n >(\n {\n method: 'prompts/list',\n },\n ListPromptsResultSchema,\n 'prompts',\n )\n\n return results.flatMap(({ client, result }) =>\n result.prompts?.map(_ => {\n const argNames = Object.values(_.arguments ?? {}).map(k => k.name)\n return {\n type: 'prompt',\n name: 'mcp__' + client.name + '__' + _.name,\n description: _.description ?? '',\n isEnabled: true,\n isHidden: false,\n progressMessage: 'running',\n userFacingName() {\n return `${client.name}:${_.name} (MCP)`\n },\n argNames,\n async getPromptForCommand(args: string) {\n const argsArray = args.split(' ')\n return await runCommand(\n { name: _.name, client },\n zipObject(argNames, argsArray),\n )\n },\n }\n }),\n )\n})\n\nexport async function runCommand(\n { name, client }: { name: string; client: ConnectedClient },\n args: Record<string, string>,\n): Promise<MessageParam[]> {\n try {\n const result = await client.client.getPrompt({ name, arguments: args })\n return result.messages.map(\n (message): MessageParam => ({\n role: message.role,\n content: [\n message.content.type === 'text'\n ? {\n type: 'text',\n text: message.content.text,\n }\n : message.content.type === 'resource'\n ? {\n type: 'text',\n text:\n typeof (message.content as any).resource?.text === 'string'\n ? (message.content as any).resource.text\n : `[MCP Resource: ${(message.content as any).resource?.uri ?? 'unknown'}]`,\n }\n : {\n type: 'image',\n source: {\n data: String(message.content.data),\n media_type: message.content\n .mimeType as ImageBlockParam.Source['media_type'],\n type: 'base64',\n },\n },\n ],\n }),\n )\n } catch (error) {\n logMCPError(\n client.name,\n `Error running command '${name}': ${error instanceof Error ? error.message : String(error)}`,\n )\n throw error\n }\n}\n\n/**\n * Close a list of MCP clients in the background (fire-and-forget).\n * Does not block the caller \u2014 errors are silently ignored.\n */\nfunction closeClientsInBackground(clients: WrappedClient[]): void {\n const CLOSE_TIMEOUT_MS = 5000\n const closePromises = clients.map(async wrappedClient => {\n if (wrappedClient.type === 'connected') {\n try {\n await wrappedClient.client.close()\n } catch (e) {\n debugLogger.warn(\n 'MCP_CLOSE',\n `Failed to close MCP client: ${e instanceof Error ? e.message : String(e)}`,\n )\n }\n }\n })\n\n // Race against timeout so orphaned closes don't linger forever\n Promise.race([\n Promise.allSettled(closePromises),\n new Promise(resolve => setTimeout(resolve, CLOSE_TIMEOUT_MS)),\n ]).catch(() => {})\n}\n\n/**\n * Refreshes MCP connections by clearing caches and allowing fresh\n * reconnections on next getClients() call.\n *\n * Old connections are closed in the background AFTER cache clear,\n * so new connections can be established independently without waiting\n * for old ones to shut down (which could fail or be slow).\n *\n * Uses an async lock to prevent concurrent execution and race conditions.\n * Also resets circuit breakers to allow reconnection attempts.\n */\nexport async function refreshMCPConnections(): Promise<void> {\n // Early return if refresh is already pending to avoid request accumulation\n if (pendingRefresh) {\n return\n }\n\n pendingRefresh = true\n\n const release = await acquireConnectionLock()\n\n try {\n // Snapshot old connections for background cleanup\n const oldClients = connectedClients\n connectedClients = []\n\n // Clear memoization caches so next getClients() reconnects fresh\n if (typeof getClients.cache?.clear === 'function') {\n getClients.cache.clear()\n }\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n if (typeof getMCPCommands.cache?.clear === 'function') {\n getMCPCommands.cache.clear()\n }\n\n // Reset circuit breakers to allow reconnection attempts\n mcpCircuitBreakers.resetAll()\n\n // Clear deferred/activated tool state so it's rebuilt fresh\n clearDeferredState()\n\n // Also invalidate the main tools cache since MCP tools are part of it\n // This is imported lazily to avoid circular dependencies\n try {\n const { invalidateToolsCache } = await import('../tools')\n invalidateToolsCache()\n } catch {\n // Ignore errors - tools.ts may not be loaded yet\n }\n\n // Close old connections in background \u2014 don't block the refresh.\n // New connections (via getClients()) will be created independently.\n if (oldClients.length > 0) {\n closeClientsInBackground(oldClients)\n }\n } finally {\n pendingRefresh = false\n release()\n }\n}\n\n/**\n * Shuts down all MCP client connections gracefully.\n * Should be called when the application is exiting.\n * Unlike refresh, shutdown WAITS for connections to close.\n */\nexport async function shutdownMCPClients(): Promise<void> {\n // Stop health check timer to prevent checks on closing/closed clients\n stopHealthCheck()\n\n if (connectedClients.length === 0) return\n\n const oldClients = connectedClients\n connectedClients = []\n\n const SHUTDOWN_TIMEOUT_MS = 3000\n const closePromises = oldClients.map(async wrappedClient => {\n if (wrappedClient.type === 'connected') {\n try {\n await wrappedClient.client.close()\n } catch {\n // Ignore errors during shutdown\n }\n }\n })\n\n try {\n await Promise.race([\n Promise.allSettled(closePromises),\n new Promise(resolve => setTimeout(resolve, SHUTDOWN_TIMEOUT_MS)),\n ])\n } catch {\n // Ignore timeout errors\n }\n\n // Clear caches so no stale state remains\n if (typeof getClients.cache?.clear === 'function') {\n getClients.cache.clear()\n }\n if (typeof getMCPTools.cache?.clear === 'function') {\n getMCPTools.cache.clear()\n }\n if (typeof getMCPCommands.cache?.clear === 'function') {\n getMCPCommands.cache.clear()\n }\n}\n\n/**\n * MCP Resource type for listing\n */\nexport type McpResource = {\n uri: string\n name: string\n description?: string\n mimeType?: string\n serverName: string\n}\n\n/**\n * List all resources from MCP servers\n * @param serverFilter Optional server name to filter resources from a specific server\n */\nexport async function listMCPResources(\n serverFilter?: string,\n): Promise<McpResource[]> {\n const results = await requestAll<\n ListResourcesResult,\n typeof ListResourcesResultSchema\n >(\n {\n method: 'resources/list',\n },\n ListResourcesResultSchema,\n 'resources',\n )\n\n return results\n .filter(({ client }) => !serverFilter || client.name === serverFilter)\n .flatMap(({ client, result }) =>\n (result.resources || []).map(resource => ({\n uri: resource.uri,\n name: resource.name,\n description: resource.description,\n mimeType: resource.mimeType,\n serverName: client.name,\n })),\n )\n}\n\n/**\n * MCP Resource content type\n */\nexport type McpResourceContent = {\n uri: string\n mimeType?: string\n text?: string\n blob?: string // Base64 encoded binary data\n}\n\n/**\n * Read a specific resource from an MCP server\n * @param uri The resource URI to read\n * @param serverName Optional server name to target a specific server\n */\nexport async function readMCPResource(\n uri: string,\n serverName?: string,\n): Promise<McpResourceContent | null> {\n const clients = await getClients()\n\n // Find the appropriate client\n const targetClients = serverName\n ? (clients.filter(\n c => c.type === 'connected' && c.name === serverName,\n ) as ConnectedClient[])\n : (clients.filter(c => c.type === 'connected') as ConnectedClient[])\n\n for (const client of targetClients) {\n try {\n // Check if this server has resources capability\n const capabilities = await client.client.getServerCapabilities()\n if (!capabilities?.resources) {\n continue\n }\n\n // Try to read the resource\n const result = (await client.client.request(\n {\n method: 'resources/read',\n params: { uri },\n },\n ReadResourceResultSchema,\n )) as ReadResourceResult\n\n if (result.contents && result.contents.length > 0) {\n const content = result.contents[0]\n return {\n uri: content.uri,\n mimeType: content.mimeType,\n text: 'text' in content ? String(content.text) : undefined,\n blob: 'blob' in content ? String(content.blob) : undefined,\n }\n } else {\n emitReminderEvent('mcp:resource_no_content', { resourceUri: uri })\n }\n } catch (error) {\n // Log but continue trying other servers\n logMCPError(\n client.name,\n `Failed to read resource '${uri}': ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n }\n\n return null\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -0,0 +1,299 @@
1
+ import { z } from "zod";
2
+ import { MACRO } from "../constants/macros.js";
3
+ import { setApiKey, getApiKey, deleteApiKey } from "../utils/credentials/index.js";
4
+ import { getModelManager, reloadModelManager } from "../utils/model.js";
5
+ import { t } from "../i18n/index.js";
6
+ import * as readline from "readline";
7
+ const MintoLoginResponseSchema = z.object({
8
+ user: z.string(),
9
+ profile: z.object({
10
+ userId: z.string(),
11
+ staff: z.string(),
12
+ email: z.string(),
13
+ permission: z.array(z.string()),
14
+ status: z.string()
15
+ }),
16
+ model: z.object({
17
+ base_url: z.string(),
18
+ api_key: z.string(),
19
+ qwen_asr_api_key: z.string().optional()
20
+ })
21
+ });
22
+ const LiteLLMModelInfoSchema = z.object({
23
+ model_name: z.string(),
24
+ litellm_params: z.record(z.unknown()).optional().default({}),
25
+ model_info: z.object({
26
+ id: z.string().optional(),
27
+ key: z.string().optional(),
28
+ max_tokens: z.number().optional(),
29
+ max_input_tokens: z.number().optional(),
30
+ max_output_tokens: z.number().optional(),
31
+ input_cost_per_token: z.number().optional(),
32
+ output_cost_per_token: z.number().optional(),
33
+ litellm_provider: z.string().optional(),
34
+ mode: z.string().optional()
35
+ }).optional().default({})
36
+ });
37
+ const LiteLLMModelListResponseSchema = z.object({
38
+ data: z.array(LiteLLMModelInfoSchema)
39
+ });
40
+ const LITELLM_CREDENTIAL_KEY = "minto-litellm";
41
+ async function mintoLogin(user, password) {
42
+ const url = MACRO.LOGIN_API_URL;
43
+ const res = await fetch(url, {
44
+ method: "POST",
45
+ headers: { "Content-Type": "application/json" },
46
+ body: JSON.stringify({ user, password })
47
+ });
48
+ if (res.status === 401 || res.status === 403) {
49
+ throw new Error(t("commands.login.invalidCredentials"));
50
+ }
51
+ if (!res.ok) {
52
+ throw new Error(
53
+ `${t("commands.login.failed")}: HTTP ${res.status} ${res.statusText}`
54
+ );
55
+ }
56
+ const json = await res.json();
57
+ return MintoLoginResponseSchema.parse(json);
58
+ }
59
+ async function fetchLiteLLMModels(baseUrl, apiKey) {
60
+ const base = baseUrl.replace(/\/+$/, "");
61
+ const url = `${base}/model/info`;
62
+ const res = await fetch(url, {
63
+ method: "GET",
64
+ headers: {
65
+ Authorization: `Bearer ${apiKey}`,
66
+ "Content-Type": "application/json"
67
+ }
68
+ });
69
+ if (!res.ok) {
70
+ throw new Error(
71
+ `${t("commands.login.fetchModelsFailed")}: HTTP ${res.status}`
72
+ );
73
+ }
74
+ const json = await res.json();
75
+ const parsed = LiteLLMModelListResponseSchema.parse(json);
76
+ return parsed.data;
77
+ }
78
+ function deduplicateLiteLLMModels(models) {
79
+ const seen = /* @__PURE__ */ new Map();
80
+ for (const model of models) {
81
+ const baseName = getBaseModelName(model.model_name);
82
+ if (!seen.has(baseName)) {
83
+ seen.set(baseName, model);
84
+ }
85
+ }
86
+ return [...seen.values()];
87
+ }
88
+ function getBaseModelName(modelName) {
89
+ const slashIdx = modelName.indexOf("/");
90
+ return slashIdx >= 0 ? modelName.slice(slashIdx + 1) : modelName;
91
+ }
92
+ function sortModelsByCost(models) {
93
+ return [...models].sort((a, b) => {
94
+ const costA = (a.model_info?.input_cost_per_token ?? 0) + (a.model_info?.output_cost_per_token ?? 0);
95
+ const costB = (b.model_info?.input_cost_per_token ?? 0) + (b.model_info?.output_cost_per_token ?? 0);
96
+ return costA - costB;
97
+ });
98
+ }
99
+ function assignModelPointers(modelNames) {
100
+ const find = (...patterns) => {
101
+ for (const pattern of patterns) {
102
+ const match = modelNames.find(
103
+ (n) => n.toLowerCase().includes(pattern.toLowerCase())
104
+ );
105
+ if (match) return match;
106
+ }
107
+ return null;
108
+ };
109
+ const fallback = modelNames[0] ?? "";
110
+ const main = find("glm-5", "qwen-3.5-plus", "claude-sonnet", "gpt-4o") ?? fallback;
111
+ const task = find("claude-sonnet", "gpt-4o-mini", "glm-5") ?? main;
112
+ const reasoning = find("o3", "o1", "deepseek-r1") ?? main;
113
+ const quick = find("gpt-4o-mini", "claude-haiku", "qwen-3.5-plus") ?? task;
114
+ const compact = quick;
115
+ return { main, task, reasoning, quick, compact };
116
+ }
117
+ async function configureAiterVoiceInput(qwenAsrApiKey) {
118
+ if (process.env.AITER_TERMINAL !== "1") return false;
119
+ const { execFile } = await import("child_process");
120
+ const { promisify } = await import("util");
121
+ const execFileAsync = promisify(execFile);
122
+ const voiceInputConfig = JSON.stringify({
123
+ enabled: true,
124
+ provider: "qwen-asr",
125
+ qwenApiKey: qwenAsrApiKey,
126
+ qwenRegion: "cn"
127
+ });
128
+ try {
129
+ await execFileAsync("aiter", [
130
+ "settings",
131
+ "set",
132
+ "voiceInput",
133
+ voiceInputConfig
134
+ ]);
135
+ return true;
136
+ } catch {
137
+ return false;
138
+ }
139
+ }
140
+ async function performLogin(user, password, onProgress) {
141
+ onProgress?.(t("commands.login.loggingIn"));
142
+ const loginResponse = await mintoLogin(user, password);
143
+ if (loginResponse.profile.status !== "active") {
144
+ throw new Error(t("commands.login.inactive"));
145
+ }
146
+ await setApiKey(LITELLM_CREDENTIAL_KEY, loginResponse.model.api_key);
147
+ onProgress?.(t("commands.login.configuringModels"));
148
+ const mm = getModelManager();
149
+ const existingModels = mm.getAllConfiguredModels();
150
+ const loginBaseURL = loginResponse.model.base_url.replace(/\/+$/, "");
151
+ for (const model of existingModels) {
152
+ if (model.baseURL && model.baseURL.replace(/\/+$/, "") === loginBaseURL) {
153
+ mm.removeModel(model.modelName);
154
+ }
155
+ }
156
+ onProgress?.(t("commands.login.fetchingModels"));
157
+ let litellmModels;
158
+ try {
159
+ litellmModels = await fetchLiteLLMModels(
160
+ loginResponse.model.base_url,
161
+ loginResponse.model.api_key
162
+ );
163
+ } catch (err) {
164
+ return {
165
+ user: loginResponse.user,
166
+ modelsConfigured: 0,
167
+ modelNames: []
168
+ };
169
+ }
170
+ const deduplicated = sortModelsByCost(deduplicateLiteLLMModels(litellmModels));
171
+ const configuredNames = [];
172
+ for (const model of deduplicated) {
173
+ const baseName = getBaseModelName(model.model_name);
174
+ const maxOutputTokens = model.model_info?.max_output_tokens ?? 4096;
175
+ const maxInputTokens = model.model_info?.max_input_tokens ?? 128e3;
176
+ try {
177
+ await mm.addModel({
178
+ name: baseName,
179
+ provider: "custom-openai",
180
+ modelName: baseName,
181
+ baseURL: loginBaseURL,
182
+ apiKey: `encrypted:${LITELLM_CREDENTIAL_KEY}`,
183
+ maxTokens: maxOutputTokens,
184
+ contextLength: maxInputTokens
185
+ });
186
+ configuredNames.push(baseName);
187
+ } catch {
188
+ }
189
+ }
190
+ if (configuredNames.length > 0) {
191
+ const pointers = assignModelPointers(configuredNames);
192
+ for (const [pointer, modelName] of Object.entries(pointers)) {
193
+ try {
194
+ mm.setPointer(pointer, modelName);
195
+ } catch {
196
+ }
197
+ }
198
+ }
199
+ reloadModelManager();
200
+ let aiterVoiceConfigured = false;
201
+ if (loginResponse.model.qwen_asr_api_key) {
202
+ aiterVoiceConfigured = await configureAiterVoiceInput(
203
+ loginResponse.model.qwen_asr_api_key
204
+ );
205
+ }
206
+ return {
207
+ user: loginResponse.user,
208
+ modelsConfigured: configuredNames.length,
209
+ modelNames: configuredNames,
210
+ aiterVoiceConfigured
211
+ };
212
+ }
213
+ async function isLoggedIn() {
214
+ const key = await getApiKey(LITELLM_CREDENTIAL_KEY);
215
+ return key !== null;
216
+ }
217
+ async function logout() {
218
+ await deleteApiKey(LITELLM_CREDENTIAL_KEY);
219
+ }
220
+ async function handleLoginCLI(user, password) {
221
+ const rl = readline.createInterface({
222
+ input: process.stdin,
223
+ output: process.stdout
224
+ });
225
+ const ask = (prompt) => new Promise((resolve) => rl.question(prompt, resolve));
226
+ const askPassword = (prompt) => new Promise((resolve) => {
227
+ process.stdout.write(prompt);
228
+ const stdin = process.stdin;
229
+ const wasRaw = stdin.isRaw;
230
+ if (stdin.setRawMode) stdin.setRawMode(true);
231
+ stdin.resume();
232
+ let pwd = "";
233
+ const onData = (ch) => {
234
+ const c = ch.toString("utf8");
235
+ if (c === "\n" || c === "\r") {
236
+ if (stdin.setRawMode) stdin.setRawMode(wasRaw ?? false);
237
+ stdin.removeListener("data", onData);
238
+ process.stdout.write("\n");
239
+ resolve(pwd);
240
+ } else if (c === "") {
241
+ process.exit(1);
242
+ } else if (c === "\x7F" || c === "\b") {
243
+ if (pwd.length > 0) {
244
+ pwd = pwd.slice(0, -1);
245
+ process.stdout.write("\b \b");
246
+ }
247
+ } else {
248
+ pwd += c;
249
+ process.stdout.write("*");
250
+ }
251
+ };
252
+ stdin.on("data", onData);
253
+ });
254
+ try {
255
+ if (!user) {
256
+ user = await ask("Username: ");
257
+ }
258
+ if (!password) {
259
+ password = await askPassword("Password: ");
260
+ }
261
+ rl.close();
262
+ console.log(`
263
+ ${t("commands.login.loggingIn")}...`);
264
+ const result = await performLogin(user, password, (phase) => {
265
+ console.log(` ${phase}...`);
266
+ });
267
+ console.log(`
268
+ \u2713 ${t("commands.login.success")}`);
269
+ console.log(` User: ${result.user}`);
270
+ console.log(
271
+ ` ${t("commands.login.modelsConfigured")}: ${result.modelsConfigured}`
272
+ );
273
+ if (result.modelNames.length > 0) {
274
+ console.log(` Models: ${result.modelNames.join(", ")}`);
275
+ }
276
+ if (result.aiterVoiceConfigured) {
277
+ console.log(` AiTer voice input: configured`);
278
+ }
279
+ } catch (err) {
280
+ rl.close();
281
+ console.error(
282
+ `
283
+ \u2717 ${t("commands.login.failed")}: ${err instanceof Error ? err.message : String(err)}`
284
+ );
285
+ process.exit(1);
286
+ }
287
+ }
288
+ export {
289
+ assignModelPointers,
290
+ deduplicateLiteLLMModels,
291
+ fetchLiteLLMModels,
292
+ handleLoginCLI,
293
+ isLoggedIn,
294
+ logout,
295
+ mintoLogin,
296
+ performLogin,
297
+ sortModelsByCost
298
+ };
299
+ //# sourceMappingURL=mintoAuth.js.map