@xelauvas/xela-cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1920) hide show
  1. package/README.md +200 -0
  2. package/bin/xela +100 -0
  3. package/package.json +88 -0
  4. package/src/QueryEngine.ts +1295 -0
  5. package/src/Task.ts +125 -0
  6. package/src/Tool.ts +792 -0
  7. package/src/_shims/_generated_stubs/_universal_stub.mjs +168 -0
  8. package/src/_shims/_generated_stubs/text_stub.mjs +1 -0
  9. package/src/_shims/bun_bundle.js +9 -0
  10. package/src/_shims/cjs_stub.cjs +23 -0
  11. package/src/_shims/empty_stub.js +33 -0
  12. package/src/_shims/loader.js +352 -0
  13. package/src/_shims/openai_adapter.ts +486 -0
  14. package/src/_shims/react_compiler_runtime.js +17 -0
  15. package/src/_shims/register.js +148 -0
  16. package/src/assistant/sessionHistory.ts +87 -0
  17. package/src/bootstrap/state.ts +1759 -0
  18. package/src/bridge/bridgeApi.ts +539 -0
  19. package/src/bridge/bridgeConfig.ts +48 -0
  20. package/src/bridge/bridgeDebug.ts +135 -0
  21. package/src/bridge/bridgeEnabled.ts +202 -0
  22. package/src/bridge/bridgeMain.ts +2999 -0
  23. package/src/bridge/bridgeMessaging.ts +461 -0
  24. package/src/bridge/bridgePermissionCallbacks.ts +43 -0
  25. package/src/bridge/bridgePointer.ts +210 -0
  26. package/src/bridge/bridgeStatusUtil.ts +163 -0
  27. package/src/bridge/bridgeUI.ts +530 -0
  28. package/src/bridge/capacityWake.ts +56 -0
  29. package/src/bridge/codeSessionApi.ts +168 -0
  30. package/src/bridge/createSession.ts +384 -0
  31. package/src/bridge/debugUtils.ts +141 -0
  32. package/src/bridge/envLessBridgeConfig.ts +165 -0
  33. package/src/bridge/flushGate.ts +71 -0
  34. package/src/bridge/inboundAttachments.ts +175 -0
  35. package/src/bridge/inboundMessages.ts +80 -0
  36. package/src/bridge/initReplBridge.ts +569 -0
  37. package/src/bridge/jwtUtils.ts +256 -0
  38. package/src/bridge/pollConfig.ts +110 -0
  39. package/src/bridge/pollConfigDefaults.ts +82 -0
  40. package/src/bridge/remoteBridgeCore.ts +1008 -0
  41. package/src/bridge/replBridge.ts +2406 -0
  42. package/src/bridge/replBridgeHandle.ts +36 -0
  43. package/src/bridge/replBridgeTransport.ts +370 -0
  44. package/src/bridge/sessionIdCompat.ts +57 -0
  45. package/src/bridge/sessionRunner.ts +550 -0
  46. package/src/bridge/trustedDevice.ts +210 -0
  47. package/src/bridge/types.ts +262 -0
  48. package/src/bridge/workSecret.ts +127 -0
  49. package/src/buddy/CompanionSprite.tsx +371 -0
  50. package/src/buddy/companion.ts +133 -0
  51. package/src/buddy/prompt.ts +36 -0
  52. package/src/buddy/sprites.ts +514 -0
  53. package/src/buddy/types.ts +148 -0
  54. package/src/buddy/useBuddyNotification.tsx +98 -0
  55. package/src/cli/exit.ts +31 -0
  56. package/src/cli/handlers/agents.ts +70 -0
  57. package/src/cli/handlers/auth.ts +330 -0
  58. package/src/cli/handlers/autoMode.ts +170 -0
  59. package/src/cli/handlers/mcp.tsx +362 -0
  60. package/src/cli/handlers/plugins.ts +878 -0
  61. package/src/cli/handlers/util.tsx +110 -0
  62. package/src/cli/ndjsonSafeStringify.ts +32 -0
  63. package/src/cli/print.ts +5594 -0
  64. package/src/cli/remoteIO.ts +255 -0
  65. package/src/cli/structuredIO.ts +859 -0
  66. package/src/cli/transports/HybridTransport.ts +282 -0
  67. package/src/cli/transports/SSETransport.ts +711 -0
  68. package/src/cli/transports/SerialBatchEventUploader.ts +275 -0
  69. package/src/cli/transports/WebSocketTransport.ts +800 -0
  70. package/src/cli/transports/WorkerStateUploader.ts +131 -0
  71. package/src/cli/transports/ccrClient.ts +998 -0
  72. package/src/cli/transports/transportUtils.ts +45 -0
  73. package/src/cli/update.ts +422 -0
  74. package/src/commands/add-dir/add-dir.tsx +126 -0
  75. package/src/commands/add-dir/index.ts +11 -0
  76. package/src/commands/add-dir/validation.ts +110 -0
  77. package/src/commands/advisor.ts +109 -0
  78. package/src/commands/agents/agents.tsx +12 -0
  79. package/src/commands/agents/index.ts +10 -0
  80. package/src/commands/ant-trace/index.js +1 -0
  81. package/src/commands/autofix-pr/index.js +1 -0
  82. package/src/commands/backfill-sessions/index.js +1 -0
  83. package/src/commands/branch/branch.ts +296 -0
  84. package/src/commands/branch/index.ts +14 -0
  85. package/src/commands/break-cache/index.js +1 -0
  86. package/src/commands/bridge/bridge.tsx +509 -0
  87. package/src/commands/bridge/index.ts +26 -0
  88. package/src/commands/bridge-kick.ts +200 -0
  89. package/src/commands/brief.ts +130 -0
  90. package/src/commands/btw/btw.tsx +243 -0
  91. package/src/commands/btw/index.ts +13 -0
  92. package/src/commands/bughunter/index.js +1 -0
  93. package/src/commands/chrome/chrome.tsx +285 -0
  94. package/src/commands/chrome/index.ts +13 -0
  95. package/src/commands/clear/caches.ts +144 -0
  96. package/src/commands/clear/clear.ts +7 -0
  97. package/src/commands/clear/conversation.ts +251 -0
  98. package/src/commands/clear/index.ts +19 -0
  99. package/src/commands/color/color.ts +93 -0
  100. package/src/commands/color/index.ts +16 -0
  101. package/src/commands/commit-push-pr.ts +158 -0
  102. package/src/commands/commit.ts +92 -0
  103. package/src/commands/compact/compact.ts +287 -0
  104. package/src/commands/compact/index.ts +15 -0
  105. package/src/commands/config/config.tsx +7 -0
  106. package/src/commands/config/index.ts +11 -0
  107. package/src/commands/context/context-noninteractive.ts +325 -0
  108. package/src/commands/context/context.tsx +64 -0
  109. package/src/commands/context/index.ts +24 -0
  110. package/src/commands/copy/copy.tsx +371 -0
  111. package/src/commands/copy/index.ts +15 -0
  112. package/src/commands/cost/cost.ts +24 -0
  113. package/src/commands/cost/index.ts +23 -0
  114. package/src/commands/createMovedToPluginCommand.ts +65 -0
  115. package/src/commands/ctx_viz/index.js +1 -0
  116. package/src/commands/debug-tool-call/index.js +1 -0
  117. package/src/commands/desktop/desktop.tsx +9 -0
  118. package/src/commands/desktop/index.ts +26 -0
  119. package/src/commands/diff/diff.tsx +9 -0
  120. package/src/commands/diff/index.ts +8 -0
  121. package/src/commands/doctor/doctor.tsx +7 -0
  122. package/src/commands/doctor/index.ts +12 -0
  123. package/src/commands/effort/effort.tsx +183 -0
  124. package/src/commands/effort/index.ts +13 -0
  125. package/src/commands/env/index.js +1 -0
  126. package/src/commands/exit/exit.tsx +33 -0
  127. package/src/commands/exit/index.ts +12 -0
  128. package/src/commands/export/export.tsx +91 -0
  129. package/src/commands/export/index.ts +11 -0
  130. package/src/commands/extra-usage/extra-usage-core.ts +118 -0
  131. package/src/commands/extra-usage/extra-usage-noninteractive.ts +16 -0
  132. package/src/commands/extra-usage/extra-usage.tsx +17 -0
  133. package/src/commands/extra-usage/index.ts +31 -0
  134. package/src/commands/fast/fast.tsx +269 -0
  135. package/src/commands/fast/index.ts +26 -0
  136. package/src/commands/feedback/feedback.tsx +25 -0
  137. package/src/commands/feedback/index.ts +26 -0
  138. package/src/commands/files/files.ts +19 -0
  139. package/src/commands/files/index.ts +12 -0
  140. package/src/commands/good-claude/index.js +1 -0
  141. package/src/commands/heapdump/heapdump.ts +17 -0
  142. package/src/commands/heapdump/index.ts +12 -0
  143. package/src/commands/help/help.tsx +11 -0
  144. package/src/commands/help/index.ts +10 -0
  145. package/src/commands/hooks/hooks.tsx +13 -0
  146. package/src/commands/hooks/index.ts +11 -0
  147. package/src/commands/ide/ide.tsx +646 -0
  148. package/src/commands/ide/index.ts +11 -0
  149. package/src/commands/init-verifiers.ts +262 -0
  150. package/src/commands/init.ts +256 -0
  151. package/src/commands/insights.ts +3200 -0
  152. package/src/commands/install-github-app/ApiKeyStep.tsx +231 -0
  153. package/src/commands/install-github-app/CheckExistingSecretStep.tsx +190 -0
  154. package/src/commands/install-github-app/CheckGitHubStep.tsx +15 -0
  155. package/src/commands/install-github-app/ChooseRepoStep.tsx +211 -0
  156. package/src/commands/install-github-app/CreatingStep.tsx +65 -0
  157. package/src/commands/install-github-app/ErrorStep.tsx +85 -0
  158. package/src/commands/install-github-app/ExistingWorkflowStep.tsx +103 -0
  159. package/src/commands/install-github-app/InstallAppStep.tsx +94 -0
  160. package/src/commands/install-github-app/OAuthFlowStep.tsx +276 -0
  161. package/src/commands/install-github-app/SuccessStep.tsx +96 -0
  162. package/src/commands/install-github-app/WarningsStep.tsx +73 -0
  163. package/src/commands/install-github-app/index.ts +13 -0
  164. package/src/commands/install-github-app/install-github-app.tsx +587 -0
  165. package/src/commands/install-github-app/setupGitHubActions.ts +325 -0
  166. package/src/commands/install-slack-app/index.ts +12 -0
  167. package/src/commands/install-slack-app/install-slack-app.ts +30 -0
  168. package/src/commands/install.tsx +300 -0
  169. package/src/commands/issue/index.js +1 -0
  170. package/src/commands/keybindings/index.ts +13 -0
  171. package/src/commands/keybindings/keybindings.ts +53 -0
  172. package/src/commands/login/index.ts +14 -0
  173. package/src/commands/login/login.tsx +104 -0
  174. package/src/commands/logout/index.ts +10 -0
  175. package/src/commands/logout/logout.tsx +82 -0
  176. package/src/commands/mcp/addCommand.ts +280 -0
  177. package/src/commands/mcp/index.ts +12 -0
  178. package/src/commands/mcp/mcp.tsx +85 -0
  179. package/src/commands/mcp/xaaIdpCommand.ts +266 -0
  180. package/src/commands/memory/index.ts +10 -0
  181. package/src/commands/memory/memory.tsx +90 -0
  182. package/src/commands/mobile/index.ts +11 -0
  183. package/src/commands/mobile/mobile.tsx +274 -0
  184. package/src/commands/mock-limits/index.js +1 -0
  185. package/src/commands/model/index.ts +16 -0
  186. package/src/commands/model/model.tsx +297 -0
  187. package/src/commands/oauth-refresh/index.js +1 -0
  188. package/src/commands/onboarding/index.js +1 -0
  189. package/src/commands/output-style/index.ts +11 -0
  190. package/src/commands/output-style/output-style.tsx +7 -0
  191. package/src/commands/passes/index.ts +22 -0
  192. package/src/commands/passes/passes.tsx +24 -0
  193. package/src/commands/perf-issue/index.js +1 -0
  194. package/src/commands/permissions/index.ts +11 -0
  195. package/src/commands/permissions/permissions.tsx +10 -0
  196. package/src/commands/plan/index.ts +11 -0
  197. package/src/commands/plan/plan.tsx +122 -0
  198. package/src/commands/plugin/AddMarketplace.tsx +162 -0
  199. package/src/commands/plugin/BrowseMarketplace.tsx +802 -0
  200. package/src/commands/plugin/DiscoverPlugins.tsx +781 -0
  201. package/src/commands/plugin/ManageMarketplaces.tsx +838 -0
  202. package/src/commands/plugin/ManagePlugins.tsx +2215 -0
  203. package/src/commands/plugin/PluginErrors.tsx +124 -0
  204. package/src/commands/plugin/PluginOptionsDialog.tsx +357 -0
  205. package/src/commands/plugin/PluginOptionsFlow.tsx +135 -0
  206. package/src/commands/plugin/PluginSettings.tsx +1072 -0
  207. package/src/commands/plugin/PluginTrustWarning.tsx +32 -0
  208. package/src/commands/plugin/UnifiedInstalledCell.tsx +565 -0
  209. package/src/commands/plugin/ValidatePlugin.tsx +98 -0
  210. package/src/commands/plugin/index.tsx +11 -0
  211. package/src/commands/plugin/parseArgs.ts +103 -0
  212. package/src/commands/plugin/plugin.tsx +7 -0
  213. package/src/commands/plugin/pluginDetailsHelpers.tsx +117 -0
  214. package/src/commands/plugin/usePagination.ts +171 -0
  215. package/src/commands/pr_comments/index.ts +50 -0
  216. package/src/commands/privacy-settings/index.ts +14 -0
  217. package/src/commands/privacy-settings/privacy-settings.tsx +58 -0
  218. package/src/commands/rate-limit-options/index.ts +19 -0
  219. package/src/commands/rate-limit-options/rate-limit-options.tsx +210 -0
  220. package/src/commands/release-notes/index.ts +11 -0
  221. package/src/commands/release-notes/release-notes.ts +50 -0
  222. package/src/commands/reload-plugins/index.ts +18 -0
  223. package/src/commands/reload-plugins/reload-plugins.ts +61 -0
  224. package/src/commands/remote-env/index.ts +15 -0
  225. package/src/commands/remote-env/remote-env.tsx +7 -0
  226. package/src/commands/remote-setup/api.ts +182 -0
  227. package/src/commands/remote-setup/index.ts +20 -0
  228. package/src/commands/remote-setup/remote-setup.tsx +187 -0
  229. package/src/commands/rename/generateSessionName.ts +67 -0
  230. package/src/commands/rename/index.ts +12 -0
  231. package/src/commands/rename/rename.ts +87 -0
  232. package/src/commands/reset-limits/index.js +4 -0
  233. package/src/commands/resume/index.ts +12 -0
  234. package/src/commands/resume/resume.tsx +275 -0
  235. package/src/commands/review/UltrareviewOverageDialog.tsx +96 -0
  236. package/src/commands/review/reviewRemote.ts +316 -0
  237. package/src/commands/review/ultrareviewCommand.tsx +58 -0
  238. package/src/commands/review/ultrareviewEnabled.ts +14 -0
  239. package/src/commands/review.ts +57 -0
  240. package/src/commands/rewind/index.ts +13 -0
  241. package/src/commands/rewind/rewind.ts +13 -0
  242. package/src/commands/sandbox-toggle/index.ts +50 -0
  243. package/src/commands/sandbox-toggle/sandbox-toggle.tsx +83 -0
  244. package/src/commands/security-review.ts +243 -0
  245. package/src/commands/session/index.ts +16 -0
  246. package/src/commands/session/session.tsx +140 -0
  247. package/src/commands/share/index.js +1 -0
  248. package/src/commands/skills/index.ts +10 -0
  249. package/src/commands/skills/skills.tsx +8 -0
  250. package/src/commands/stats/index.ts +10 -0
  251. package/src/commands/stats/stats.tsx +7 -0
  252. package/src/commands/status/index.ts +12 -0
  253. package/src/commands/status/status.tsx +8 -0
  254. package/src/commands/statusline.tsx +24 -0
  255. package/src/commands/stickers/index.ts +11 -0
  256. package/src/commands/stickers/stickers.ts +16 -0
  257. package/src/commands/summary/index.js +1 -0
  258. package/src/commands/tag/index.ts +12 -0
  259. package/src/commands/tag/tag.tsx +215 -0
  260. package/src/commands/tasks/index.ts +11 -0
  261. package/src/commands/tasks/tasks.tsx +8 -0
  262. package/src/commands/teleport/index.js +1 -0
  263. package/src/commands/terminalSetup/index.ts +23 -0
  264. package/src/commands/terminalSetup/terminalSetup.tsx +531 -0
  265. package/src/commands/theme/index.ts +10 -0
  266. package/src/commands/theme/theme.tsx +57 -0
  267. package/src/commands/thinkback/index.ts +13 -0
  268. package/src/commands/thinkback/thinkback.tsx +554 -0
  269. package/src/commands/thinkback-play/index.ts +17 -0
  270. package/src/commands/thinkback-play/thinkback-play.ts +43 -0
  271. package/src/commands/ultraplan.tsx +471 -0
  272. package/src/commands/upgrade/index.ts +16 -0
  273. package/src/commands/upgrade/upgrade.tsx +38 -0
  274. package/src/commands/usage/index.ts +9 -0
  275. package/src/commands/usage/usage.tsx +7 -0
  276. package/src/commands/version.ts +22 -0
  277. package/src/commands/vim/index.ts +11 -0
  278. package/src/commands/vim/vim.ts +38 -0
  279. package/src/commands/voice/index.ts +20 -0
  280. package/src/commands/voice/voice.ts +150 -0
  281. package/src/commands.ts +754 -0
  282. package/src/components/AgentProgressLine.tsx +136 -0
  283. package/src/components/App.tsx +56 -0
  284. package/src/components/ApproveApiKey.tsx +123 -0
  285. package/src/components/AutoModeOptInDialog.tsx +142 -0
  286. package/src/components/AutoUpdater.tsx +198 -0
  287. package/src/components/AutoUpdaterWrapper.tsx +91 -0
  288. package/src/components/AwsAuthStatusBox.tsx +82 -0
  289. package/src/components/BaseTextInput.tsx +136 -0
  290. package/src/components/BashModeProgress.tsx +56 -0
  291. package/src/components/BridgeDialog.tsx +401 -0
  292. package/src/components/BypassPermissionsModeDialog.tsx +87 -0
  293. package/src/components/ChannelDowngradeDialog.tsx +102 -0
  294. package/src/components/ClaudeCodeHint/PluginHintMenu.tsx +78 -0
  295. package/src/components/ClaudeInChromeOnboarding.tsx +121 -0
  296. package/src/components/ClaudeMdExternalIncludesDialog.tsx +137 -0
  297. package/src/components/ClickableImageRef.tsx +73 -0
  298. package/src/components/CompactSummary.tsx +118 -0
  299. package/src/components/ConfigurableShortcutHint.tsx +57 -0
  300. package/src/components/ConsoleOAuthFlow.tsx +631 -0
  301. package/src/components/ContextSuggestions.tsx +47 -0
  302. package/src/components/ContextVisualization.tsx +489 -0
  303. package/src/components/CoordinatorAgentStatus.tsx +273 -0
  304. package/src/components/CostThresholdDialog.tsx +50 -0
  305. package/src/components/CtrlOToExpand.tsx +51 -0
  306. package/src/components/CustomSelect/SelectMulti.tsx +213 -0
  307. package/src/components/CustomSelect/index.ts +3 -0
  308. package/src/components/CustomSelect/option-map.ts +50 -0
  309. package/src/components/CustomSelect/select-input-option.tsx +488 -0
  310. package/src/components/CustomSelect/select-option.tsx +68 -0
  311. package/src/components/CustomSelect/select.tsx +690 -0
  312. package/src/components/CustomSelect/use-multi-select-state.ts +414 -0
  313. package/src/components/CustomSelect/use-select-input.ts +287 -0
  314. package/src/components/CustomSelect/use-select-navigation.ts +653 -0
  315. package/src/components/CustomSelect/use-select-state.ts +157 -0
  316. package/src/components/DesktopHandoff.tsx +193 -0
  317. package/src/components/DesktopUpsell/DesktopUpsellStartup.tsx +171 -0
  318. package/src/components/DevBar.tsx +49 -0
  319. package/src/components/DevChannelsDialog.tsx +105 -0
  320. package/src/components/DiagnosticsDisplay.tsx +95 -0
  321. package/src/components/EffortCallout.tsx +265 -0
  322. package/src/components/EffortIndicator.ts +42 -0
  323. package/src/components/ExitFlow.tsx +48 -0
  324. package/src/components/ExportDialog.tsx +128 -0
  325. package/src/components/FallbackToolUseErrorMessage.tsx +116 -0
  326. package/src/components/FallbackToolUseRejectedMessage.tsx +16 -0
  327. package/src/components/FastIcon.tsx +46 -0
  328. package/src/components/Feedback.tsx +592 -0
  329. package/src/components/FeedbackSurvey/FeedbackSurvey.tsx +174 -0
  330. package/src/components/FeedbackSurvey/FeedbackSurveyView.tsx +108 -0
  331. package/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx +88 -0
  332. package/src/components/FeedbackSurvey/submitTranscriptShare.ts +112 -0
  333. package/src/components/FeedbackSurvey/useDebouncedDigitInput.ts +82 -0
  334. package/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +296 -0
  335. package/src/components/FeedbackSurvey/useMemorySurvey.tsx +213 -0
  336. package/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +206 -0
  337. package/src/components/FeedbackSurvey/useSurveyState.tsx +100 -0
  338. package/src/components/FileEditToolDiff.tsx +181 -0
  339. package/src/components/FileEditToolUpdatedMessage.tsx +124 -0
  340. package/src/components/FileEditToolUseRejectedMessage.tsx +170 -0
  341. package/src/components/FilePathLink.tsx +43 -0
  342. package/src/components/FullscreenLayout.tsx +637 -0
  343. package/src/components/GlobalSearchDialog.tsx +343 -0
  344. package/src/components/HelpV2/Commands.tsx +82 -0
  345. package/src/components/HelpV2/General.tsx +23 -0
  346. package/src/components/HelpV2/HelpV2.tsx +184 -0
  347. package/src/components/HighlightedCode/Fallback.tsx +193 -0
  348. package/src/components/HighlightedCode.tsx +190 -0
  349. package/src/components/HistorySearchDialog.tsx +118 -0
  350. package/src/components/IdeAutoConnectDialog.tsx +154 -0
  351. package/src/components/IdeOnboardingDialog.tsx +167 -0
  352. package/src/components/IdeStatusIndicator.tsx +58 -0
  353. package/src/components/IdleReturnDialog.tsx +118 -0
  354. package/src/components/InterruptedByUser.tsx +15 -0
  355. package/src/components/InvalidConfigDialog.tsx +156 -0
  356. package/src/components/InvalidSettingsDialog.tsx +89 -0
  357. package/src/components/KeybindingWarnings.tsx +55 -0
  358. package/src/components/LanguagePicker.tsx +86 -0
  359. package/src/components/LogSelector.tsx +1575 -0
  360. package/src/components/LogoV2/AnimatedAsterisk.tsx +50 -0
  361. package/src/components/LogoV2/AnimatedClawd.tsx +124 -0
  362. package/src/components/LogoV2/ChannelsNotice.tsx +266 -0
  363. package/src/components/LogoV2/Clawd.tsx +240 -0
  364. package/src/components/LogoV2/CondensedLogo.tsx +161 -0
  365. package/src/components/LogoV2/EmergencyTip.tsx +58 -0
  366. package/src/components/LogoV2/Feed.tsx +112 -0
  367. package/src/components/LogoV2/FeedColumn.tsx +59 -0
  368. package/src/components/LogoV2/GuestPassesUpsell.tsx +70 -0
  369. package/src/components/LogoV2/LogoV2.tsx +543 -0
  370. package/src/components/LogoV2/Opus1mMergeNotice.tsx +55 -0
  371. package/src/components/LogoV2/OverageCreditUpsell.tsx +166 -0
  372. package/src/components/LogoV2/VoiceModeNotice.tsx +68 -0
  373. package/src/components/LogoV2/WelcomeV2.tsx +433 -0
  374. package/src/components/LogoV2/feedConfigs.tsx +92 -0
  375. package/src/components/LspRecommendation/LspRecommendationMenu.tsx +88 -0
  376. package/src/components/MCPServerApprovalDialog.tsx +115 -0
  377. package/src/components/MCPServerDesktopImportDialog.tsx +203 -0
  378. package/src/components/MCPServerDialogCopy.tsx +15 -0
  379. package/src/components/MCPServerMultiselectDialog.tsx +133 -0
  380. package/src/components/ManagedSettingsSecurityDialog/ManagedSettingsSecurityDialog.tsx +149 -0
  381. package/src/components/ManagedSettingsSecurityDialog/utils.ts +144 -0
  382. package/src/components/Markdown.tsx +236 -0
  383. package/src/components/MarkdownTable.tsx +322 -0
  384. package/src/components/MemoryUsageIndicator.tsx +37 -0
  385. package/src/components/Message.tsx +627 -0
  386. package/src/components/MessageModel.tsx +43 -0
  387. package/src/components/MessageResponse.tsx +78 -0
  388. package/src/components/MessageRow.tsx +383 -0
  389. package/src/components/MessageSelector.tsx +831 -0
  390. package/src/components/MessageTimestamp.tsx +63 -0
  391. package/src/components/Messages.tsx +834 -0
  392. package/src/components/ModelPicker.tsx +448 -0
  393. package/src/components/NativeAutoUpdater.tsx +193 -0
  394. package/src/components/NotebookEditToolUseRejectedMessage.tsx +92 -0
  395. package/src/components/OffscreenFreeze.tsx +44 -0
  396. package/src/components/Onboarding.tsx +244 -0
  397. package/src/components/OutputStylePicker.tsx +112 -0
  398. package/src/components/PackageManagerAutoUpdater.tsx +104 -0
  399. package/src/components/Passes/Passes.tsx +184 -0
  400. package/src/components/PrBadge.tsx +97 -0
  401. package/src/components/PressEnterToContinue.tsx +15 -0
  402. package/src/components/PromptInput/HistorySearchInput.tsx +51 -0
  403. package/src/components/PromptInput/IssueFlagBanner.tsx +12 -0
  404. package/src/components/PromptInput/Notifications.tsx +332 -0
  405. package/src/components/PromptInput/PromptInput.tsx +2339 -0
  406. package/src/components/PromptInput/PromptInputFooter.tsx +191 -0
  407. package/src/components/PromptInput/PromptInputFooterLeftSide.tsx +517 -0
  408. package/src/components/PromptInput/PromptInputFooterSuggestions.tsx +293 -0
  409. package/src/components/PromptInput/PromptInputHelpMenu.tsx +358 -0
  410. package/src/components/PromptInput/PromptInputModeIndicator.tsx +93 -0
  411. package/src/components/PromptInput/PromptInputQueuedCommands.tsx +117 -0
  412. package/src/components/PromptInput/PromptInputStashNotice.tsx +25 -0
  413. package/src/components/PromptInput/SandboxPromptFooterHint.tsx +64 -0
  414. package/src/components/PromptInput/ShimmeredInput.tsx +143 -0
  415. package/src/components/PromptInput/VoiceIndicator.tsx +137 -0
  416. package/src/components/PromptInput/inputModes.ts +33 -0
  417. package/src/components/PromptInput/inputPaste.ts +90 -0
  418. package/src/components/PromptInput/useMaybeTruncateInput.ts +58 -0
  419. package/src/components/PromptInput/usePromptInputPlaceholder.ts +76 -0
  420. package/src/components/PromptInput/useShowFastIconHint.ts +31 -0
  421. package/src/components/PromptInput/useSwarmBanner.ts +155 -0
  422. package/src/components/PromptInput/utils.ts +60 -0
  423. package/src/components/QuickOpenDialog.tsx +244 -0
  424. package/src/components/RemoteCallout.tsx +76 -0
  425. package/src/components/RemoteEnvironmentDialog.tsx +340 -0
  426. package/src/components/ResumeTask.tsx +268 -0
  427. package/src/components/SandboxViolationExpandedView.tsx +99 -0
  428. package/src/components/ScrollKeybindingHandler.tsx +1012 -0
  429. package/src/components/SearchBox.tsx +72 -0
  430. package/src/components/SentryErrorBoundary.ts +28 -0
  431. package/src/components/SessionBackgroundHint.tsx +108 -0
  432. package/src/components/SessionPreview.tsx +194 -0
  433. package/src/components/Settings/Config.tsx +1822 -0
  434. package/src/components/Settings/Settings.tsx +137 -0
  435. package/src/components/Settings/Status.tsx +241 -0
  436. package/src/components/Settings/Usage.tsx +377 -0
  437. package/src/components/ShowInIDEPrompt.tsx +170 -0
  438. package/src/components/SkillImprovementSurvey.tsx +152 -0
  439. package/src/components/Spinner/FlashingChar.tsx +61 -0
  440. package/src/components/Spinner/GlimmerMessage.tsx +328 -0
  441. package/src/components/Spinner/ShimmerChar.tsx +36 -0
  442. package/src/components/Spinner/SpinnerAnimationRow.tsx +265 -0
  443. package/src/components/Spinner/SpinnerGlyph.tsx +80 -0
  444. package/src/components/Spinner/TeammateSpinnerLine.tsx +233 -0
  445. package/src/components/Spinner/TeammateSpinnerTree.tsx +272 -0
  446. package/src/components/Spinner/index.ts +10 -0
  447. package/src/components/Spinner/teammateSelectHint.ts +1 -0
  448. package/src/components/Spinner/useShimmerAnimation.ts +31 -0
  449. package/src/components/Spinner/useStalledAnimation.ts +75 -0
  450. package/src/components/Spinner/utils.ts +84 -0
  451. package/src/components/Spinner.tsx +562 -0
  452. package/src/components/Stats.tsx +1228 -0
  453. package/src/components/StatusLine.tsx +324 -0
  454. package/src/components/StatusNotices.tsx +55 -0
  455. package/src/components/StructuredDiff/Fallback.tsx +487 -0
  456. package/src/components/StructuredDiff/colorDiff.ts +37 -0
  457. package/src/components/StructuredDiff.tsx +190 -0
  458. package/src/components/StructuredDiffList.tsx +30 -0
  459. package/src/components/TagTabs.tsx +139 -0
  460. package/src/components/TaskListV2.tsx +378 -0
  461. package/src/components/TeammateViewHeader.tsx +82 -0
  462. package/src/components/TeleportError.tsx +189 -0
  463. package/src/components/TeleportProgress.tsx +140 -0
  464. package/src/components/TeleportRepoMismatchDialog.tsx +104 -0
  465. package/src/components/TeleportResumeWrapper.tsx +167 -0
  466. package/src/components/TeleportStash.tsx +116 -0
  467. package/src/components/TextInput.tsx +124 -0
  468. package/src/components/ThemePicker.tsx +333 -0
  469. package/src/components/ThinkingToggle.tsx +153 -0
  470. package/src/components/TokenWarning.tsx +179 -0
  471. package/src/components/ToolUseLoader.tsx +42 -0
  472. package/src/components/TrustDialog/TrustDialog.tsx +290 -0
  473. package/src/components/TrustDialog/utils.ts +245 -0
  474. package/src/components/ValidationErrorsList.tsx +148 -0
  475. package/src/components/VimTextInput.tsx +140 -0
  476. package/src/components/VirtualMessageList.tsx +1082 -0
  477. package/src/components/WorkflowMultiselectDialog.tsx +128 -0
  478. package/src/components/WorktreeExitDialog.tsx +231 -0
  479. package/src/components/agents/AgentDetail.tsx +220 -0
  480. package/src/components/agents/AgentEditor.tsx +178 -0
  481. package/src/components/agents/AgentNavigationFooter.tsx +26 -0
  482. package/src/components/agents/AgentsList.tsx +440 -0
  483. package/src/components/agents/AgentsMenu.tsx +800 -0
  484. package/src/components/agents/ColorPicker.tsx +112 -0
  485. package/src/components/agents/ModelSelector.tsx +68 -0
  486. package/src/components/agents/ToolSelector.tsx +562 -0
  487. package/src/components/agents/agentFileUtils.ts +272 -0
  488. package/src/components/agents/generateAgent.ts +197 -0
  489. package/src/components/agents/new-agent-creation/CreateAgentWizard.tsx +97 -0
  490. package/src/components/agents/new-agent-creation/wizard-steps/ColorStep.tsx +84 -0
  491. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStep.tsx +378 -0
  492. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStepWrapper.tsx +74 -0
  493. package/src/components/agents/new-agent-creation/wizard-steps/DescriptionStep.tsx +123 -0
  494. package/src/components/agents/new-agent-creation/wizard-steps/GenerateStep.tsx +143 -0
  495. package/src/components/agents/new-agent-creation/wizard-steps/LocationStep.tsx +80 -0
  496. package/src/components/agents/new-agent-creation/wizard-steps/MemoryStep.tsx +113 -0
  497. package/src/components/agents/new-agent-creation/wizard-steps/MethodStep.tsx +80 -0
  498. package/src/components/agents/new-agent-creation/wizard-steps/ModelStep.tsx +52 -0
  499. package/src/components/agents/new-agent-creation/wizard-steps/PromptStep.tsx +128 -0
  500. package/src/components/agents/new-agent-creation/wizard-steps/ToolsStep.tsx +61 -0
  501. package/src/components/agents/new-agent-creation/wizard-steps/TypeStep.tsx +103 -0
  502. package/src/components/agents/types.ts +27 -0
  503. package/src/components/agents/utils.ts +18 -0
  504. package/src/components/agents/validateAgent.ts +109 -0
  505. package/src/components/design-system/Byline.tsx +77 -0
  506. package/src/components/design-system/Dialog.tsx +138 -0
  507. package/src/components/design-system/Divider.tsx +149 -0
  508. package/src/components/design-system/FuzzyPicker.tsx +312 -0
  509. package/src/components/design-system/KeyboardShortcutHint.tsx +81 -0
  510. package/src/components/design-system/ListItem.tsx +244 -0
  511. package/src/components/design-system/LoadingState.tsx +94 -0
  512. package/src/components/design-system/Pane.tsx +77 -0
  513. package/src/components/design-system/ProgressBar.tsx +86 -0
  514. package/src/components/design-system/Ratchet.tsx +80 -0
  515. package/src/components/design-system/StatusIcon.tsx +95 -0
  516. package/src/components/design-system/Tabs.tsx +340 -0
  517. package/src/components/design-system/ThemeProvider.tsx +174 -0
  518. package/src/components/design-system/ThemedBox.tsx +156 -0
  519. package/src/components/design-system/ThemedText.tsx +124 -0
  520. package/src/components/design-system/color.ts +30 -0
  521. package/src/components/diff/DiffDetailView.tsx +281 -0
  522. package/src/components/diff/DiffDialog.tsx +383 -0
  523. package/src/components/diff/DiffFileList.tsx +292 -0
  524. package/src/components/grove/Grove.tsx +463 -0
  525. package/src/components/hooks/HooksConfigMenu.tsx +578 -0
  526. package/src/components/hooks/PromptDialog.tsx +90 -0
  527. package/src/components/hooks/SelectEventMode.tsx +127 -0
  528. package/src/components/hooks/SelectHookMode.tsx +112 -0
  529. package/src/components/hooks/SelectMatcherMode.tsx +144 -0
  530. package/src/components/hooks/ViewHookMode.tsx +199 -0
  531. package/src/components/mcp/CapabilitiesSection.tsx +61 -0
  532. package/src/components/mcp/ElicitationDialog.tsx +1169 -0
  533. package/src/components/mcp/MCPAgentServerMenu.tsx +183 -0
  534. package/src/components/mcp/MCPListPanel.tsx +504 -0
  535. package/src/components/mcp/MCPReconnect.tsx +167 -0
  536. package/src/components/mcp/MCPRemoteServerMenu.tsx +649 -0
  537. package/src/components/mcp/MCPSettings.tsx +398 -0
  538. package/src/components/mcp/MCPStdioServerMenu.tsx +177 -0
  539. package/src/components/mcp/MCPToolDetailView.tsx +212 -0
  540. package/src/components/mcp/MCPToolListView.tsx +141 -0
  541. package/src/components/mcp/McpParsingWarnings.tsx +213 -0
  542. package/src/components/mcp/index.ts +9 -0
  543. package/src/components/mcp/utils/reconnectHelpers.tsx +49 -0
  544. package/src/components/memory/MemoryFileSelector.tsx +438 -0
  545. package/src/components/memory/MemoryUpdateNotification.tsx +45 -0
  546. package/src/components/messageActions.tsx +450 -0
  547. package/src/components/messages/AdvisorMessage.tsx +158 -0
  548. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +31 -0
  549. package/src/components/messages/AssistantTextMessage.tsx +270 -0
  550. package/src/components/messages/AssistantThinkingMessage.tsx +86 -0
  551. package/src/components/messages/AssistantToolUseMessage.tsx +368 -0
  552. package/src/components/messages/AttachmentMessage.tsx +536 -0
  553. package/src/components/messages/CollapsedReadSearchContent.tsx +484 -0
  554. package/src/components/messages/CompactBoundaryMessage.tsx +18 -0
  555. package/src/components/messages/GroupedToolUseContent.tsx +58 -0
  556. package/src/components/messages/HighlightedThinkingText.tsx +162 -0
  557. package/src/components/messages/HookProgressMessage.tsx +116 -0
  558. package/src/components/messages/PlanApprovalMessage.tsx +222 -0
  559. package/src/components/messages/RateLimitMessage.tsx +161 -0
  560. package/src/components/messages/ShutdownMessage.tsx +132 -0
  561. package/src/components/messages/SystemAPIErrorMessage.tsx +141 -0
  562. package/src/components/messages/SystemTextMessage.tsx +827 -0
  563. package/src/components/messages/TaskAssignmentMessage.tsx +76 -0
  564. package/src/components/messages/UserAgentNotificationMessage.tsx +83 -0
  565. package/src/components/messages/UserBashInputMessage.tsx +58 -0
  566. package/src/components/messages/UserBashOutputMessage.tsx +54 -0
  567. package/src/components/messages/UserChannelMessage.tsx +137 -0
  568. package/src/components/messages/UserCommandMessage.tsx +108 -0
  569. package/src/components/messages/UserImageMessage.tsx +59 -0
  570. package/src/components/messages/UserLocalCommandOutputMessage.tsx +167 -0
  571. package/src/components/messages/UserMemoryInputMessage.tsx +75 -0
  572. package/src/components/messages/UserPlanMessage.tsx +42 -0
  573. package/src/components/messages/UserPromptMessage.tsx +80 -0
  574. package/src/components/messages/UserResourceUpdateMessage.tsx +121 -0
  575. package/src/components/messages/UserTeammateMessage.tsx +206 -0
  576. package/src/components/messages/UserTextMessage.tsx +275 -0
  577. package/src/components/messages/UserToolResultMessage/RejectedPlanMessage.tsx +31 -0
  578. package/src/components/messages/UserToolResultMessage/RejectedToolUseMessage.tsx +16 -0
  579. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +16 -0
  580. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +103 -0
  581. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +95 -0
  582. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +106 -0
  583. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +104 -0
  584. package/src/components/messages/UserToolResultMessage/utils.tsx +44 -0
  585. package/src/components/messages/nullRenderingAttachments.ts +70 -0
  586. package/src/components/messages/teamMemCollapsed.tsx +140 -0
  587. package/src/components/messages/teamMemSaved.ts +19 -0
  588. package/src/components/permissions/AskUserQuestionPermissionRequest/AskUserQuestionPermissionRequest.tsx +645 -0
  589. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewBox.tsx +229 -0
  590. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewQuestionView.tsx +328 -0
  591. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionNavigationBar.tsx +178 -0
  592. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionView.tsx +465 -0
  593. package/src/components/permissions/AskUserQuestionPermissionRequest/SubmitQuestionsView.tsx +144 -0
  594. package/src/components/permissions/AskUserQuestionPermissionRequest/use-multiple-choice-state.ts +179 -0
  595. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +482 -0
  596. package/src/components/permissions/BashPermissionRequest/bashToolUseOptions.tsx +147 -0
  597. package/src/components/permissions/ComputerUseApproval/ComputerUseApproval.tsx +441 -0
  598. package/src/components/permissions/EnterPlanModePermissionRequest/EnterPlanModePermissionRequest.tsx +122 -0
  599. package/src/components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.tsx +768 -0
  600. package/src/components/permissions/FallbackPermissionRequest.tsx +333 -0
  601. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  602. package/src/components/permissions/FilePermissionDialog/FilePermissionDialog.tsx +204 -0
  603. package/src/components/permissions/FilePermissionDialog/ideDiffConfig.ts +42 -0
  604. package/src/components/permissions/FilePermissionDialog/permissionOptions.tsx +177 -0
  605. package/src/components/permissions/FilePermissionDialog/useFilePermissionDialog.ts +212 -0
  606. package/src/components/permissions/FilePermissionDialog/usePermissionHandler.ts +185 -0
  607. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +161 -0
  608. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +89 -0
  609. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +115 -0
  610. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditPermissionRequest.tsx +166 -0
  611. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditToolDiff.tsx +235 -0
  612. package/src/components/permissions/PermissionDecisionDebugInfo.tsx +460 -0
  613. package/src/components/permissions/PermissionDialog.tsx +72 -0
  614. package/src/components/permissions/PermissionExplanation.tsx +272 -0
  615. package/src/components/permissions/PermissionPrompt.tsx +336 -0
  616. package/src/components/permissions/PermissionRequest.tsx +217 -0
  617. package/src/components/permissions/PermissionRequestTitle.tsx +66 -0
  618. package/src/components/permissions/PermissionRuleExplanation.tsx +121 -0
  619. package/src/components/permissions/PowerShellPermissionRequest/PowerShellPermissionRequest.tsx +235 -0
  620. package/src/components/permissions/PowerShellPermissionRequest/powershellToolUseOptions.tsx +91 -0
  621. package/src/components/permissions/SandboxPermissionRequest.tsx +163 -0
  622. package/src/components/permissions/SedEditPermissionRequest/SedEditPermissionRequest.tsx +230 -0
  623. package/src/components/permissions/SkillPermissionRequest/SkillPermissionRequest.tsx +369 -0
  624. package/src/components/permissions/WebFetchPermissionRequest/WebFetchPermissionRequest.tsx +258 -0
  625. package/src/components/permissions/WorkerBadge.tsx +49 -0
  626. package/src/components/permissions/WorkerPendingPermission.tsx +105 -0
  627. package/src/components/permissions/hooks.ts +209 -0
  628. package/src/components/permissions/rules/AddPermissionRules.tsx +180 -0
  629. package/src/components/permissions/rules/AddWorkspaceDirectory.tsx +340 -0
  630. package/src/components/permissions/rules/PermissionRuleDescription.tsx +76 -0
  631. package/src/components/permissions/rules/PermissionRuleInput.tsx +138 -0
  632. package/src/components/permissions/rules/PermissionRuleList.tsx +1179 -0
  633. package/src/components/permissions/rules/RecentDenialsTab.tsx +207 -0
  634. package/src/components/permissions/rules/RemoveWorkspaceDirectory.tsx +110 -0
  635. package/src/components/permissions/rules/WorkspaceTab.tsx +150 -0
  636. package/src/components/permissions/shellPermissionHelpers.tsx +164 -0
  637. package/src/components/permissions/useShellPermissionFeedback.ts +148 -0
  638. package/src/components/permissions/utils.ts +25 -0
  639. package/src/components/sandbox/SandboxConfigTab.tsx +45 -0
  640. package/src/components/sandbox/SandboxDependenciesTab.tsx +120 -0
  641. package/src/components/sandbox/SandboxDoctorSection.tsx +46 -0
  642. package/src/components/sandbox/SandboxOverridesTab.tsx +193 -0
  643. package/src/components/sandbox/SandboxSettings.tsx +296 -0
  644. package/src/components/shell/ExpandShellOutputContext.tsx +36 -0
  645. package/src/components/shell/OutputLine.tsx +118 -0
  646. package/src/components/shell/ShellProgressMessage.tsx +150 -0
  647. package/src/components/shell/ShellTimeDisplay.tsx +74 -0
  648. package/src/components/skills/SkillsMenu.tsx +237 -0
  649. package/src/components/tasks/AsyncAgentDetailDialog.tsx +229 -0
  650. package/src/components/tasks/BackgroundTask.tsx +345 -0
  651. package/src/components/tasks/BackgroundTaskStatus.tsx +429 -0
  652. package/src/components/tasks/BackgroundTasksDialog.tsx +652 -0
  653. package/src/components/tasks/DreamDetailDialog.tsx +251 -0
  654. package/src/components/tasks/InProcessTeammateDetailDialog.tsx +266 -0
  655. package/src/components/tasks/RemoteSessionDetailDialog.tsx +904 -0
  656. package/src/components/tasks/RemoteSessionProgress.tsx +243 -0
  657. package/src/components/tasks/ShellDetailDialog.tsx +404 -0
  658. package/src/components/tasks/ShellProgress.tsx +87 -0
  659. package/src/components/tasks/renderToolActivity.tsx +33 -0
  660. package/src/components/tasks/taskStatusUtils.tsx +107 -0
  661. package/src/components/teams/TeamStatus.tsx +80 -0
  662. package/src/components/teams/TeamsDialog.tsx +715 -0
  663. package/src/components/ui/OrderedList.tsx +71 -0
  664. package/src/components/ui/OrderedListItem.tsx +45 -0
  665. package/src/components/ui/TreeSelect.tsx +397 -0
  666. package/src/components/wizard/WizardDialogLayout.tsx +65 -0
  667. package/src/components/wizard/WizardNavigationFooter.tsx +24 -0
  668. package/src/components/wizard/WizardProvider.tsx +213 -0
  669. package/src/components/wizard/index.ts +9 -0
  670. package/src/components/wizard/useWizard.ts +13 -0
  671. package/src/constants/apiLimits.ts +94 -0
  672. package/src/constants/betas.ts +52 -0
  673. package/src/constants/common.ts +33 -0
  674. package/src/constants/cyberRiskInstruction.ts +24 -0
  675. package/src/constants/errorIds.ts +15 -0
  676. package/src/constants/figures.ts +45 -0
  677. package/src/constants/files.ts +156 -0
  678. package/src/constants/github-app.ts +144 -0
  679. package/src/constants/keys.ts +11 -0
  680. package/src/constants/messages.ts +1 -0
  681. package/src/constants/oauth.ts +234 -0
  682. package/src/constants/outputStyles.ts +216 -0
  683. package/src/constants/product.ts +76 -0
  684. package/src/constants/prompts.ts +914 -0
  685. package/src/constants/spinnerVerbs.ts +204 -0
  686. package/src/constants/system.ts +95 -0
  687. package/src/constants/systemPromptSections.ts +68 -0
  688. package/src/constants/toolLimits.ts +56 -0
  689. package/src/constants/tools.ts +112 -0
  690. package/src/constants/turnCompletionVerbs.ts +12 -0
  691. package/src/constants/xml.ts +86 -0
  692. package/src/context/QueuedMessageContext.tsx +63 -0
  693. package/src/context/fpsMetrics.tsx +30 -0
  694. package/src/context/mailbox.tsx +38 -0
  695. package/src/context/modalContext.tsx +58 -0
  696. package/src/context/notifications.tsx +240 -0
  697. package/src/context/overlayContext.tsx +151 -0
  698. package/src/context/promptOverlayContext.tsx +125 -0
  699. package/src/context/stats.tsx +220 -0
  700. package/src/context/voice.tsx +88 -0
  701. package/src/context.ts +189 -0
  702. package/src/coordinator/coordinatorMode.ts +369 -0
  703. package/src/cost-tracker.ts +323 -0
  704. package/src/costHook.ts +22 -0
  705. package/src/dialogLaunchers.tsx +133 -0
  706. package/src/entrypoints/agentSdkTypes.ts +443 -0
  707. package/src/entrypoints/cli.tsx +303 -0
  708. package/src/entrypoints/init.ts +340 -0
  709. package/src/entrypoints/mcp.ts +196 -0
  710. package/src/entrypoints/sandboxTypes.ts +156 -0
  711. package/src/entrypoints/sdk/controlSchemas.ts +663 -0
  712. package/src/entrypoints/sdk/coreSchemas.ts +1889 -0
  713. package/src/entrypoints/sdk/coreTypes.ts +62 -0
  714. package/src/history.ts +464 -0
  715. package/src/hooks/fileSuggestions.ts +811 -0
  716. package/src/hooks/notifs/useAutoModeUnavailableNotification.ts +56 -0
  717. package/src/hooks/notifs/useCanSwitchToExistingSubscription.tsx +60 -0
  718. package/src/hooks/notifs/useDeprecationWarningNotification.tsx +44 -0
  719. package/src/hooks/notifs/useFastModeNotification.tsx +162 -0
  720. package/src/hooks/notifs/useIDEStatusIndicator.tsx +186 -0
  721. package/src/hooks/notifs/useInstallMessages.tsx +26 -0
  722. package/src/hooks/notifs/useLspInitializationNotification.tsx +143 -0
  723. package/src/hooks/notifs/useMcpConnectivityStatus.tsx +88 -0
  724. package/src/hooks/notifs/useModelMigrationNotifications.tsx +52 -0
  725. package/src/hooks/notifs/useNpmDeprecationNotification.tsx +25 -0
  726. package/src/hooks/notifs/usePluginAutoupdateNotification.tsx +83 -0
  727. package/src/hooks/notifs/usePluginInstallationStatus.tsx +128 -0
  728. package/src/hooks/notifs/useRateLimitWarningNotification.tsx +114 -0
  729. package/src/hooks/notifs/useSettingsErrors.tsx +69 -0
  730. package/src/hooks/notifs/useStartupNotification.ts +41 -0
  731. package/src/hooks/notifs/useTeammateShutdownNotification.ts +78 -0
  732. package/src/hooks/renderPlaceholder.ts +51 -0
  733. package/src/hooks/toolPermission/PermissionContext.ts +388 -0
  734. package/src/hooks/toolPermission/handlers/coordinatorHandler.ts +65 -0
  735. package/src/hooks/toolPermission/handlers/interactiveHandler.ts +536 -0
  736. package/src/hooks/toolPermission/handlers/swarmWorkerHandler.ts +159 -0
  737. package/src/hooks/toolPermission/permissionLogging.ts +238 -0
  738. package/src/hooks/unifiedSuggestions.ts +202 -0
  739. package/src/hooks/useAfterFirstRender.ts +17 -0
  740. package/src/hooks/useApiKeyVerification.ts +84 -0
  741. package/src/hooks/useArrowKeyHistory.tsx +229 -0
  742. package/src/hooks/useAssistantHistory.ts +250 -0
  743. package/src/hooks/useAwaySummary.ts +125 -0
  744. package/src/hooks/useBackgroundTaskNavigation.ts +251 -0
  745. package/src/hooks/useBlink.ts +34 -0
  746. package/src/hooks/useCanUseTool.tsx +204 -0
  747. package/src/hooks/useCancelRequest.ts +276 -0
  748. package/src/hooks/useChromeExtensionNotification.tsx +50 -0
  749. package/src/hooks/useClaudeCodeHintRecommendation.tsx +129 -0
  750. package/src/hooks/useClipboardImageHint.ts +77 -0
  751. package/src/hooks/useCommandKeybindings.tsx +108 -0
  752. package/src/hooks/useCommandQueue.ts +15 -0
  753. package/src/hooks/useCopyOnSelect.ts +98 -0
  754. package/src/hooks/useDeferredHookMessages.ts +46 -0
  755. package/src/hooks/useDiffData.ts +110 -0
  756. package/src/hooks/useDiffInIDE.ts +379 -0
  757. package/src/hooks/useDirectConnect.ts +229 -0
  758. package/src/hooks/useDoublePress.ts +62 -0
  759. package/src/hooks/useDynamicConfig.ts +22 -0
  760. package/src/hooks/useElapsedTime.ts +37 -0
  761. package/src/hooks/useExitOnCtrlCD.ts +95 -0
  762. package/src/hooks/useExitOnCtrlCDWithKeybindings.ts +24 -0
  763. package/src/hooks/useFileHistorySnapshotInit.ts +25 -0
  764. package/src/hooks/useGlobalKeybindings.tsx +249 -0
  765. package/src/hooks/useHistorySearch.ts +303 -0
  766. package/src/hooks/useIDEIntegration.tsx +70 -0
  767. package/src/hooks/useIdeAtMentioned.ts +76 -0
  768. package/src/hooks/useIdeConnectionStatus.ts +33 -0
  769. package/src/hooks/useIdeLogging.ts +41 -0
  770. package/src/hooks/useIdeSelection.ts +150 -0
  771. package/src/hooks/useInboxPoller.ts +969 -0
  772. package/src/hooks/useInputBuffer.ts +132 -0
  773. package/src/hooks/useIssueFlagBanner.ts +133 -0
  774. package/src/hooks/useLogMessages.ts +119 -0
  775. package/src/hooks/useLspPluginRecommendation.tsx +194 -0
  776. package/src/hooks/useMailboxBridge.ts +21 -0
  777. package/src/hooks/useMainLoopModel.ts +34 -0
  778. package/src/hooks/useManagePlugins.ts +304 -0
  779. package/src/hooks/useMemoryUsage.ts +39 -0
  780. package/src/hooks/useMergedClients.ts +23 -0
  781. package/src/hooks/useMergedCommands.ts +15 -0
  782. package/src/hooks/useMergedTools.ts +44 -0
  783. package/src/hooks/useMinDisplayTime.ts +35 -0
  784. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  785. package/src/hooks/useOfficialMarketplaceNotification.tsx +48 -0
  786. package/src/hooks/usePasteHandler.ts +285 -0
  787. package/src/hooks/usePluginRecommendationBase.tsx +105 -0
  788. package/src/hooks/usePrStatus.ts +106 -0
  789. package/src/hooks/usePromptSuggestion.ts +177 -0
  790. package/src/hooks/usePromptsFromClaudeInChrome.tsx +71 -0
  791. package/src/hooks/useQueueProcessor.ts +68 -0
  792. package/src/hooks/useRemoteSession.ts +605 -0
  793. package/src/hooks/useReplBridge.tsx +723 -0
  794. package/src/hooks/useSSHSession.ts +241 -0
  795. package/src/hooks/useScheduledTasks.ts +139 -0
  796. package/src/hooks/useSearchInput.ts +364 -0
  797. package/src/hooks/useSessionBackgrounding.ts +158 -0
  798. package/src/hooks/useSettings.ts +17 -0
  799. package/src/hooks/useSettingsChange.ts +25 -0
  800. package/src/hooks/useSkillImprovementSurvey.ts +105 -0
  801. package/src/hooks/useSkillsChange.ts +62 -0
  802. package/src/hooks/useSwarmInitialization.ts +81 -0
  803. package/src/hooks/useSwarmPermissionPoller.ts +330 -0
  804. package/src/hooks/useTaskListWatcher.ts +221 -0
  805. package/src/hooks/useTasksV2.ts +250 -0
  806. package/src/hooks/useTeammateViewAutoExit.ts +63 -0
  807. package/src/hooks/useTeleportResume.tsx +85 -0
  808. package/src/hooks/useTerminalSize.ts +15 -0
  809. package/src/hooks/useTextInput.ts +529 -0
  810. package/src/hooks/useTimeout.ts +14 -0
  811. package/src/hooks/useTurnDiffs.ts +213 -0
  812. package/src/hooks/useTypeahead.tsx +1385 -0
  813. package/src/hooks/useUpdateNotification.ts +34 -0
  814. package/src/hooks/useVimInput.ts +316 -0
  815. package/src/hooks/useVirtualScroll.ts +721 -0
  816. package/src/hooks/useVoice.ts +1144 -0
  817. package/src/hooks/useVoiceEnabled.ts +25 -0
  818. package/src/hooks/useVoiceIntegration.tsx +677 -0
  819. package/src/ink/Ansi.tsx +292 -0
  820. package/src/ink/bidi.ts +139 -0
  821. package/src/ink/clearTerminal.ts +74 -0
  822. package/src/ink/colorize.ts +231 -0
  823. package/src/ink/components/AlternateScreen.tsx +80 -0
  824. package/src/ink/components/App.tsx +659 -0
  825. package/src/ink/components/AppContext.ts +21 -0
  826. package/src/ink/components/Box.tsx +214 -0
  827. package/src/ink/components/Button.tsx +192 -0
  828. package/src/ink/components/ClockContext.tsx +112 -0
  829. package/src/ink/components/CursorDeclarationContext.ts +32 -0
  830. package/src/ink/components/ErrorOverview.tsx +109 -0
  831. package/src/ink/components/Link.tsx +42 -0
  832. package/src/ink/components/Newline.tsx +39 -0
  833. package/src/ink/components/NoSelect.tsx +68 -0
  834. package/src/ink/components/RawAnsi.tsx +57 -0
  835. package/src/ink/components/ScrollBox.tsx +237 -0
  836. package/src/ink/components/Spacer.tsx +20 -0
  837. package/src/ink/components/StdinContext.ts +49 -0
  838. package/src/ink/components/TerminalFocusContext.tsx +52 -0
  839. package/src/ink/components/TerminalSizeContext.tsx +7 -0
  840. package/src/ink/components/Text.tsx +254 -0
  841. package/src/ink/constants.ts +2 -0
  842. package/src/ink/dom.ts +484 -0
  843. package/src/ink/events/click-event.ts +38 -0
  844. package/src/ink/events/dispatcher.ts +233 -0
  845. package/src/ink/events/emitter.ts +39 -0
  846. package/src/ink/events/event-handlers.ts +73 -0
  847. package/src/ink/events/event.ts +11 -0
  848. package/src/ink/events/focus-event.ts +21 -0
  849. package/src/ink/events/input-event.ts +205 -0
  850. package/src/ink/events/keyboard-event.ts +51 -0
  851. package/src/ink/events/terminal-event.ts +107 -0
  852. package/src/ink/events/terminal-focus-event.ts +19 -0
  853. package/src/ink/focus.ts +181 -0
  854. package/src/ink/frame.ts +124 -0
  855. package/src/ink/get-max-width.ts +27 -0
  856. package/src/ink/hit-test.ts +130 -0
  857. package/src/ink/hooks/use-animation-frame.ts +57 -0
  858. package/src/ink/hooks/use-app.ts +8 -0
  859. package/src/ink/hooks/use-declared-cursor.ts +73 -0
  860. package/src/ink/hooks/use-input.ts +92 -0
  861. package/src/ink/hooks/use-interval.ts +67 -0
  862. package/src/ink/hooks/use-search-highlight.ts +53 -0
  863. package/src/ink/hooks/use-selection.ts +104 -0
  864. package/src/ink/hooks/use-stdin.ts +8 -0
  865. package/src/ink/hooks/use-tab-status.ts +72 -0
  866. package/src/ink/hooks/use-terminal-focus.ts +16 -0
  867. package/src/ink/hooks/use-terminal-title.ts +31 -0
  868. package/src/ink/hooks/use-terminal-viewport.ts +96 -0
  869. package/src/ink/ink.tsx +1728 -0
  870. package/src/ink/instances.ts +10 -0
  871. package/src/ink/layout/engine.ts +6 -0
  872. package/src/ink/layout/geometry.ts +97 -0
  873. package/src/ink/layout/node.ts +152 -0
  874. package/src/ink/layout/yoga.ts +308 -0
  875. package/src/ink/line-width-cache.ts +24 -0
  876. package/src/ink/log-update.ts +773 -0
  877. package/src/ink/measure-element.ts +23 -0
  878. package/src/ink/measure-text.ts +47 -0
  879. package/src/ink/node-cache.ts +54 -0
  880. package/src/ink/optimizer.ts +93 -0
  881. package/src/ink/output.ts +797 -0
  882. package/src/ink/parse-keypress.ts +801 -0
  883. package/src/ink/reconciler.ts +512 -0
  884. package/src/ink/render-border.ts +231 -0
  885. package/src/ink/render-node-to-output.ts +1462 -0
  886. package/src/ink/render-to-screen.ts +231 -0
  887. package/src/ink/renderer.ts +178 -0
  888. package/src/ink/root.ts +184 -0
  889. package/src/ink/screen.ts +1486 -0
  890. package/src/ink/searchHighlight.ts +93 -0
  891. package/src/ink/selection.ts +917 -0
  892. package/src/ink/squash-text-nodes.ts +92 -0
  893. package/src/ink/stringWidth.ts +222 -0
  894. package/src/ink/styles.ts +771 -0
  895. package/src/ink/supports-hyperlinks.ts +57 -0
  896. package/src/ink/tabstops.ts +46 -0
  897. package/src/ink/terminal-focus-state.ts +47 -0
  898. package/src/ink/terminal-querier.ts +212 -0
  899. package/src/ink/terminal.ts +248 -0
  900. package/src/ink/termio/ansi.ts +75 -0
  901. package/src/ink/termio/csi.ts +319 -0
  902. package/src/ink/termio/dec.ts +60 -0
  903. package/src/ink/termio/esc.ts +67 -0
  904. package/src/ink/termio/osc.ts +493 -0
  905. package/src/ink/termio/parser.ts +394 -0
  906. package/src/ink/termio/sgr.ts +308 -0
  907. package/src/ink/termio/tokenize.ts +319 -0
  908. package/src/ink/termio/types.ts +236 -0
  909. package/src/ink/termio.ts +42 -0
  910. package/src/ink/useTerminalNotification.ts +126 -0
  911. package/src/ink/warn.ts +9 -0
  912. package/src/ink/widest-line.ts +19 -0
  913. package/src/ink/wrap-text.ts +74 -0
  914. package/src/ink/wrapAnsi.ts +20 -0
  915. package/src/ink.ts +85 -0
  916. package/src/interactiveHelpers.tsx +367 -0
  917. package/src/keybindings/KeybindingContext.tsx +243 -0
  918. package/src/keybindings/KeybindingProviderSetup.tsx +308 -0
  919. package/src/keybindings/defaultBindings.ts +340 -0
  920. package/src/keybindings/loadUserBindings.ts +472 -0
  921. package/src/keybindings/match.ts +120 -0
  922. package/src/keybindings/parser.ts +203 -0
  923. package/src/keybindings/reservedShortcuts.ts +127 -0
  924. package/src/keybindings/resolver.ts +244 -0
  925. package/src/keybindings/schema.ts +236 -0
  926. package/src/keybindings/shortcutFormat.ts +63 -0
  927. package/src/keybindings/template.ts +52 -0
  928. package/src/keybindings/useKeybinding.ts +196 -0
  929. package/src/keybindings/useShortcutDisplay.ts +59 -0
  930. package/src/keybindings/validate.ts +498 -0
  931. package/src/main.tsx +4684 -0
  932. package/src/memdir/findRelevantMemories.ts +141 -0
  933. package/src/memdir/memdir.ts +507 -0
  934. package/src/memdir/memoryAge.ts +53 -0
  935. package/src/memdir/memoryScan.ts +94 -0
  936. package/src/memdir/memoryTypes.ts +271 -0
  937. package/src/memdir/paths.ts +278 -0
  938. package/src/memdir/teamMemPaths.ts +292 -0
  939. package/src/memdir/teamMemPrompts.ts +100 -0
  940. package/src/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  941. package/src/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  942. package/src/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  943. package/src/migrations/migrateFennecToOpus.ts +45 -0
  944. package/src/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  945. package/src/migrations/migrateOpusToOpus1m.ts +43 -0
  946. package/src/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  947. package/src/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  948. package/src/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  949. package/src/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  950. package/src/migrations/resetProToOpusDefault.ts +51 -0
  951. package/src/moreright/useMoreRight.tsx +26 -0
  952. package/src/native-ts/color-diff/index.ts +999 -0
  953. package/src/native-ts/file-index/index.ts +370 -0
  954. package/src/native-ts/yoga-layout/enums.ts +134 -0
  955. package/src/native-ts/yoga-layout/index.ts +2578 -0
  956. package/src/outputStyles/loadOutputStylesDir.ts +98 -0
  957. package/src/plugins/builtinPlugins.ts +159 -0
  958. package/src/plugins/bundled/index.ts +23 -0
  959. package/src/projectOnboardingState.ts +83 -0
  960. package/src/query/config.ts +46 -0
  961. package/src/query/deps.ts +40 -0
  962. package/src/query/stopHooks.ts +473 -0
  963. package/src/query/tokenBudget.ts +93 -0
  964. package/src/query.ts +1729 -0
  965. package/src/remote/RemoteSessionManager.ts +343 -0
  966. package/src/remote/SessionsWebSocket.ts +404 -0
  967. package/src/remote/remotePermissionBridge.ts +78 -0
  968. package/src/remote/sdkMessageAdapter.ts +302 -0
  969. package/src/replLauncher.tsx +23 -0
  970. package/src/schemas/hooks.ts +222 -0
  971. package/src/screens/Doctor.tsx +575 -0
  972. package/src/screens/REPL.tsx +5006 -0
  973. package/src/screens/ResumeConversation.tsx +399 -0
  974. package/src/server/createDirectConnectSession.ts +88 -0
  975. package/src/server/directConnectManager.ts +213 -0
  976. package/src/server/types.ts +57 -0
  977. package/src/services/AgentSummary/agentSummary.ts +179 -0
  978. package/src/services/MagicDocs/magicDocs.ts +254 -0
  979. package/src/services/MagicDocs/prompts.ts +127 -0
  980. package/src/services/PromptSuggestion/promptSuggestion.ts +523 -0
  981. package/src/services/PromptSuggestion/speculation.ts +991 -0
  982. package/src/services/SessionMemory/prompts.ts +324 -0
  983. package/src/services/SessionMemory/sessionMemory.ts +495 -0
  984. package/src/services/SessionMemory/sessionMemoryUtils.ts +207 -0
  985. package/src/services/analytics/config.ts +38 -0
  986. package/src/services/analytics/datadog.ts +307 -0
  987. package/src/services/analytics/firstPartyEventLogger.ts +449 -0
  988. package/src/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
  989. package/src/services/analytics/growthbook.ts +1157 -0
  990. package/src/services/analytics/index.ts +173 -0
  991. package/src/services/analytics/metadata.ts +973 -0
  992. package/src/services/analytics/sink.ts +114 -0
  993. package/src/services/analytics/sinkKillswitch.ts +25 -0
  994. package/src/services/api/adminRequests.ts +119 -0
  995. package/src/services/api/bootstrap.ts +142 -0
  996. package/src/services/api/claude.ts +3433 -0
  997. package/src/services/api/client.ts +395 -0
  998. package/src/services/api/dumpPrompts.ts +226 -0
  999. package/src/services/api/emptyUsage.ts +22 -0
  1000. package/src/services/api/errorUtils.ts +260 -0
  1001. package/src/services/api/errors.ts +1207 -0
  1002. package/src/services/api/filesApi.ts +748 -0
  1003. package/src/services/api/firstTokenDate.ts +60 -0
  1004. package/src/services/api/grove.ts +357 -0
  1005. package/src/services/api/logging.ts +788 -0
  1006. package/src/services/api/metricsOptOut.ts +159 -0
  1007. package/src/services/api/overageCreditGrant.ts +137 -0
  1008. package/src/services/api/promptCacheBreakDetection.ts +727 -0
  1009. package/src/services/api/referral.ts +281 -0
  1010. package/src/services/api/sessionIngress.ts +514 -0
  1011. package/src/services/api/ultrareviewQuota.ts +38 -0
  1012. package/src/services/api/usage.ts +63 -0
  1013. package/src/services/api/withRetry.ts +826 -0
  1014. package/src/services/autoDream/autoDream.ts +324 -0
  1015. package/src/services/autoDream/config.ts +21 -0
  1016. package/src/services/autoDream/consolidationLock.ts +140 -0
  1017. package/src/services/autoDream/consolidationPrompt.ts +65 -0
  1018. package/src/services/awaySummary.ts +74 -0
  1019. package/src/services/claudeAiLimits.ts +515 -0
  1020. package/src/services/claudeAiLimitsHook.ts +23 -0
  1021. package/src/services/compact/apiMicrocompact.ts +153 -0
  1022. package/src/services/compact/autoCompact.ts +351 -0
  1023. package/src/services/compact/compact.ts +1705 -0
  1024. package/src/services/compact/compactWarningHook.ts +16 -0
  1025. package/src/services/compact/compactWarningState.ts +18 -0
  1026. package/src/services/compact/grouping.ts +63 -0
  1027. package/src/services/compact/microCompact.ts +530 -0
  1028. package/src/services/compact/postCompactCleanup.ts +77 -0
  1029. package/src/services/compact/prompt.ts +374 -0
  1030. package/src/services/compact/sessionMemoryCompact.ts +630 -0
  1031. package/src/services/compact/timeBasedMCConfig.ts +43 -0
  1032. package/src/services/diagnosticTracking.ts +397 -0
  1033. package/src/services/extractMemories/extractMemories.ts +615 -0
  1034. package/src/services/extractMemories/prompts.ts +154 -0
  1035. package/src/services/internalLogging.ts +90 -0
  1036. package/src/services/lsp/LSPClient.ts +447 -0
  1037. package/src/services/lsp/LSPDiagnosticRegistry.ts +386 -0
  1038. package/src/services/lsp/LSPServerInstance.ts +511 -0
  1039. package/src/services/lsp/LSPServerManager.ts +420 -0
  1040. package/src/services/lsp/config.ts +79 -0
  1041. package/src/services/lsp/manager.ts +289 -0
  1042. package/src/services/lsp/passiveFeedback.ts +328 -0
  1043. package/src/services/mcp/InProcessTransport.ts +63 -0
  1044. package/src/services/mcp/MCPConnectionManager.tsx +73 -0
  1045. package/src/services/mcp/SdkControlTransport.ts +136 -0
  1046. package/src/services/mcp/auth.ts +2465 -0
  1047. package/src/services/mcp/channelAllowlist.ts +76 -0
  1048. package/src/services/mcp/channelNotification.ts +316 -0
  1049. package/src/services/mcp/channelPermissions.ts +240 -0
  1050. package/src/services/mcp/claudeai.ts +164 -0
  1051. package/src/services/mcp/client.ts +3348 -0
  1052. package/src/services/mcp/config.ts +1578 -0
  1053. package/src/services/mcp/elicitationHandler.ts +313 -0
  1054. package/src/services/mcp/envExpansion.ts +38 -0
  1055. package/src/services/mcp/headersHelper.ts +138 -0
  1056. package/src/services/mcp/mcpStringUtils.ts +106 -0
  1057. package/src/services/mcp/normalization.ts +23 -0
  1058. package/src/services/mcp/oauthPort.ts +78 -0
  1059. package/src/services/mcp/officialRegistry.ts +72 -0
  1060. package/src/services/mcp/types.ts +258 -0
  1061. package/src/services/mcp/useManageMCPConnections.ts +1141 -0
  1062. package/src/services/mcp/utils.ts +575 -0
  1063. package/src/services/mcp/vscodeSdkMcp.ts +112 -0
  1064. package/src/services/mcp/xaa.ts +511 -0
  1065. package/src/services/mcp/xaaIdpLogin.ts +487 -0
  1066. package/src/services/mcpServerApproval.tsx +41 -0
  1067. package/src/services/mockRateLimits.ts +882 -0
  1068. package/src/services/notifier.ts +156 -0
  1069. package/src/services/oauth/auth-code-listener.ts +211 -0
  1070. package/src/services/oauth/client.ts +566 -0
  1071. package/src/services/oauth/crypto.ts +23 -0
  1072. package/src/services/oauth/getOauthProfile.ts +53 -0
  1073. package/src/services/oauth/index.ts +198 -0
  1074. package/src/services/plugins/PluginInstallationManager.ts +184 -0
  1075. package/src/services/plugins/pluginCliCommands.ts +344 -0
  1076. package/src/services/plugins/pluginOperations.ts +1088 -0
  1077. package/src/services/policyLimits/index.ts +664 -0
  1078. package/src/services/policyLimits/types.ts +27 -0
  1079. package/src/services/preventSleep.ts +165 -0
  1080. package/src/services/rateLimitMessages.ts +344 -0
  1081. package/src/services/rateLimitMocking.ts +144 -0
  1082. package/src/services/remoteManagedSettings/index.ts +639 -0
  1083. package/src/services/remoteManagedSettings/securityCheck.tsx +74 -0
  1084. package/src/services/remoteManagedSettings/syncCache.ts +112 -0
  1085. package/src/services/remoteManagedSettings/syncCacheState.ts +96 -0
  1086. package/src/services/remoteManagedSettings/types.ts +31 -0
  1087. package/src/services/settingsSync/index.ts +581 -0
  1088. package/src/services/settingsSync/types.ts +67 -0
  1089. package/src/services/teamMemorySync/index.ts +1256 -0
  1090. package/src/services/teamMemorySync/secretScanner.ts +324 -0
  1091. package/src/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
  1092. package/src/services/teamMemorySync/types.ts +156 -0
  1093. package/src/services/teamMemorySync/watcher.ts +387 -0
  1094. package/src/services/tips/tipHistory.ts +17 -0
  1095. package/src/services/tips/tipRegistry.ts +686 -0
  1096. package/src/services/tips/tipScheduler.ts +58 -0
  1097. package/src/services/tokenEstimation.ts +495 -0
  1098. package/src/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
  1099. package/src/services/tools/StreamingToolExecutor.ts +530 -0
  1100. package/src/services/tools/toolExecution.ts +1745 -0
  1101. package/src/services/tools/toolHooks.ts +650 -0
  1102. package/src/services/tools/toolOrchestration.ts +188 -0
  1103. package/src/services/vcr.ts +406 -0
  1104. package/src/services/voice.ts +525 -0
  1105. package/src/services/voiceKeyterms.ts +106 -0
  1106. package/src/services/voiceStreamSTT.ts +544 -0
  1107. package/src/setup.ts +477 -0
  1108. package/src/skills/bundled/batch.ts +124 -0
  1109. package/src/skills/bundled/claudeApi.ts +196 -0
  1110. package/src/skills/bundled/claudeApiContent.ts +75 -0
  1111. package/src/skills/bundled/claudeInChrome.ts +34 -0
  1112. package/src/skills/bundled/debug.ts +103 -0
  1113. package/src/skills/bundled/index.ts +79 -0
  1114. package/src/skills/bundled/keybindings.ts +339 -0
  1115. package/src/skills/bundled/loop.ts +92 -0
  1116. package/src/skills/bundled/loremIpsum.ts +282 -0
  1117. package/src/skills/bundled/remember.ts +82 -0
  1118. package/src/skills/bundled/scheduleRemoteAgents.ts +447 -0
  1119. package/src/skills/bundled/simplify.ts +69 -0
  1120. package/src/skills/bundled/skillify.ts +197 -0
  1121. package/src/skills/bundled/stuck.ts +79 -0
  1122. package/src/skills/bundled/updateConfig.ts +475 -0
  1123. package/src/skills/bundled/verify.ts +30 -0
  1124. package/src/skills/bundled/verifyContent.ts +13 -0
  1125. package/src/skills/bundledSkills.ts +220 -0
  1126. package/src/skills/loadSkillsDir.ts +1086 -0
  1127. package/src/skills/mcpSkillBuilders.ts +44 -0
  1128. package/src/state/AppState.tsx +200 -0
  1129. package/src/state/AppStateStore.ts +569 -0
  1130. package/src/state/onChangeAppState.ts +171 -0
  1131. package/src/state/selectors.ts +76 -0
  1132. package/src/state/store.ts +34 -0
  1133. package/src/state/teammateViewHelpers.ts +141 -0
  1134. package/src/tasks/DreamTask/DreamTask.ts +157 -0
  1135. package/src/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  1136. package/src/tasks/InProcessTeammateTask/types.ts +121 -0
  1137. package/src/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  1138. package/src/tasks/LocalMainSessionTask.ts +479 -0
  1139. package/src/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  1140. package/src/tasks/LocalShellTask/guards.ts +41 -0
  1141. package/src/tasks/LocalShellTask/killShellTasks.ts +76 -0
  1142. package/src/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  1143. package/src/tasks/pillLabel.ts +82 -0
  1144. package/src/tasks/stopTask.ts +100 -0
  1145. package/src/tasks/types.ts +46 -0
  1146. package/src/tasks.ts +39 -0
  1147. package/src/tools/AgentTool/AgentTool.tsx +1398 -0
  1148. package/src/tools/AgentTool/UI.tsx +872 -0
  1149. package/src/tools/AgentTool/agentColorManager.ts +66 -0
  1150. package/src/tools/AgentTool/agentDisplay.ts +104 -0
  1151. package/src/tools/AgentTool/agentMemory.ts +177 -0
  1152. package/src/tools/AgentTool/agentMemorySnapshot.ts +197 -0
  1153. package/src/tools/AgentTool/agentToolUtils.ts +686 -0
  1154. package/src/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
  1155. package/src/tools/AgentTool/built-in/exploreAgent.ts +83 -0
  1156. package/src/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
  1157. package/src/tools/AgentTool/built-in/planAgent.ts +92 -0
  1158. package/src/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
  1159. package/src/tools/AgentTool/built-in/verificationAgent.ts +152 -0
  1160. package/src/tools/AgentTool/builtInAgents.ts +72 -0
  1161. package/src/tools/AgentTool/constants.ts +12 -0
  1162. package/src/tools/AgentTool/forkSubagent.ts +210 -0
  1163. package/src/tools/AgentTool/loadAgentsDir.ts +755 -0
  1164. package/src/tools/AgentTool/prompt.ts +287 -0
  1165. package/src/tools/AgentTool/resumeAgent.ts +265 -0
  1166. package/src/tools/AgentTool/runAgent.ts +973 -0
  1167. package/src/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
  1168. package/src/tools/AskUserQuestionTool/prompt.ts +44 -0
  1169. package/src/tools/BashTool/BashTool.tsx +1144 -0
  1170. package/src/tools/BashTool/BashToolResultMessage.tsx +191 -0
  1171. package/src/tools/BashTool/UI.tsx +185 -0
  1172. package/src/tools/BashTool/bashCommandHelpers.ts +265 -0
  1173. package/src/tools/BashTool/bashPermissions.ts +2621 -0
  1174. package/src/tools/BashTool/bashSecurity.ts +2592 -0
  1175. package/src/tools/BashTool/commandSemantics.ts +140 -0
  1176. package/src/tools/BashTool/commentLabel.ts +13 -0
  1177. package/src/tools/BashTool/destructiveCommandWarning.ts +102 -0
  1178. package/src/tools/BashTool/modeValidation.ts +115 -0
  1179. package/src/tools/BashTool/pathValidation.ts +1303 -0
  1180. package/src/tools/BashTool/prompt.ts +369 -0
  1181. package/src/tools/BashTool/readOnlyValidation.ts +1990 -0
  1182. package/src/tools/BashTool/sedEditParser.ts +322 -0
  1183. package/src/tools/BashTool/sedValidation.ts +684 -0
  1184. package/src/tools/BashTool/shouldUseSandbox.ts +153 -0
  1185. package/src/tools/BashTool/toolName.ts +2 -0
  1186. package/src/tools/BashTool/utils.ts +223 -0
  1187. package/src/tools/BriefTool/BriefTool.ts +204 -0
  1188. package/src/tools/BriefTool/UI.tsx +101 -0
  1189. package/src/tools/BriefTool/attachments.ts +110 -0
  1190. package/src/tools/BriefTool/prompt.ts +22 -0
  1191. package/src/tools/BriefTool/upload.ts +174 -0
  1192. package/src/tools/ConfigTool/ConfigTool.ts +467 -0
  1193. package/src/tools/ConfigTool/UI.tsx +38 -0
  1194. package/src/tools/ConfigTool/constants.ts +1 -0
  1195. package/src/tools/ConfigTool/prompt.ts +93 -0
  1196. package/src/tools/ConfigTool/supportedSettings.ts +211 -0
  1197. package/src/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
  1198. package/src/tools/EnterPlanModeTool/UI.tsx +33 -0
  1199. package/src/tools/EnterPlanModeTool/constants.ts +1 -0
  1200. package/src/tools/EnterPlanModeTool/prompt.ts +170 -0
  1201. package/src/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
  1202. package/src/tools/EnterWorktreeTool/UI.tsx +20 -0
  1203. package/src/tools/EnterWorktreeTool/constants.ts +1 -0
  1204. package/src/tools/EnterWorktreeTool/prompt.ts +30 -0
  1205. package/src/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
  1206. package/src/tools/ExitPlanModeTool/UI.tsx +82 -0
  1207. package/src/tools/ExitPlanModeTool/constants.ts +2 -0
  1208. package/src/tools/ExitPlanModeTool/prompt.ts +29 -0
  1209. package/src/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
  1210. package/src/tools/ExitWorktreeTool/UI.tsx +25 -0
  1211. package/src/tools/ExitWorktreeTool/constants.ts +1 -0
  1212. package/src/tools/ExitWorktreeTool/prompt.ts +32 -0
  1213. package/src/tools/FileEditTool/FileEditTool.ts +625 -0
  1214. package/src/tools/FileEditTool/UI.tsx +289 -0
  1215. package/src/tools/FileEditTool/constants.ts +11 -0
  1216. package/src/tools/FileEditTool/prompt.ts +28 -0
  1217. package/src/tools/FileEditTool/types.ts +85 -0
  1218. package/src/tools/FileEditTool/utils.ts +775 -0
  1219. package/src/tools/FileReadTool/FileReadTool.ts +1183 -0
  1220. package/src/tools/FileReadTool/UI.tsx +185 -0
  1221. package/src/tools/FileReadTool/imageProcessor.ts +94 -0
  1222. package/src/tools/FileReadTool/limits.ts +92 -0
  1223. package/src/tools/FileReadTool/prompt.ts +49 -0
  1224. package/src/tools/FileWriteTool/FileWriteTool.ts +434 -0
  1225. package/src/tools/FileWriteTool/UI.tsx +405 -0
  1226. package/src/tools/FileWriteTool/prompt.ts +18 -0
  1227. package/src/tools/GlobTool/GlobTool.ts +198 -0
  1228. package/src/tools/GlobTool/UI.tsx +63 -0
  1229. package/src/tools/GlobTool/prompt.ts +7 -0
  1230. package/src/tools/GrepTool/GrepTool.ts +577 -0
  1231. package/src/tools/GrepTool/UI.tsx +201 -0
  1232. package/src/tools/GrepTool/prompt.ts +18 -0
  1233. package/src/tools/LSPTool/LSPTool.ts +860 -0
  1234. package/src/tools/LSPTool/UI.tsx +228 -0
  1235. package/src/tools/LSPTool/formatters.ts +592 -0
  1236. package/src/tools/LSPTool/prompt.ts +21 -0
  1237. package/src/tools/LSPTool/schemas.ts +215 -0
  1238. package/src/tools/LSPTool/symbolContext.ts +90 -0
  1239. package/src/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
  1240. package/src/tools/ListMcpResourcesTool/UI.tsx +29 -0
  1241. package/src/tools/ListMcpResourcesTool/prompt.ts +20 -0
  1242. package/src/tools/MCPTool/MCPTool.ts +77 -0
  1243. package/src/tools/MCPTool/UI.tsx +403 -0
  1244. package/src/tools/MCPTool/classifyForCollapse.ts +604 -0
  1245. package/src/tools/MCPTool/prompt.ts +3 -0
  1246. package/src/tools/McpAuthTool/McpAuthTool.ts +215 -0
  1247. package/src/tools/NotebookEditTool/NotebookEditTool.ts +490 -0
  1248. package/src/tools/NotebookEditTool/UI.tsx +93 -0
  1249. package/src/tools/NotebookEditTool/constants.ts +2 -0
  1250. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  1251. package/src/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
  1252. package/src/tools/PowerShellTool/UI.tsx +131 -0
  1253. package/src/tools/PowerShellTool/clmTypes.ts +211 -0
  1254. package/src/tools/PowerShellTool/commandSemantics.ts +142 -0
  1255. package/src/tools/PowerShellTool/commonParameters.ts +30 -0
  1256. package/src/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
  1257. package/src/tools/PowerShellTool/gitSafety.ts +176 -0
  1258. package/src/tools/PowerShellTool/modeValidation.ts +404 -0
  1259. package/src/tools/PowerShellTool/pathValidation.ts +2049 -0
  1260. package/src/tools/PowerShellTool/powershellPermissions.ts +1648 -0
  1261. package/src/tools/PowerShellTool/powershellSecurity.ts +1090 -0
  1262. package/src/tools/PowerShellTool/prompt.ts +145 -0
  1263. package/src/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
  1264. package/src/tools/PowerShellTool/toolName.ts +2 -0
  1265. package/src/tools/REPLTool/constants.ts +46 -0
  1266. package/src/tools/REPLTool/primitiveTools.ts +39 -0
  1267. package/src/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
  1268. package/src/tools/ReadMcpResourceTool/UI.tsx +37 -0
  1269. package/src/tools/ReadMcpResourceTool/prompt.ts +16 -0
  1270. package/src/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
  1271. package/src/tools/RemoteTriggerTool/UI.tsx +17 -0
  1272. package/src/tools/RemoteTriggerTool/prompt.ts +15 -0
  1273. package/src/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
  1274. package/src/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
  1275. package/src/tools/ScheduleCronTool/CronListTool.ts +97 -0
  1276. package/src/tools/ScheduleCronTool/UI.tsx +60 -0
  1277. package/src/tools/ScheduleCronTool/prompt.ts +135 -0
  1278. package/src/tools/SendMessageTool/SendMessageTool.ts +917 -0
  1279. package/src/tools/SendMessageTool/UI.tsx +31 -0
  1280. package/src/tools/SendMessageTool/constants.ts +1 -0
  1281. package/src/tools/SendMessageTool/prompt.ts +49 -0
  1282. package/src/tools/SkillTool/SkillTool.ts +1108 -0
  1283. package/src/tools/SkillTool/UI.tsx +128 -0
  1284. package/src/tools/SkillTool/constants.ts +1 -0
  1285. package/src/tools/SkillTool/prompt.ts +241 -0
  1286. package/src/tools/SleepTool/prompt.ts +17 -0
  1287. package/src/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
  1288. package/src/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
  1289. package/src/tools/TaskCreateTool/constants.ts +1 -0
  1290. package/src/tools/TaskCreateTool/prompt.ts +56 -0
  1291. package/src/tools/TaskGetTool/TaskGetTool.ts +128 -0
  1292. package/src/tools/TaskGetTool/constants.ts +1 -0
  1293. package/src/tools/TaskGetTool/prompt.ts +24 -0
  1294. package/src/tools/TaskListTool/TaskListTool.ts +116 -0
  1295. package/src/tools/TaskListTool/constants.ts +1 -0
  1296. package/src/tools/TaskListTool/prompt.ts +49 -0
  1297. package/src/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
  1298. package/src/tools/TaskOutputTool/constants.ts +1 -0
  1299. package/src/tools/TaskStopTool/TaskStopTool.ts +131 -0
  1300. package/src/tools/TaskStopTool/UI.tsx +41 -0
  1301. package/src/tools/TaskStopTool/prompt.ts +8 -0
  1302. package/src/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
  1303. package/src/tools/TaskUpdateTool/constants.ts +1 -0
  1304. package/src/tools/TaskUpdateTool/prompt.ts +77 -0
  1305. package/src/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
  1306. package/src/tools/TeamCreateTool/UI.tsx +6 -0
  1307. package/src/tools/TeamCreateTool/constants.ts +1 -0
  1308. package/src/tools/TeamCreateTool/prompt.ts +113 -0
  1309. package/src/tools/TeamDeleteTool/TeamDeleteTool.ts +139 -0
  1310. package/src/tools/TeamDeleteTool/UI.tsx +20 -0
  1311. package/src/tools/TeamDeleteTool/constants.ts +1 -0
  1312. package/src/tools/TeamDeleteTool/prompt.ts +16 -0
  1313. package/src/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
  1314. package/src/tools/TodoWriteTool/constants.ts +1 -0
  1315. package/src/tools/TodoWriteTool/prompt.ts +184 -0
  1316. package/src/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
  1317. package/src/tools/ToolSearchTool/constants.ts +1 -0
  1318. package/src/tools/ToolSearchTool/prompt.ts +121 -0
  1319. package/src/tools/TungstenTool/TungstenTool.js +2 -0
  1320. package/src/tools/TungstenTool/TungstenTool.ts +1 -0
  1321. package/src/tools/WebFetchTool/UI.tsx +72 -0
  1322. package/src/tools/WebFetchTool/WebFetchTool.ts +318 -0
  1323. package/src/tools/WebFetchTool/preapproved.ts +166 -0
  1324. package/src/tools/WebFetchTool/prompt.ts +46 -0
  1325. package/src/tools/WebFetchTool/utils.ts +530 -0
  1326. package/src/tools/WebSearchTool/UI.tsx +101 -0
  1327. package/src/tools/WebSearchTool/WebSearchTool.ts +435 -0
  1328. package/src/tools/WebSearchTool/prompt.ts +34 -0
  1329. package/src/tools/shared/gitOperationTracking.ts +277 -0
  1330. package/src/tools/shared/spawnMultiAgent.ts +1093 -0
  1331. package/src/tools/testing/TestingPermissionTool.tsx +74 -0
  1332. package/src/tools/utils.ts +40 -0
  1333. package/src/tools.ts +389 -0
  1334. package/src/types/command.ts +216 -0
  1335. package/src/types/connectorText.js +5 -0
  1336. package/src/types/connectorText.ts +1 -0
  1337. package/src/types/generated/events_mono/claude_code/v1/claude_code_internal_event.ts +865 -0
  1338. package/src/types/generated/events_mono/common/v1/auth.ts +100 -0
  1339. package/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +223 -0
  1340. package/src/types/generated/google/protobuf/timestamp.ts +187 -0
  1341. package/src/types/hooks.ts +290 -0
  1342. package/src/types/ids.ts +44 -0
  1343. package/src/types/logs.ts +330 -0
  1344. package/src/types/permissions.ts +441 -0
  1345. package/src/types/plugin.ts +363 -0
  1346. package/src/types/textInputTypes.ts +387 -0
  1347. package/src/upstreamproxy/relay.ts +455 -0
  1348. package/src/upstreamproxy/upstreamproxy.ts +285 -0
  1349. package/src/utils/CircularBuffer.ts +84 -0
  1350. package/src/utils/Cursor.ts +1530 -0
  1351. package/src/utils/QueryGuard.ts +121 -0
  1352. package/src/utils/Shell.ts +474 -0
  1353. package/src/utils/ShellCommand.ts +465 -0
  1354. package/src/utils/abortController.ts +99 -0
  1355. package/src/utils/activityManager.ts +164 -0
  1356. package/src/utils/advisor.ts +145 -0
  1357. package/src/utils/agentContext.ts +178 -0
  1358. package/src/utils/agentId.ts +99 -0
  1359. package/src/utils/agentSwarmsEnabled.ts +44 -0
  1360. package/src/utils/agenticSessionSearch.ts +307 -0
  1361. package/src/utils/analyzeContext.ts +1382 -0
  1362. package/src/utils/ansiToPng.ts +334 -0
  1363. package/src/utils/ansiToSvg.ts +272 -0
  1364. package/src/utils/api.ts +718 -0
  1365. package/src/utils/apiPreconnect.ts +72 -0
  1366. package/src/utils/appleTerminalBackup.ts +124 -0
  1367. package/src/utils/argumentSubstitution.ts +145 -0
  1368. package/src/utils/array.ts +13 -0
  1369. package/src/utils/asciicast.ts +239 -0
  1370. package/src/utils/attachments.ts +3997 -0
  1371. package/src/utils/attribution.ts +393 -0
  1372. package/src/utils/auth.ts +2007 -0
  1373. package/src/utils/authFileDescriptor.ts +196 -0
  1374. package/src/utils/authPortable.ts +19 -0
  1375. package/src/utils/autoModeDenials.ts +26 -0
  1376. package/src/utils/autoRunIssue.tsx +122 -0
  1377. package/src/utils/autoUpdater.ts +562 -0
  1378. package/src/utils/aws.ts +74 -0
  1379. package/src/utils/awsAuthStatusManager.ts +81 -0
  1380. package/src/utils/background/remote/preconditions.ts +235 -0
  1381. package/src/utils/background/remote/remoteSession.ts +98 -0
  1382. package/src/utils/backgroundHousekeeping.ts +94 -0
  1383. package/src/utils/bash/ParsedCommand.ts +318 -0
  1384. package/src/utils/bash/ShellSnapshot.ts +582 -0
  1385. package/src/utils/bash/ast.ts +2679 -0
  1386. package/src/utils/bash/bashParser.ts +4436 -0
  1387. package/src/utils/bash/bashPipeCommand.ts +294 -0
  1388. package/src/utils/bash/commands.ts +1339 -0
  1389. package/src/utils/bash/heredoc.ts +733 -0
  1390. package/src/utils/bash/parser.ts +230 -0
  1391. package/src/utils/bash/prefix.ts +204 -0
  1392. package/src/utils/bash/registry.ts +53 -0
  1393. package/src/utils/bash/shellCompletion.ts +259 -0
  1394. package/src/utils/bash/shellPrefix.ts +28 -0
  1395. package/src/utils/bash/shellQuote.ts +304 -0
  1396. package/src/utils/bash/shellQuoting.ts +128 -0
  1397. package/src/utils/bash/specs/alias.ts +14 -0
  1398. package/src/utils/bash/specs/index.ts +18 -0
  1399. package/src/utils/bash/specs/nohup.ts +13 -0
  1400. package/src/utils/bash/specs/pyright.ts +91 -0
  1401. package/src/utils/bash/specs/sleep.ts +13 -0
  1402. package/src/utils/bash/specs/srun.ts +31 -0
  1403. package/src/utils/bash/specs/time.ts +13 -0
  1404. package/src/utils/bash/specs/timeout.ts +20 -0
  1405. package/src/utils/bash/treeSitterAnalysis.ts +506 -0
  1406. package/src/utils/betas.ts +438 -0
  1407. package/src/utils/billing.ts +78 -0
  1408. package/src/utils/binaryCheck.ts +53 -0
  1409. package/src/utils/browser.ts +68 -0
  1410. package/src/utils/bufferedWriter.ts +100 -0
  1411. package/src/utils/bundledMode.ts +22 -0
  1412. package/src/utils/caCerts.ts +115 -0
  1413. package/src/utils/caCertsConfig.ts +88 -0
  1414. package/src/utils/cachePaths.ts +38 -0
  1415. package/src/utils/classifierApprovals.ts +88 -0
  1416. package/src/utils/classifierApprovalsHook.ts +17 -0
  1417. package/src/utils/claudeCodeHints.ts +193 -0
  1418. package/src/utils/claudeDesktop.ts +152 -0
  1419. package/src/utils/claudeInChrome/chromeNativeHost.ts +527 -0
  1420. package/src/utils/claudeInChrome/common.ts +540 -0
  1421. package/src/utils/claudeInChrome/mcpServer.ts +293 -0
  1422. package/src/utils/claudeInChrome/prompt.ts +83 -0
  1423. package/src/utils/claudeInChrome/setup.ts +400 -0
  1424. package/src/utils/claudeInChrome/setupPortable.ts +233 -0
  1425. package/src/utils/claudeInChrome/toolRendering.tsx +262 -0
  1426. package/src/utils/claudemd.ts +1479 -0
  1427. package/src/utils/cleanup.ts +602 -0
  1428. package/src/utils/cleanupRegistry.ts +25 -0
  1429. package/src/utils/cliArgs.ts +60 -0
  1430. package/src/utils/cliHighlight.ts +54 -0
  1431. package/src/utils/codeIndexing.ts +206 -0
  1432. package/src/utils/collapseBackgroundBashNotifications.ts +84 -0
  1433. package/src/utils/collapseHookSummaries.ts +59 -0
  1434. package/src/utils/collapseReadSearch.ts +1109 -0
  1435. package/src/utils/collapseTeammateShutdowns.ts +55 -0
  1436. package/src/utils/combinedAbortSignal.ts +47 -0
  1437. package/src/utils/commandLifecycle.ts +21 -0
  1438. package/src/utils/commitAttribution.ts +961 -0
  1439. package/src/utils/completionCache.ts +166 -0
  1440. package/src/utils/computerUse/appNames.ts +196 -0
  1441. package/src/utils/computerUse/cleanup.ts +86 -0
  1442. package/src/utils/computerUse/common.ts +61 -0
  1443. package/src/utils/computerUse/computerUseLock.ts +215 -0
  1444. package/src/utils/computerUse/drainRunLoop.ts +79 -0
  1445. package/src/utils/computerUse/escHotkey.ts +54 -0
  1446. package/src/utils/computerUse/executor.ts +658 -0
  1447. package/src/utils/computerUse/gates.ts +72 -0
  1448. package/src/utils/computerUse/hostAdapter.ts +69 -0
  1449. package/src/utils/computerUse/inputLoader.ts +30 -0
  1450. package/src/utils/computerUse/mcpServer.ts +106 -0
  1451. package/src/utils/computerUse/setup.ts +53 -0
  1452. package/src/utils/computerUse/swiftLoader.ts +23 -0
  1453. package/src/utils/computerUse/toolRendering.tsx +125 -0
  1454. package/src/utils/computerUse/wrapper.tsx +336 -0
  1455. package/src/utils/concurrentSessions.ts +204 -0
  1456. package/src/utils/config.ts +1817 -0
  1457. package/src/utils/configConstants.ts +21 -0
  1458. package/src/utils/contentArray.ts +51 -0
  1459. package/src/utils/context.ts +221 -0
  1460. package/src/utils/contextAnalysis.ts +272 -0
  1461. package/src/utils/contextSuggestions.ts +235 -0
  1462. package/src/utils/controlMessageCompat.ts +32 -0
  1463. package/src/utils/conversationRecovery.ts +597 -0
  1464. package/src/utils/cron.ts +308 -0
  1465. package/src/utils/cronJitterConfig.ts +75 -0
  1466. package/src/utils/cronScheduler.ts +565 -0
  1467. package/src/utils/cronTasks.ts +458 -0
  1468. package/src/utils/cronTasksLock.ts +195 -0
  1469. package/src/utils/crossProjectResume.ts +75 -0
  1470. package/src/utils/crypto.ts +13 -0
  1471. package/src/utils/cwd.ts +32 -0
  1472. package/src/utils/debug.ts +268 -0
  1473. package/src/utils/debugFilter.ts +157 -0
  1474. package/src/utils/deepLink/banner.ts +123 -0
  1475. package/src/utils/deepLink/parseDeepLink.ts +170 -0
  1476. package/src/utils/deepLink/protocolHandler.ts +136 -0
  1477. package/src/utils/deepLink/registerProtocol.ts +348 -0
  1478. package/src/utils/deepLink/terminalLauncher.ts +557 -0
  1479. package/src/utils/deepLink/terminalPreference.ts +54 -0
  1480. package/src/utils/desktopDeepLink.ts +236 -0
  1481. package/src/utils/detectRepository.ts +178 -0
  1482. package/src/utils/diagLogs.ts +94 -0
  1483. package/src/utils/diff.ts +177 -0
  1484. package/src/utils/directMemberMessage.ts +69 -0
  1485. package/src/utils/displayTags.ts +51 -0
  1486. package/src/utils/doctorContextWarnings.ts +265 -0
  1487. package/src/utils/doctorDiagnostic.ts +625 -0
  1488. package/src/utils/dxt/helpers.ts +88 -0
  1489. package/src/utils/dxt/zip.ts +226 -0
  1490. package/src/utils/earlyInput.ts +191 -0
  1491. package/src/utils/editor.ts +183 -0
  1492. package/src/utils/effort.ts +329 -0
  1493. package/src/utils/embeddedTools.ts +29 -0
  1494. package/src/utils/env.ts +347 -0
  1495. package/src/utils/envDynamic.ts +151 -0
  1496. package/src/utils/envUtils.ts +183 -0
  1497. package/src/utils/envValidation.ts +38 -0
  1498. package/src/utils/errorLogSink.ts +235 -0
  1499. package/src/utils/errors.ts +238 -0
  1500. package/src/utils/exampleCommands.ts +184 -0
  1501. package/src/utils/execFileNoThrow.ts +150 -0
  1502. package/src/utils/execFileNoThrowPortable.ts +89 -0
  1503. package/src/utils/execSyncWrapper.ts +38 -0
  1504. package/src/utils/exportRenderer.tsx +98 -0
  1505. package/src/utils/extraUsage.ts +23 -0
  1506. package/src/utils/fastMode.ts +532 -0
  1507. package/src/utils/file.ts +584 -0
  1508. package/src/utils/fileHistory.ts +1115 -0
  1509. package/src/utils/fileOperationAnalytics.ts +71 -0
  1510. package/src/utils/filePersistence/filePersistence.ts +287 -0
  1511. package/src/utils/filePersistence/outputsScanner.ts +126 -0
  1512. package/src/utils/fileRead.ts +102 -0
  1513. package/src/utils/fileReadCache.ts +96 -0
  1514. package/src/utils/fileStateCache.ts +142 -0
  1515. package/src/utils/findExecutable.ts +17 -0
  1516. package/src/utils/fingerprint.ts +76 -0
  1517. package/src/utils/forkedAgent.ts +689 -0
  1518. package/src/utils/format.ts +308 -0
  1519. package/src/utils/formatBriefTimestamp.ts +81 -0
  1520. package/src/utils/fpsTracker.ts +47 -0
  1521. package/src/utils/frontmatterParser.ts +370 -0
  1522. package/src/utils/fsOperations.ts +770 -0
  1523. package/src/utils/fullscreen.ts +202 -0
  1524. package/src/utils/generatedFiles.ts +136 -0
  1525. package/src/utils/generators.ts +88 -0
  1526. package/src/utils/genericProcessUtils.ts +184 -0
  1527. package/src/utils/getWorktreePaths.ts +70 -0
  1528. package/src/utils/getWorktreePathsPortable.ts +27 -0
  1529. package/src/utils/ghPrStatus.ts +106 -0
  1530. package/src/utils/git/gitConfigParser.ts +277 -0
  1531. package/src/utils/git/gitFilesystem.ts +699 -0
  1532. package/src/utils/git/gitignore.ts +99 -0
  1533. package/src/utils/git.ts +926 -0
  1534. package/src/utils/gitDiff.ts +532 -0
  1535. package/src/utils/gitSettings.ts +18 -0
  1536. package/src/utils/github/ghAuthStatus.ts +29 -0
  1537. package/src/utils/githubRepoPathMapping.ts +162 -0
  1538. package/src/utils/glob.ts +130 -0
  1539. package/src/utils/gracefulShutdown.ts +529 -0
  1540. package/src/utils/groupToolUses.ts +182 -0
  1541. package/src/utils/handlePromptSubmit.ts +610 -0
  1542. package/src/utils/hash.ts +46 -0
  1543. package/src/utils/headlessProfiler.ts +178 -0
  1544. package/src/utils/heapDumpService.ts +303 -0
  1545. package/src/utils/heatmap.ts +198 -0
  1546. package/src/utils/highlightMatch.tsx +28 -0
  1547. package/src/utils/hooks/AsyncHookRegistry.ts +309 -0
  1548. package/src/utils/hooks/apiQueryHookHelper.ts +141 -0
  1549. package/src/utils/hooks/execAgentHook.ts +339 -0
  1550. package/src/utils/hooks/execHttpHook.ts +242 -0
  1551. package/src/utils/hooks/execPromptHook.ts +211 -0
  1552. package/src/utils/hooks/fileChangedWatcher.ts +191 -0
  1553. package/src/utils/hooks/hookEvents.ts +192 -0
  1554. package/src/utils/hooks/hookHelpers.ts +83 -0
  1555. package/src/utils/hooks/hooksConfigManager.ts +400 -0
  1556. package/src/utils/hooks/hooksConfigSnapshot.ts +133 -0
  1557. package/src/utils/hooks/hooksSettings.ts +271 -0
  1558. package/src/utils/hooks/postSamplingHooks.ts +70 -0
  1559. package/src/utils/hooks/registerFrontmatterHooks.ts +67 -0
  1560. package/src/utils/hooks/registerSkillHooks.ts +64 -0
  1561. package/src/utils/hooks/sessionHooks.ts +447 -0
  1562. package/src/utils/hooks/skillImprovement.ts +267 -0
  1563. package/src/utils/hooks/ssrfGuard.ts +294 -0
  1564. package/src/utils/hooks.ts +5022 -0
  1565. package/src/utils/horizontalScroll.ts +137 -0
  1566. package/src/utils/http.ts +136 -0
  1567. package/src/utils/hyperlink.ts +39 -0
  1568. package/src/utils/iTermBackup.ts +73 -0
  1569. package/src/utils/ide.ts +1494 -0
  1570. package/src/utils/idePathConversion.ts +90 -0
  1571. package/src/utils/idleTimeout.ts +53 -0
  1572. package/src/utils/imagePaste.ts +416 -0
  1573. package/src/utils/imageResizer.ts +880 -0
  1574. package/src/utils/imageStore.ts +167 -0
  1575. package/src/utils/imageValidation.ts +104 -0
  1576. package/src/utils/immediateCommand.ts +15 -0
  1577. package/src/utils/inProcessTeammateHelpers.ts +102 -0
  1578. package/src/utils/ink.ts +26 -0
  1579. package/src/utils/intl.ts +94 -0
  1580. package/src/utils/jetbrains.ts +191 -0
  1581. package/src/utils/json.ts +277 -0
  1582. package/src/utils/jsonRead.ts +16 -0
  1583. package/src/utils/keyboardShortcuts.ts +14 -0
  1584. package/src/utils/lazySchema.ts +8 -0
  1585. package/src/utils/listSessionsImpl.ts +454 -0
  1586. package/src/utils/localInstaller.ts +162 -0
  1587. package/src/utils/lockfile.ts +43 -0
  1588. package/src/utils/log.ts +362 -0
  1589. package/src/utils/logoV2Utils.ts +350 -0
  1590. package/src/utils/mailbox.ts +73 -0
  1591. package/src/utils/managedEnv.ts +199 -0
  1592. package/src/utils/managedEnvConstants.ts +191 -0
  1593. package/src/utils/markdown.ts +381 -0
  1594. package/src/utils/markdownConfigLoader.ts +600 -0
  1595. package/src/utils/mcp/dateTimeParser.ts +121 -0
  1596. package/src/utils/mcp/elicitationValidation.ts +336 -0
  1597. package/src/utils/mcpInstructionsDelta.ts +130 -0
  1598. package/src/utils/mcpOutputStorage.ts +189 -0
  1599. package/src/utils/mcpValidation.ts +208 -0
  1600. package/src/utils/mcpWebSocketTransport.ts +200 -0
  1601. package/src/utils/memoize.ts +269 -0
  1602. package/src/utils/memory/types.ts +12 -0
  1603. package/src/utils/memory/versions.ts +8 -0
  1604. package/src/utils/memoryFileDetection.ts +289 -0
  1605. package/src/utils/messagePredicates.ts +8 -0
  1606. package/src/utils/messageQueueManager.ts +547 -0
  1607. package/src/utils/messages/mappers.ts +290 -0
  1608. package/src/utils/messages/systemInit.ts +96 -0
  1609. package/src/utils/messages.ts +5512 -0
  1610. package/src/utils/model/agent.ts +157 -0
  1611. package/src/utils/model/aliases.ts +25 -0
  1612. package/src/utils/model/antModels.ts +64 -0
  1613. package/src/utils/model/bedrock.ts +265 -0
  1614. package/src/utils/model/check1mAccess.ts +72 -0
  1615. package/src/utils/model/configs.ts +118 -0
  1616. package/src/utils/model/contextWindowUpgradeCheck.ts +47 -0
  1617. package/src/utils/model/deprecation.ts +101 -0
  1618. package/src/utils/model/model.ts +634 -0
  1619. package/src/utils/model/modelAllowlist.ts +170 -0
  1620. package/src/utils/model/modelCapabilities.ts +118 -0
  1621. package/src/utils/model/modelOptions.ts +540 -0
  1622. package/src/utils/model/modelStrings.ts +166 -0
  1623. package/src/utils/model/modelSupportOverrides.ts +50 -0
  1624. package/src/utils/model/providers.ts +46 -0
  1625. package/src/utils/model/validateModel.ts +159 -0
  1626. package/src/utils/modelCost.ts +235 -0
  1627. package/src/utils/modifiers.ts +36 -0
  1628. package/src/utils/mtls.ts +179 -0
  1629. package/src/utils/nativeInstaller/download.ts +523 -0
  1630. package/src/utils/nativeInstaller/index.ts +18 -0
  1631. package/src/utils/nativeInstaller/installer.ts +1708 -0
  1632. package/src/utils/nativeInstaller/packageManagers.ts +336 -0
  1633. package/src/utils/nativeInstaller/pidLock.ts +433 -0
  1634. package/src/utils/notebook.ts +224 -0
  1635. package/src/utils/objectGroupBy.ts +18 -0
  1636. package/src/utils/pasteStore.ts +104 -0
  1637. package/src/utils/path.ts +155 -0
  1638. package/src/utils/pdf.ts +300 -0
  1639. package/src/utils/pdfUtils.ts +70 -0
  1640. package/src/utils/peerAddress.ts +21 -0
  1641. package/src/utils/permissions/PermissionMode.ts +141 -0
  1642. package/src/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
  1643. package/src/utils/permissions/PermissionResult.ts +35 -0
  1644. package/src/utils/permissions/PermissionRule.ts +40 -0
  1645. package/src/utils/permissions/PermissionUpdate.ts +389 -0
  1646. package/src/utils/permissions/PermissionUpdateSchema.ts +78 -0
  1647. package/src/utils/permissions/autoModeState.ts +39 -0
  1648. package/src/utils/permissions/bashClassifier.ts +61 -0
  1649. package/src/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
  1650. package/src/utils/permissions/classifierDecision.ts +98 -0
  1651. package/src/utils/permissions/classifierShared.ts +39 -0
  1652. package/src/utils/permissions/dangerousPatterns.ts +80 -0
  1653. package/src/utils/permissions/denialTracking.ts +45 -0
  1654. package/src/utils/permissions/filesystem.ts +1777 -0
  1655. package/src/utils/permissions/getNextPermissionMode.ts +101 -0
  1656. package/src/utils/permissions/pathValidation.ts +485 -0
  1657. package/src/utils/permissions/permissionExplainer.ts +250 -0
  1658. package/src/utils/permissions/permissionRuleParser.ts +198 -0
  1659. package/src/utils/permissions/permissionSetup.ts +1532 -0
  1660. package/src/utils/permissions/permissions.ts +1486 -0
  1661. package/src/utils/permissions/permissionsLoader.ts +296 -0
  1662. package/src/utils/permissions/shadowedRuleDetection.ts +234 -0
  1663. package/src/utils/permissions/shellRuleMatching.ts +228 -0
  1664. package/src/utils/permissions/yoloClassifier.ts +1495 -0
  1665. package/src/utils/planModeV2.ts +95 -0
  1666. package/src/utils/plans.ts +397 -0
  1667. package/src/utils/platform.ts +150 -0
  1668. package/src/utils/plugins/addDirPluginSettings.ts +71 -0
  1669. package/src/utils/plugins/cacheUtils.ts +196 -0
  1670. package/src/utils/plugins/dependencyResolver.ts +305 -0
  1671. package/src/utils/plugins/fetchTelemetry.ts +135 -0
  1672. package/src/utils/plugins/gitAvailability.ts +69 -0
  1673. package/src/utils/plugins/headlessPluginInstall.ts +174 -0
  1674. package/src/utils/plugins/hintRecommendation.ts +164 -0
  1675. package/src/utils/plugins/installCounts.ts +292 -0
  1676. package/src/utils/plugins/installedPluginsManager.ts +1268 -0
  1677. package/src/utils/plugins/loadPluginAgents.ts +348 -0
  1678. package/src/utils/plugins/loadPluginCommands.ts +946 -0
  1679. package/src/utils/plugins/loadPluginHooks.ts +287 -0
  1680. package/src/utils/plugins/loadPluginOutputStyles.ts +178 -0
  1681. package/src/utils/plugins/lspPluginIntegration.ts +387 -0
  1682. package/src/utils/plugins/lspRecommendation.ts +374 -0
  1683. package/src/utils/plugins/managedPlugins.ts +27 -0
  1684. package/src/utils/plugins/marketplaceHelpers.ts +592 -0
  1685. package/src/utils/plugins/marketplaceManager.ts +2643 -0
  1686. package/src/utils/plugins/mcpPluginIntegration.ts +634 -0
  1687. package/src/utils/plugins/mcpbHandler.ts +968 -0
  1688. package/src/utils/plugins/officialMarketplace.ts +25 -0
  1689. package/src/utils/plugins/officialMarketplaceGcs.ts +216 -0
  1690. package/src/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
  1691. package/src/utils/plugins/orphanedPluginFilter.ts +114 -0
  1692. package/src/utils/plugins/parseMarketplaceInput.ts +162 -0
  1693. package/src/utils/plugins/performStartupChecks.tsx +70 -0
  1694. package/src/utils/plugins/pluginAutoupdate.ts +284 -0
  1695. package/src/utils/plugins/pluginBlocklist.ts +127 -0
  1696. package/src/utils/plugins/pluginDirectories.ts +178 -0
  1697. package/src/utils/plugins/pluginFlagging.ts +208 -0
  1698. package/src/utils/plugins/pluginIdentifier.ts +123 -0
  1699. package/src/utils/plugins/pluginInstallationHelpers.ts +595 -0
  1700. package/src/utils/plugins/pluginLoader.ts +3302 -0
  1701. package/src/utils/plugins/pluginOptionsStorage.ts +400 -0
  1702. package/src/utils/plugins/pluginPolicy.ts +20 -0
  1703. package/src/utils/plugins/pluginStartupCheck.ts +341 -0
  1704. package/src/utils/plugins/pluginVersioning.ts +157 -0
  1705. package/src/utils/plugins/reconciler.ts +265 -0
  1706. package/src/utils/plugins/refresh.ts +215 -0
  1707. package/src/utils/plugins/schemas.ts +1681 -0
  1708. package/src/utils/plugins/validatePlugin.ts +903 -0
  1709. package/src/utils/plugins/walkPluginMarkdown.ts +69 -0
  1710. package/src/utils/plugins/zipCache.ts +406 -0
  1711. package/src/utils/plugins/zipCacheAdapters.ts +164 -0
  1712. package/src/utils/powershell/dangerousCmdlets.ts +185 -0
  1713. package/src/utils/powershell/parser.ts +1804 -0
  1714. package/src/utils/powershell/staticPrefix.ts +316 -0
  1715. package/src/utils/preflightChecks.tsx +151 -0
  1716. package/src/utils/privacyLevel.ts +55 -0
  1717. package/src/utils/process.ts +68 -0
  1718. package/src/utils/processUserInput/processBashCommand.tsx +140 -0
  1719. package/src/utils/processUserInput/processSlashCommand.tsx +922 -0
  1720. package/src/utils/processUserInput/processTextPrompt.ts +100 -0
  1721. package/src/utils/processUserInput/processUserInput.ts +605 -0
  1722. package/src/utils/profilerBase.ts +46 -0
  1723. package/src/utils/promptCategory.ts +49 -0
  1724. package/src/utils/promptEditor.ts +188 -0
  1725. package/src/utils/promptShellExecution.ts +183 -0
  1726. package/src/utils/proxy.ts +426 -0
  1727. package/src/utils/queryContext.ts +179 -0
  1728. package/src/utils/queryHelpers.ts +552 -0
  1729. package/src/utils/queryProfiler.ts +301 -0
  1730. package/src/utils/queueProcessor.ts +95 -0
  1731. package/src/utils/readEditContext.ts +227 -0
  1732. package/src/utils/readFileInRange.ts +383 -0
  1733. package/src/utils/releaseNotes.ts +360 -0
  1734. package/src/utils/renderOptions.ts +77 -0
  1735. package/src/utils/ripgrep.ts +679 -0
  1736. package/src/utils/sandbox/sandbox-adapter.ts +985 -0
  1737. package/src/utils/sandbox/sandbox-ui-utils.ts +12 -0
  1738. package/src/utils/sanitization.ts +91 -0
  1739. package/src/utils/screenshotClipboard.ts +121 -0
  1740. package/src/utils/sdkEventQueue.ts +134 -0
  1741. package/src/utils/secureStorage/fallbackStorage.ts +70 -0
  1742. package/src/utils/secureStorage/index.ts +17 -0
  1743. package/src/utils/secureStorage/keychainPrefetch.ts +116 -0
  1744. package/src/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
  1745. package/src/utils/secureStorage/macOsKeychainStorage.ts +231 -0
  1746. package/src/utils/secureStorage/plainTextStorage.ts +84 -0
  1747. package/src/utils/semanticBoolean.ts +29 -0
  1748. package/src/utils/semanticNumber.ts +36 -0
  1749. package/src/utils/semver.ts +59 -0
  1750. package/src/utils/sequential.ts +56 -0
  1751. package/src/utils/sessionActivity.ts +133 -0
  1752. package/src/utils/sessionEnvVars.ts +22 -0
  1753. package/src/utils/sessionEnvironment.ts +166 -0
  1754. package/src/utils/sessionFileAccessHooks.ts +250 -0
  1755. package/src/utils/sessionIngressAuth.ts +140 -0
  1756. package/src/utils/sessionRestore.ts +551 -0
  1757. package/src/utils/sessionStart.ts +232 -0
  1758. package/src/utils/sessionState.ts +150 -0
  1759. package/src/utils/sessionStorage.ts +5105 -0
  1760. package/src/utils/sessionStoragePortable.ts +793 -0
  1761. package/src/utils/sessionTitle.ts +129 -0
  1762. package/src/utils/sessionUrl.ts +64 -0
  1763. package/src/utils/set.ts +53 -0
  1764. package/src/utils/settings/allErrors.ts +32 -0
  1765. package/src/utils/settings/applySettingsChange.ts +92 -0
  1766. package/src/utils/settings/changeDetector.ts +488 -0
  1767. package/src/utils/settings/constants.ts +202 -0
  1768. package/src/utils/settings/internalWrites.ts +37 -0
  1769. package/src/utils/settings/managedPath.ts +34 -0
  1770. package/src/utils/settings/mdm/constants.ts +81 -0
  1771. package/src/utils/settings/mdm/rawRead.ts +130 -0
  1772. package/src/utils/settings/mdm/settings.ts +316 -0
  1773. package/src/utils/settings/permissionValidation.ts +262 -0
  1774. package/src/utils/settings/pluginOnlyPolicy.ts +60 -0
  1775. package/src/utils/settings/schemaOutput.ts +8 -0
  1776. package/src/utils/settings/settings.ts +1015 -0
  1777. package/src/utils/settings/settingsCache.ts +80 -0
  1778. package/src/utils/settings/toolValidationConfig.ts +103 -0
  1779. package/src/utils/settings/types.ts +1148 -0
  1780. package/src/utils/settings/validateEditTool.ts +45 -0
  1781. package/src/utils/settings/validation.ts +265 -0
  1782. package/src/utils/settings/validationTips.ts +164 -0
  1783. package/src/utils/shell/bashProvider.ts +255 -0
  1784. package/src/utils/shell/outputLimits.ts +14 -0
  1785. package/src/utils/shell/powershellDetection.ts +107 -0
  1786. package/src/utils/shell/powershellProvider.ts +123 -0
  1787. package/src/utils/shell/prefix.ts +367 -0
  1788. package/src/utils/shell/readOnlyCommandValidation.ts +1893 -0
  1789. package/src/utils/shell/resolveDefaultShell.ts +14 -0
  1790. package/src/utils/shell/shellProvider.ts +33 -0
  1791. package/src/utils/shell/shellToolUtils.ts +22 -0
  1792. package/src/utils/shell/specPrefix.ts +241 -0
  1793. package/src/utils/shellConfig.ts +167 -0
  1794. package/src/utils/sideQuery.ts +222 -0
  1795. package/src/utils/sideQuestion.ts +155 -0
  1796. package/src/utils/signal.ts +43 -0
  1797. package/src/utils/sinks.ts +16 -0
  1798. package/src/utils/skills/skillChangeDetector.ts +311 -0
  1799. package/src/utils/slashCommandParsing.ts +60 -0
  1800. package/src/utils/sleep.ts +84 -0
  1801. package/src/utils/sliceAnsi.ts +91 -0
  1802. package/src/utils/slowOperations.ts +286 -0
  1803. package/src/utils/standaloneAgent.ts +23 -0
  1804. package/src/utils/startupProfiler.ts +194 -0
  1805. package/src/utils/staticRender.tsx +116 -0
  1806. package/src/utils/stats.ts +1061 -0
  1807. package/src/utils/statsCache.ts +434 -0
  1808. package/src/utils/status.tsx +362 -0
  1809. package/src/utils/statusNoticeDefinitions.tsx +198 -0
  1810. package/src/utils/statusNoticeHelpers.ts +20 -0
  1811. package/src/utils/stream.ts +76 -0
  1812. package/src/utils/streamJsonStdoutGuard.ts +123 -0
  1813. package/src/utils/streamlinedTransform.ts +201 -0
  1814. package/src/utils/stringUtils.ts +235 -0
  1815. package/src/utils/subprocessEnv.ts +99 -0
  1816. package/src/utils/suggestions/commandSuggestions.ts +567 -0
  1817. package/src/utils/suggestions/directoryCompletion.ts +263 -0
  1818. package/src/utils/suggestions/shellHistoryCompletion.ts +119 -0
  1819. package/src/utils/suggestions/skillUsageTracking.ts +55 -0
  1820. package/src/utils/suggestions/slackChannelSuggestions.ts +209 -0
  1821. package/src/utils/swarm/It2SetupPrompt.tsx +380 -0
  1822. package/src/utils/swarm/backends/ITermBackend.ts +370 -0
  1823. package/src/utils/swarm/backends/InProcessBackend.ts +339 -0
  1824. package/src/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
  1825. package/src/utils/swarm/backends/TmuxBackend.ts +764 -0
  1826. package/src/utils/swarm/backends/detection.ts +128 -0
  1827. package/src/utils/swarm/backends/it2Setup.ts +245 -0
  1828. package/src/utils/swarm/backends/registry.ts +464 -0
  1829. package/src/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
  1830. package/src/utils/swarm/backends/types.ts +311 -0
  1831. package/src/utils/swarm/constants.ts +33 -0
  1832. package/src/utils/swarm/inProcessRunner.ts +1552 -0
  1833. package/src/utils/swarm/leaderPermissionBridge.ts +54 -0
  1834. package/src/utils/swarm/permissionSync.ts +928 -0
  1835. package/src/utils/swarm/reconnection.ts +119 -0
  1836. package/src/utils/swarm/spawnInProcess.ts +328 -0
  1837. package/src/utils/swarm/spawnUtils.ts +146 -0
  1838. package/src/utils/swarm/teamHelpers.ts +683 -0
  1839. package/src/utils/swarm/teammateInit.ts +129 -0
  1840. package/src/utils/swarm/teammateLayoutManager.ts +107 -0
  1841. package/src/utils/swarm/teammateModel.ts +10 -0
  1842. package/src/utils/swarm/teammatePromptAddendum.ts +18 -0
  1843. package/src/utils/systemDirectories.ts +74 -0
  1844. package/src/utils/systemPrompt.ts +123 -0
  1845. package/src/utils/systemPromptType.ts +14 -0
  1846. package/src/utils/systemTheme.ts +119 -0
  1847. package/src/utils/taggedId.ts +54 -0
  1848. package/src/utils/task/TaskOutput.ts +390 -0
  1849. package/src/utils/task/diskOutput.ts +451 -0
  1850. package/src/utils/task/framework.ts +308 -0
  1851. package/src/utils/task/outputFormatting.ts +38 -0
  1852. package/src/utils/task/sdkProgress.ts +36 -0
  1853. package/src/utils/tasks.ts +862 -0
  1854. package/src/utils/teamDiscovery.ts +81 -0
  1855. package/src/utils/teamMemoryOps.ts +88 -0
  1856. package/src/utils/teammate.ts +292 -0
  1857. package/src/utils/teammateContext.ts +96 -0
  1858. package/src/utils/teammateMailbox.ts +1183 -0
  1859. package/src/utils/telemetry/betaSessionTracing.ts +491 -0
  1860. package/src/utils/telemetry/bigqueryExporter.ts +252 -0
  1861. package/src/utils/telemetry/events.ts +75 -0
  1862. package/src/utils/telemetry/instrumentation.ts +825 -0
  1863. package/src/utils/telemetry/logger.ts +26 -0
  1864. package/src/utils/telemetry/perfettoTracing.ts +1120 -0
  1865. package/src/utils/telemetry/pluginTelemetry.ts +289 -0
  1866. package/src/utils/telemetry/sessionTracing.ts +927 -0
  1867. package/src/utils/telemetry/skillLoadedEvent.ts +39 -0
  1868. package/src/utils/telemetryAttributes.ts +71 -0
  1869. package/src/utils/teleport/api.ts +466 -0
  1870. package/src/utils/teleport/environmentSelection.ts +77 -0
  1871. package/src/utils/teleport/environments.ts +120 -0
  1872. package/src/utils/teleport/gitBundle.ts +292 -0
  1873. package/src/utils/teleport.tsx +1226 -0
  1874. package/src/utils/tempfile.ts +31 -0
  1875. package/src/utils/terminal.ts +131 -0
  1876. package/src/utils/terminalPanel.ts +191 -0
  1877. package/src/utils/textHighlighting.ts +166 -0
  1878. package/src/utils/theme.ts +639 -0
  1879. package/src/utils/thinking.ts +162 -0
  1880. package/src/utils/timeouts.ts +39 -0
  1881. package/src/utils/tmuxSocket.ts +427 -0
  1882. package/src/utils/todo/types.ts +18 -0
  1883. package/src/utils/tokenBudget.ts +73 -0
  1884. package/src/utils/tokens.ts +261 -0
  1885. package/src/utils/toolErrors.ts +132 -0
  1886. package/src/utils/toolPool.ts +79 -0
  1887. package/src/utils/toolResultStorage.ts +1040 -0
  1888. package/src/utils/toolSchemaCache.ts +26 -0
  1889. package/src/utils/toolSearch.ts +756 -0
  1890. package/src/utils/transcriptSearch.ts +202 -0
  1891. package/src/utils/treeify.ts +170 -0
  1892. package/src/utils/truncate.ts +179 -0
  1893. package/src/utils/ultraplan/ccrSession.ts +349 -0
  1894. package/src/utils/ultraplan/keyword.ts +127 -0
  1895. package/src/utils/ultraplan/prompt.txt +1 -0
  1896. package/src/utils/unaryLogging.ts +39 -0
  1897. package/src/utils/undercover.ts +89 -0
  1898. package/src/utils/user.ts +194 -0
  1899. package/src/utils/userAgent.ts +10 -0
  1900. package/src/utils/userPromptKeywords.ts +27 -0
  1901. package/src/utils/uuid.ts +27 -0
  1902. package/src/utils/warningHandler.ts +121 -0
  1903. package/src/utils/which.ts +82 -0
  1904. package/src/utils/windowsPaths.ts +173 -0
  1905. package/src/utils/withResolvers.ts +13 -0
  1906. package/src/utils/words.ts +800 -0
  1907. package/src/utils/workloadContext.ts +57 -0
  1908. package/src/utils/worktree.ts +1519 -0
  1909. package/src/utils/worktreeModeEnabled.ts +11 -0
  1910. package/src/utils/xdg.ts +65 -0
  1911. package/src/utils/xml.ts +16 -0
  1912. package/src/utils/yaml.ts +15 -0
  1913. package/src/utils/zodToJsonSchema.ts +23 -0
  1914. package/src/vim/motions.ts +82 -0
  1915. package/src/vim/operators.ts +556 -0
  1916. package/src/vim/textObjects.ts +186 -0
  1917. package/src/vim/transitions.ts +490 -0
  1918. package/src/vim/types.ts +199 -0
  1919. package/src/voice/voiceModeEnabled.ts +54 -0
  1920. package/start.js +1 -0
@@ -0,0 +1,2465 @@
1
+ import {
2
+ discoverAuthorizationServerMetadata,
3
+ discoverOAuthServerInfo,
4
+ type OAuthClientProvider,
5
+ type OAuthDiscoveryState,
6
+ auth as sdkAuth,
7
+ refreshAuthorization as sdkRefreshAuthorization,
8
+ } from '@modelcontextprotocol/sdk/client/auth.js'
9
+ import {
10
+ InvalidGrantError,
11
+ OAuthError,
12
+ ServerError,
13
+ TemporarilyUnavailableError,
14
+ TooManyRequestsError,
15
+ } from '@modelcontextprotocol/sdk/server/auth/errors.js'
16
+ import {
17
+ type AuthorizationServerMetadata,
18
+ type OAuthClientInformation,
19
+ type OAuthClientInformationFull,
20
+ type OAuthClientMetadata,
21
+ OAuthErrorResponseSchema,
22
+ OAuthMetadataSchema,
23
+ type OAuthTokens,
24
+ OAuthTokensSchema,
25
+ } from '@modelcontextprotocol/sdk/shared/auth.js'
26
+ import type { FetchLike } from '@modelcontextprotocol/sdk/shared/transport.js'
27
+ import axios from 'axios'
28
+ import { createHash, randomBytes, randomUUID } from 'crypto'
29
+ import { mkdir } from 'fs/promises'
30
+ import { createServer, type Server } from 'http'
31
+ import { join } from 'path'
32
+ import { parse } from 'url'
33
+ import xss from 'xss'
34
+ import { MCP_CLIENT_METADATA_URL } from '../../constants/oauth.js'
35
+ import { openBrowser } from '../../utils/browser.js'
36
+ import { getClaudeConfigHomeDir } from '../../utils/envUtils.js'
37
+ import { errorMessage, getErrnoCode } from '../../utils/errors.js'
38
+ import * as lockfile from '../../utils/lockfile.js'
39
+ import { logMCPDebug } from '../../utils/log.js'
40
+ import { getPlatform } from '../../utils/platform.js'
41
+ import { getSecureStorage } from '../../utils/secureStorage/index.js'
42
+ import { clearKeychainCache } from '../../utils/secureStorage/macOsKeychainHelpers.js'
43
+ import type { SecureStorageData } from '../../utils/secureStorage/types.js'
44
+ import { sleep } from '../../utils/sleep.js'
45
+ import { jsonParse, jsonStringify } from '../../utils/slowOperations.js'
46
+ import { logEvent } from '../analytics/index.js'
47
+ import type { AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS } from '../analytics/metadata.js'
48
+ import { buildRedirectUri, findAvailablePort } from './oauthPort.js'
49
+ import type { McpHTTPServerConfig, McpSSEServerConfig } from './types.js'
50
+ import { getLoggingSafeMcpBaseUrl } from './utils.js'
51
+ import { performCrossAppAccess, XaaTokenExchangeError } from './xaa.js'
52
+ import {
53
+ acquireIdpIdToken,
54
+ clearIdpIdToken,
55
+ discoverOidc,
56
+ getCachedIdpIdToken,
57
+ getIdpClientSecret,
58
+ getXaaIdpSettings,
59
+ isXaaEnabled,
60
+ } from './xaaIdpLogin.js'
61
+
62
+ /**
63
+ * Timeout for individual OAuth requests (metadata discovery, token refresh, etc.)
64
+ */
65
+ const AUTH_REQUEST_TIMEOUT_MS = 30000
66
+
67
+ /**
68
+ * Failure reasons for the `tengu_mcp_oauth_refresh_failure` event. Values
69
+ * are emitted to analytics — keep them stable (do not rename; add new ones).
70
+ */
71
+ type MCPRefreshFailureReason =
72
+ | 'metadata_discovery_failed'
73
+ | 'no_client_info'
74
+ | 'no_tokens_returned'
75
+ | 'invalid_grant'
76
+ | 'transient_retries_exhausted'
77
+ | 'request_failed'
78
+
79
+ /**
80
+ * Failure reasons for the `tengu_mcp_oauth_flow_error` event. Values are
81
+ * emitted to analytics for attribution in BigQuery. Keep stable (do not
82
+ * rename; add new ones).
83
+ */
84
+ type MCPOAuthFlowErrorReason =
85
+ | 'cancelled'
86
+ | 'timeout'
87
+ | 'provider_denied'
88
+ | 'state_mismatch'
89
+ | 'port_unavailable'
90
+ | 'sdk_auth_failed'
91
+ | 'token_exchange_failed'
92
+ | 'unknown'
93
+
94
+ const MAX_LOCK_RETRIES = 5
95
+
96
+ /**
97
+ * OAuth query parameters that should be redacted from logs.
98
+ * These contain sensitive values that could enable CSRF or session fixation attacks.
99
+ */
100
+ const SENSITIVE_OAUTH_PARAMS = [
101
+ 'state',
102
+ 'nonce',
103
+ 'code_challenge',
104
+ 'code_verifier',
105
+ 'code',
106
+ ]
107
+
108
+ /**
109
+ * Redacts sensitive OAuth query parameters from a URL for safe logging.
110
+ * Prevents exposure of state, nonce, code_challenge, code_verifier, and authorization codes.
111
+ */
112
+ function redactSensitiveUrlParams(url: string): string {
113
+ try {
114
+ const parsedUrl = new URL(url)
115
+ for (const param of SENSITIVE_OAUTH_PARAMS) {
116
+ if (parsedUrl.searchParams.has(param)) {
117
+ parsedUrl.searchParams.set(param, '[REDACTED]')
118
+ }
119
+ }
120
+ return parsedUrl.toString()
121
+ } catch {
122
+ // Return as-is if not a valid URL
123
+ return url
124
+ }
125
+ }
126
+
127
+ /**
128
+ * Some OAuth servers (notably Slack) return HTTP 200 for all responses,
129
+ * signaling errors via the JSON body instead. The SDK's executeTokenRequest
130
+ * only calls parseErrorResponse when !response.ok, so a 200 with
131
+ * {"error":"invalid_grant"} gets fed to OAuthTokensSchema.parse() and
132
+ * surfaces as a ZodError — which the refresh retry/invalidation logic
133
+ * treats as opaque request_failed instead of invalid_grant.
134
+ *
135
+ * This wrapper peeks at 2xx POST response bodies and rewrites ones that
136
+ * match OAuthErrorResponseSchema (but not OAuthTokensSchema) to a 400
137
+ * Response, so the SDK's normal error-class mapping applies. The same
138
+ * fetchFn is also used for DCR POSTs, but DCR success responses have no
139
+ * {error: string} field so they don't match the rewrite condition.
140
+ *
141
+ * Slack uses non-standard error codes (invalid_refresh_token observed live
142
+ * at oauth.v2.user.access; expired_refresh_token/token_expired per Slack's
143
+ * token rotation docs) where RFC 6749 specifies invalid_grant. We normalize
144
+ * those so OAUTH_ERRORS['invalid_grant'] → InvalidGrantError matches and
145
+ * token invalidation fires correctly.
146
+ */
147
+ const NONSTANDARD_INVALID_GRANT_ALIASES = new Set([
148
+ 'invalid_refresh_token',
149
+ 'expired_refresh_token',
150
+ 'token_expired',
151
+ ])
152
+
153
+ /* eslint-disable eslint-plugin-n/no-unsupported-features/node-builtins --
154
+ * Response has been stable in Node since 18; the rule flags it as
155
+ * experimental-until-21 which is incorrect. Pattern matches existing
156
+ * createAuthFetch suppressions in this file. */
157
+ export async function normalizeOAuthErrorBody(
158
+ response: Response,
159
+ ): Promise<Response> {
160
+ if (!response.ok) {
161
+ return response
162
+ }
163
+ const text = await response.text()
164
+ let parsed: unknown
165
+ try {
166
+ parsed = jsonParse(text)
167
+ } catch {
168
+ return new Response(text, response)
169
+ }
170
+ if (OAuthTokensSchema.safeParse(parsed).success) {
171
+ return new Response(text, response)
172
+ }
173
+ const result = OAuthErrorResponseSchema.safeParse(parsed)
174
+ if (!result.success) {
175
+ return new Response(text, response)
176
+ }
177
+ const normalized = NONSTANDARD_INVALID_GRANT_ALIASES.has(result.data.error)
178
+ ? {
179
+ error: 'invalid_grant',
180
+ error_description:
181
+ result.data.error_description ??
182
+ `Server returned non-standard error code: ${result.data.error}`,
183
+ }
184
+ : result.data
185
+ return new Response(jsonStringify(normalized), {
186
+ status: 400,
187
+ statusText: 'Bad Request',
188
+ headers: response.headers,
189
+ })
190
+ }
191
+ /* eslint-enable eslint-plugin-n/no-unsupported-features/node-builtins */
192
+
193
+ /**
194
+ * Creates a fetch function with a fresh 30-second timeout for each OAuth request.
195
+ * Used by ClaudeAuthProvider for metadata discovery and token refresh.
196
+ * Prevents stale timeout signals from affecting auth operations.
197
+ */
198
+ function createAuthFetch(): FetchLike {
199
+ return async (url: string | URL, init?: RequestInit) => {
200
+ const timeoutSignal = AbortSignal.timeout(AUTH_REQUEST_TIMEOUT_MS)
201
+ const isPost = init?.method?.toUpperCase() === 'POST'
202
+
203
+ // No existing signal - just use timeout
204
+ if (!init?.signal) {
205
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
206
+ const response = await fetch(url, { ...init, signal: timeoutSignal })
207
+ return isPost ? normalizeOAuthErrorBody(response) : response
208
+ }
209
+
210
+ // Combine signals: abort when either fires
211
+ const controller = new AbortController()
212
+ const abort = () => controller.abort()
213
+
214
+ init.signal.addEventListener('abort', abort)
215
+ timeoutSignal.addEventListener('abort', abort)
216
+
217
+ // Cleanup to prevent event listener leaks after fetch completes
218
+ const cleanup = () => {
219
+ init.signal?.removeEventListener('abort', abort)
220
+ timeoutSignal.removeEventListener('abort', abort)
221
+ }
222
+
223
+ if (init.signal.aborted) {
224
+ controller.abort()
225
+ }
226
+
227
+ try {
228
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
229
+ const response = await fetch(url, { ...init, signal: controller.signal })
230
+ cleanup()
231
+ return isPost ? normalizeOAuthErrorBody(response) : response
232
+ } catch (error) {
233
+ cleanup()
234
+ throw error
235
+ }
236
+ }
237
+ }
238
+
239
+ /**
240
+ * Fetches authorization server metadata, using a configured metadata URL if available,
241
+ * otherwise performing RFC 9728 → RFC 8414 discovery via the SDK.
242
+ *
243
+ * Discovery order when no configured URL:
244
+ * 1. RFC 9728: probe /.well-known/oauth-protected-resource on the MCP server,
245
+ * read authorization_servers[0], then RFC 8414 against that URL.
246
+ * 2. Fallback: RFC 8414 directly against the MCP server URL (path-aware). Covers
247
+ * legacy servers that co-host auth metadata at /.well-known/oauth-authorization-server/{path}
248
+ * without implementing RFC 9728. The SDK's own fallback strips the path, so this
249
+ * preserves the pre-existing path-aware probe for backward compatibility.
250
+ *
251
+ * Note: configuredMetadataUrl is user-controlled via .mcp.json. Project-scoped MCP
252
+ * servers require user approval before connecting (same trust level as the MCP server
253
+ * URL itself). The HTTPS requirement here is defense-in-depth beyond schema validation
254
+ * — RFC 8414 mandates OAuth metadata retrieval over TLS.
255
+ */
256
+ async function fetchAuthServerMetadata(
257
+ serverName: string,
258
+ serverUrl: string,
259
+ configuredMetadataUrl: string | undefined,
260
+ fetchFn?: FetchLike,
261
+ resourceMetadataUrl?: URL,
262
+ ): Promise<Awaited<ReturnType<typeof discoverAuthorizationServerMetadata>>> {
263
+ if (configuredMetadataUrl) {
264
+ if (!configuredMetadataUrl.startsWith('https://')) {
265
+ throw new Error(
266
+ `authServerMetadataUrl must use https:// (got: ${configuredMetadataUrl})`,
267
+ )
268
+ }
269
+ const authFetch = fetchFn ?? createAuthFetch()
270
+ const response = await authFetch(configuredMetadataUrl, {
271
+ headers: { Accept: 'application/json' },
272
+ })
273
+ if (response.ok) {
274
+ return OAuthMetadataSchema.parse(await response.json())
275
+ }
276
+ throw new Error(
277
+ `HTTP ${response.status} fetching configured auth server metadata from ${configuredMetadataUrl}`,
278
+ )
279
+ }
280
+
281
+ try {
282
+ const { authorizationServerMetadata } = await discoverOAuthServerInfo(
283
+ serverUrl,
284
+ {
285
+ ...(fetchFn && { fetchFn }),
286
+ ...(resourceMetadataUrl && { resourceMetadataUrl }),
287
+ },
288
+ )
289
+ if (authorizationServerMetadata) {
290
+ return authorizationServerMetadata
291
+ }
292
+ } catch (err) {
293
+ // Any error from the RFC 9728 → RFC 8414 chain (5xx from the root or
294
+ // resolved-AS probe, schema parse failure, network error) — fall through
295
+ // to the legacy path-aware retry.
296
+ logMCPDebug(
297
+ serverName,
298
+ `RFC 9728 discovery failed, falling back: ${errorMessage(err)}`,
299
+ )
300
+ }
301
+
302
+ // Fallback only when the URL has a path component; for root URLs the SDK's
303
+ // own fallback already probed the same endpoints.
304
+ const url = new URL(serverUrl)
305
+ if (url.pathname === '/') {
306
+ return undefined
307
+ }
308
+ return discoverAuthorizationServerMetadata(url, {
309
+ ...(fetchFn && { fetchFn }),
310
+ })
311
+ }
312
+
313
+ export class AuthenticationCancelledError extends Error {
314
+ constructor() {
315
+ super('Authentication was cancelled')
316
+ this.name = 'AuthenticationCancelledError'
317
+ }
318
+ }
319
+
320
+ /**
321
+ * Generates a unique key for server credentials based on both name and config hash
322
+ * This prevents credentials from being reused across different servers
323
+ * with the same name or different configurations
324
+ */
325
+ export function getServerKey(
326
+ serverName: string,
327
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
328
+ ): string {
329
+ const configJson = jsonStringify({
330
+ type: serverConfig.type,
331
+ url: serverConfig.url,
332
+ headers: serverConfig.headers || {},
333
+ })
334
+
335
+ const hash = createHash('sha256')
336
+ .update(configJson)
337
+ .digest('hex')
338
+ .substring(0, 16)
339
+
340
+ return `${serverName}|${hash}`
341
+ }
342
+
343
+ /**
344
+ * True when we have probed this server before (OAuth discovery state is
345
+ * stored) but hold no credentials to try. A connection attempt in this
346
+ * state is guaranteed to 401 — the only way out is the user running
347
+ * /mcp to authenticate.
348
+ */
349
+ export function hasMcpDiscoveryButNoToken(
350
+ serverName: string,
351
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
352
+ ): boolean {
353
+ // XAA servers can silently re-auth via cached id_token even without an
354
+ // access/refresh token — tokens() fires the xaaRefresh path. Skipping the
355
+ // connection here would make that auto-auth branch unreachable after
356
+ // invalidateCredentials('tokens') clears the stored tokens.
357
+ if (isXaaEnabled() && serverConfig.oauth?.xaa) {
358
+ return false
359
+ }
360
+ const serverKey = getServerKey(serverName, serverConfig)
361
+ const entry = getSecureStorage().read()?.mcpOAuth?.[serverKey]
362
+ return entry !== undefined && !entry.accessToken && !entry.refreshToken
363
+ }
364
+
365
+ /**
366
+ * Revokes a single token on the OAuth server.
367
+ *
368
+ * Per RFC 7009, public clients (like Claude Code) should authenticate by including
369
+ * client_id in the request body, NOT via an Authorization header. The Bearer token
370
+ * in an Authorization header is meant for resource owner authentication, not client
371
+ * authentication.
372
+ *
373
+ * However, the MCP spec doesn't explicitly define token revocation behavior, so some
374
+ * servers may not be RFC 7009 compliant. As defensive programming, we:
375
+ * 1. First try the RFC 7009 compliant approach (client_id in body, no Authorization header)
376
+ * 2. If we get a 401, retry with Bearer auth as a fallback for non-compliant servers
377
+ *
378
+ * This fallback should rarely be needed - most servers either accept the compliant
379
+ * approach or ignore unexpected headers.
380
+ */
381
+ async function revokeToken({
382
+ serverName,
383
+ endpoint,
384
+ token,
385
+ tokenTypeHint,
386
+ clientId,
387
+ clientSecret,
388
+ accessToken,
389
+ authMethod = 'client_secret_basic',
390
+ }: {
391
+ serverName: string
392
+ endpoint: string
393
+ token: string
394
+ tokenTypeHint: 'access_token' | 'refresh_token'
395
+ clientId?: string
396
+ clientSecret?: string
397
+ accessToken?: string
398
+ authMethod?: 'client_secret_basic' | 'client_secret_post'
399
+ }): Promise<void> {
400
+ const params = new URLSearchParams()
401
+ params.set('token', token)
402
+ params.set('token_type_hint', tokenTypeHint)
403
+
404
+ const headers: Record<string, string> = {
405
+ 'Content-Type': 'application/x-www-form-urlencoded',
406
+ }
407
+
408
+ // RFC 7009 §2.1 requires client auth per RFC 6749 §2.3. XAA always uses a
409
+ // confidential client at the AS — strict ASes (Okta/Stytch) reject public-
410
+ // client revocation of confidential-client tokens.
411
+ if (clientId && clientSecret) {
412
+ if (authMethod === 'client_secret_post') {
413
+ params.set('client_id', clientId)
414
+ params.set('client_secret', clientSecret)
415
+ } else {
416
+ const basic = Buffer.from(
417
+ `${encodeURIComponent(clientId)}:${encodeURIComponent(clientSecret)}`,
418
+ ).toString('base64')
419
+ headers.Authorization = `Basic ${basic}`
420
+ }
421
+ } else if (clientId) {
422
+ params.set('client_id', clientId)
423
+ } else {
424
+ logMCPDebug(
425
+ serverName,
426
+ `No client_id available for ${tokenTypeHint} revocation - server may reject`,
427
+ )
428
+ }
429
+
430
+ try {
431
+ await axios.post(endpoint, params, { headers })
432
+ logMCPDebug(serverName, `Successfully revoked ${tokenTypeHint}`)
433
+ } catch (error: unknown) {
434
+ // Fallback for non-RFC-7009-compliant servers that require Bearer auth
435
+ if (
436
+ axios.isAxiosError(error) &&
437
+ error.response?.status === 401 &&
438
+ accessToken
439
+ ) {
440
+ logMCPDebug(
441
+ serverName,
442
+ `Got 401, retrying ${tokenTypeHint} revocation with Bearer auth`,
443
+ )
444
+ // RFC 6749 §2.3.1: must not send more than one auth method. The retry
445
+ // switches to Bearer — clear any client creds from the body.
446
+ params.delete('client_id')
447
+ params.delete('client_secret')
448
+ await axios.post(endpoint, params, {
449
+ headers: { ...headers, Authorization: `Bearer ${accessToken}` },
450
+ })
451
+ logMCPDebug(
452
+ serverName,
453
+ `Successfully revoked ${tokenTypeHint} with Bearer auth`,
454
+ )
455
+ } else {
456
+ throw error
457
+ }
458
+ }
459
+ }
460
+
461
+ /**
462
+ * Revokes tokens on the OAuth server if a revocation endpoint is available.
463
+ * Per RFC 7009, we revoke the refresh token first (the long-lived credential),
464
+ * then the access token. Revoking the refresh token prevents generation of new
465
+ * access tokens and many servers implicitly invalidate associated access tokens.
466
+ */
467
+ export async function revokeServerTokens(
468
+ serverName: string,
469
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
470
+ { preserveStepUpState = false }: { preserveStepUpState?: boolean } = {},
471
+ ): Promise<void> {
472
+ const storage = getSecureStorage()
473
+ const existingData = storage.read()
474
+ if (!existingData?.mcpOAuth) return
475
+
476
+ const serverKey = getServerKey(serverName, serverConfig)
477
+ const tokenData = existingData.mcpOAuth[serverKey]
478
+
479
+ // Attempt server-side revocation if there are tokens to revoke (best-effort)
480
+ if (tokenData?.accessToken || tokenData?.refreshToken) {
481
+ try {
482
+ // For XAA (and any PRM-discovered auth), the AS is at a different host
483
+ // than the MCP URL — use the persisted discoveryState if we have it.
484
+ const asUrl =
485
+ tokenData.discoveryState?.authorizationServerUrl ?? serverConfig.url
486
+ const metadata = await fetchAuthServerMetadata(
487
+ serverName,
488
+ asUrl,
489
+ serverConfig.oauth?.authServerMetadataUrl,
490
+ )
491
+
492
+ if (!metadata) {
493
+ logMCPDebug(serverName, 'No OAuth metadata found')
494
+ } else {
495
+ const revocationEndpoint =
496
+ 'revocation_endpoint' in metadata
497
+ ? metadata.revocation_endpoint
498
+ : null
499
+ if (!revocationEndpoint) {
500
+ logMCPDebug(serverName, 'Server does not support token revocation')
501
+ } else {
502
+ const revocationEndpointStr = String(revocationEndpoint)
503
+ // RFC 7009 defines revocation_endpoint_auth_methods_supported
504
+ // separately from the token endpoint's list; prefer it if present.
505
+ const authMethods =
506
+ ('revocation_endpoint_auth_methods_supported' in metadata
507
+ ? metadata.revocation_endpoint_auth_methods_supported
508
+ : undefined) ??
509
+ ('token_endpoint_auth_methods_supported' in metadata
510
+ ? metadata.token_endpoint_auth_methods_supported
511
+ : undefined)
512
+ const authMethod: 'client_secret_basic' | 'client_secret_post' =
513
+ authMethods &&
514
+ !authMethods.includes('client_secret_basic') &&
515
+ authMethods.includes('client_secret_post')
516
+ ? 'client_secret_post'
517
+ : 'client_secret_basic'
518
+ logMCPDebug(
519
+ serverName,
520
+ `Revoking tokens via ${revocationEndpointStr} (${authMethod})`,
521
+ )
522
+
523
+ // Revoke refresh token first (more important - prevents future access token generation)
524
+ if (tokenData.refreshToken) {
525
+ try {
526
+ await revokeToken({
527
+ serverName,
528
+ endpoint: revocationEndpointStr,
529
+ token: tokenData.refreshToken,
530
+ tokenTypeHint: 'refresh_token',
531
+ clientId: tokenData.clientId,
532
+ clientSecret: tokenData.clientSecret,
533
+ accessToken: tokenData.accessToken,
534
+ authMethod,
535
+ })
536
+ } catch (error: unknown) {
537
+ // Log but continue
538
+ logMCPDebug(
539
+ serverName,
540
+ `Failed to revoke refresh token: ${errorMessage(error)}`,
541
+ )
542
+ }
543
+ }
544
+
545
+ // Then revoke access token (may already be invalidated by refresh token revocation)
546
+ if (tokenData.accessToken) {
547
+ try {
548
+ await revokeToken({
549
+ serverName,
550
+ endpoint: revocationEndpointStr,
551
+ token: tokenData.accessToken,
552
+ tokenTypeHint: 'access_token',
553
+ clientId: tokenData.clientId,
554
+ clientSecret: tokenData.clientSecret,
555
+ accessToken: tokenData.accessToken,
556
+ authMethod,
557
+ })
558
+ } catch (error: unknown) {
559
+ logMCPDebug(
560
+ serverName,
561
+ `Failed to revoke access token: ${errorMessage(error)}`,
562
+ )
563
+ }
564
+ }
565
+ }
566
+ }
567
+ } catch (error: unknown) {
568
+ // Log error but don't throw - revocation is best-effort
569
+ logMCPDebug(serverName, `Failed to revoke tokens: ${errorMessage(error)}`)
570
+ }
571
+ } else {
572
+ logMCPDebug(serverName, 'No tokens to revoke')
573
+ }
574
+
575
+ // Always clear local tokens, regardless of server-side revocation result.
576
+ clearServerTokensFromLocalStorage(serverName, serverConfig)
577
+
578
+ // When re-authenticating, preserve step-up auth state (scope + discovery)
579
+ // so the next performMCPOAuthFlow can use cached scope instead of
580
+ // re-probing. For "Clear Auth" (default), wipe everything.
581
+ if (
582
+ preserveStepUpState &&
583
+ tokenData &&
584
+ (tokenData.stepUpScope || tokenData.discoveryState)
585
+ ) {
586
+ const freshData = storage.read() || {}
587
+ const updatedData: SecureStorageData = {
588
+ ...freshData,
589
+ mcpOAuth: {
590
+ ...freshData.mcpOAuth,
591
+ [serverKey]: {
592
+ ...freshData.mcpOAuth?.[serverKey],
593
+ serverName,
594
+ serverUrl: serverConfig.url,
595
+ accessToken: freshData.mcpOAuth?.[serverKey]?.accessToken ?? '',
596
+ expiresAt: freshData.mcpOAuth?.[serverKey]?.expiresAt ?? 0,
597
+ ...(tokenData.stepUpScope
598
+ ? { stepUpScope: tokenData.stepUpScope }
599
+ : {}),
600
+ ...(tokenData.discoveryState
601
+ ? {
602
+ // Strip legacy bulky metadata fields here too so users with
603
+ // existing overflowed blobs recover on next re-auth (#30337).
604
+ discoveryState: {
605
+ authorizationServerUrl:
606
+ tokenData.discoveryState.authorizationServerUrl,
607
+ resourceMetadataUrl:
608
+ tokenData.discoveryState.resourceMetadataUrl,
609
+ },
610
+ }
611
+ : {}),
612
+ },
613
+ },
614
+ }
615
+ storage.update(updatedData)
616
+ logMCPDebug(serverName, 'Preserved step-up auth state across revocation')
617
+ }
618
+ }
619
+
620
+ export function clearServerTokensFromLocalStorage(
621
+ serverName: string,
622
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
623
+ ): void {
624
+ const storage = getSecureStorage()
625
+ const existingData = storage.read()
626
+ if (!existingData?.mcpOAuth) return
627
+
628
+ const serverKey = getServerKey(serverName, serverConfig)
629
+ if (existingData.mcpOAuth[serverKey]) {
630
+ delete existingData.mcpOAuth[serverKey]
631
+ storage.update(existingData)
632
+ logMCPDebug(serverName, 'Cleared stored tokens')
633
+ }
634
+ }
635
+
636
+ type WWWAuthenticateParams = {
637
+ scope?: string
638
+ resourceMetadataUrl?: URL
639
+ }
640
+
641
+ type XaaFailureStage =
642
+ | 'idp_login'
643
+ | 'discovery'
644
+ | 'token_exchange'
645
+ | 'jwt_bearer'
646
+
647
+ /**
648
+ * XAA (Cross-App Access) auth.
649
+ *
650
+ * One IdP browser login is reused across all XAA-configured MCP servers:
651
+ * 1. Acquire an id_token from the IdP (cached in keychain by issuer; if
652
+ * missing/expired, runs a standard OIDC authorization_code+PKCE flow
653
+ * — this is the one browser pop)
654
+ * 2. Run the RFC 8693 + RFC 7523 exchange (no browser)
655
+ * 3. Save tokens to the same keychain slot as normal OAuth
656
+ *
657
+ * IdP connection details come from settings.xaaIdp (configured once via
658
+ * `claude mcp xaa setup`). Per-server config is just `oauth.xaa: true`
659
+ * plus the AS clientId/clientSecret.
660
+ *
661
+ * No silent fallback: if `oauth.xaa` is set, XAA is the only path.
662
+ * All errors are actionable — they tell the user what to run.
663
+ */
664
+ async function performMCPXaaAuth(
665
+ serverName: string,
666
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
667
+ onAuthorizationUrl: (url: string) => void,
668
+ abortSignal?: AbortSignal,
669
+ skipBrowserOpen?: boolean,
670
+ ): Promise<void> {
671
+ if (!serverConfig.oauth?.xaa) {
672
+ throw new Error('XAA: oauth.xaa must be set') // guarded by caller
673
+ }
674
+
675
+ // IdP config comes from user-level settings, not per-server.
676
+ const idp = getXaaIdpSettings()
677
+ if (!idp) {
678
+ throw new Error(
679
+ "XAA: no IdP connection configured. Run 'claude mcp xaa setup --issuer <url> --client-id <id> --client-secret' to configure.",
680
+ )
681
+ }
682
+
683
+ const clientId = serverConfig.oauth?.clientId
684
+ if (!clientId) {
685
+ throw new Error(
686
+ `XAA: server '${serverName}' needs an AS client_id. Re-add with --client-id.`,
687
+ )
688
+ }
689
+
690
+ const clientConfig = getMcpClientConfig(serverName, serverConfig)
691
+ const clientSecret = clientConfig?.clientSecret
692
+ if (!clientSecret) {
693
+ // Diagnostic context for serverKey mismatch debugging. Only computed
694
+ // on the error path so there's no perf cost on success.
695
+ const wantedKey = getServerKey(serverName, serverConfig)
696
+ const haveKeys = Object.keys(
697
+ getSecureStorage().read()?.mcpOAuthClientConfig ?? {},
698
+ )
699
+ const headersForLogging = Object.fromEntries(
700
+ Object.entries(serverConfig.headers ?? {}).map(([k, v]) =>
701
+ k.toLowerCase() === 'authorization' ? [k, '[REDACTED]'] : [k, v],
702
+ ),
703
+ )
704
+ logMCPDebug(
705
+ serverName,
706
+ `XAA: secret lookup miss. wanted=${wantedKey} have=[${haveKeys.join(', ')}] configHeaders=${jsonStringify(headersForLogging)}`,
707
+ )
708
+ throw new Error(
709
+ `XAA: AS client secret not found for '${serverName}'. Re-add with --client-secret.`,
710
+ )
711
+ }
712
+
713
+ logMCPDebug(serverName, 'XAA: starting cross-app access flow')
714
+
715
+ // IdP client secret lives in a separate keychain slot (keyed by IdP issuer),
716
+ // NOT the AS secret — different trust domain. Optional: if absent, PKCE-only.
717
+ const idpClientSecret = getIdpClientSecret(idp.issuer)
718
+
719
+ // Acquire id_token (cached or via one OIDC browser pop at the IdP).
720
+ // Peek the cache first so we can report idTokenCacheHit in analytics before
721
+ // acquireIdpIdToken potentially writes a fresh one.
722
+ const idTokenCacheHit = getCachedIdpIdToken(idp.issuer) !== undefined
723
+
724
+ let failureStage: XaaFailureStage = 'idp_login'
725
+ try {
726
+ let idToken
727
+ try {
728
+ idToken = await acquireIdpIdToken({
729
+ idpIssuer: idp.issuer,
730
+ idpClientId: idp.clientId,
731
+ idpClientSecret,
732
+ callbackPort: idp.callbackPort,
733
+ onAuthorizationUrl,
734
+ skipBrowserOpen,
735
+ abortSignal,
736
+ })
737
+ } catch (e) {
738
+ if (abortSignal?.aborted) throw new AuthenticationCancelledError()
739
+ throw e
740
+ }
741
+
742
+ // Discover the IdP's token endpoint for the RFC 8693 exchange.
743
+ failureStage = 'discovery'
744
+ const oidc = await discoverOidc(idp.issuer)
745
+
746
+ // Run the exchange. performCrossAppAccess throws XaaTokenExchangeError
747
+ // for the IdP leg and "jwt-bearer grant failed" for the AS leg.
748
+ failureStage = 'token_exchange'
749
+ let tokens
750
+ try {
751
+ tokens = await performCrossAppAccess(
752
+ serverConfig.url,
753
+ {
754
+ clientId,
755
+ clientSecret,
756
+ idpClientId: idp.clientId,
757
+ idpClientSecret,
758
+ idpIdToken: idToken,
759
+ idpTokenEndpoint: oidc.token_endpoint,
760
+ },
761
+ serverName,
762
+ abortSignal,
763
+ )
764
+ } catch (e) {
765
+ if (abortSignal?.aborted) throw new AuthenticationCancelledError()
766
+ const msg = errorMessage(e)
767
+ // If the IdP says the id_token is bad, drop it from the cache so the
768
+ // next attempt does a fresh IdP login. XaaTokenExchangeError carries
769
+ // shouldClearIdToken so we key off OAuth semantics (4xx / invalid body
770
+ // → clear; 5xx IdP outage → preserve) rather than substring matching.
771
+ if (e instanceof XaaTokenExchangeError) {
772
+ if (e.shouldClearIdToken) {
773
+ clearIdpIdToken(idp.issuer)
774
+ logMCPDebug(
775
+ serverName,
776
+ 'XAA: cleared cached id_token after token-exchange failure',
777
+ )
778
+ }
779
+ } else if (
780
+ msg.includes('PRM discovery failed') ||
781
+ msg.includes('AS metadata discovery failed') ||
782
+ msg.includes('no authorization server supports jwt-bearer')
783
+ ) {
784
+ // performCrossAppAccess runs PRM + AS discovery before the actual
785
+ // exchange — don't attribute their failures to 'token_exchange'.
786
+ failureStage = 'discovery'
787
+ } else if (msg.includes('jwt-bearer')) {
788
+ failureStage = 'jwt_bearer'
789
+ }
790
+ throw e
791
+ }
792
+
793
+ // Save tokens via the same storage path as normal OAuth. We write directly
794
+ // (instead of ClaudeAuthProvider.saveTokens) to avoid instantiating the
795
+ // whole provider just to write the same keys.
796
+ const storage = getSecureStorage()
797
+ const existingData = storage.read() || {}
798
+ const serverKey = getServerKey(serverName, serverConfig)
799
+ const prev = existingData.mcpOAuth?.[serverKey]
800
+ storage.update({
801
+ ...existingData,
802
+ mcpOAuth: {
803
+ ...existingData.mcpOAuth,
804
+ [serverKey]: {
805
+ ...prev,
806
+ serverName,
807
+ serverUrl: serverConfig.url,
808
+ accessToken: tokens.access_token,
809
+ // AS may omit refresh_token on jwt-bearer — preserve any existing one
810
+ refreshToken: tokens.refresh_token ?? prev?.refreshToken,
811
+ expiresAt: Date.now() + (tokens.expires_in || 3600) * 1000,
812
+ scope: tokens.scope,
813
+ clientId,
814
+ clientSecret,
815
+ // Persist the AS URL so _doRefresh and revokeServerTokens can locate
816
+ // the token/revocation endpoints when MCP URL ≠ AS URL (the common
817
+ // XAA topology).
818
+ discoveryState: {
819
+ authorizationServerUrl: tokens.authorizationServerUrl,
820
+ },
821
+ },
822
+ },
823
+ })
824
+
825
+ logMCPDebug(serverName, 'XAA: tokens saved')
826
+ logEvent('tengu_mcp_oauth_flow_success', {
827
+ authMethod:
828
+ 'xaa' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
829
+ idTokenCacheHit,
830
+ })
831
+ } catch (e) {
832
+ // User-initiated cancel (Esc during IdP browser pop) isn't a failure.
833
+ if (e instanceof AuthenticationCancelledError) {
834
+ throw e
835
+ }
836
+ logEvent('tengu_mcp_oauth_flow_failure', {
837
+ authMethod:
838
+ 'xaa' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
839
+ xaaFailureStage:
840
+ failureStage as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
841
+ idTokenCacheHit,
842
+ })
843
+ throw e
844
+ }
845
+ }
846
+
847
+ export async function performMCPOAuthFlow(
848
+ serverName: string,
849
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
850
+ onAuthorizationUrl: (url: string) => void,
851
+ abortSignal?: AbortSignal,
852
+ options?: {
853
+ skipBrowserOpen?: boolean
854
+ onWaitingForCallback?: (submit: (callbackUrl: string) => void) => void
855
+ },
856
+ ): Promise<void> {
857
+ // XAA (SEP-990): if configured, bypass the per-server consent dance.
858
+ // If the IdP id_token isn't cached, this pops the browser once at the IdP
859
+ // (shared across all XAA servers for that issuer). Subsequent servers hit
860
+ // the cache and are silent. Tokens land in the same keychain slot, so the
861
+ // rest of CC's transport wiring (ClaudeAuthProvider.tokens() in client.ts)
862
+ // works unchanged.
863
+ //
864
+ // No silent fallback: if `oauth.xaa` is set, XAA is the only path. We
865
+ // never fall through to the consent flow — that would be surprising (the
866
+ // user explicitly asked for XAA) and security-relevant (consent flow may
867
+ // have a different trust/scope posture than the org's IdP policy).
868
+ //
869
+ // Servers with `oauth.xaa` but CLAUDE_CODE_ENABLE_XAA unset hard-fail with
870
+ // actionable copy rather than silently degrade to consent.
871
+ if (serverConfig.oauth?.xaa) {
872
+ if (!isXaaEnabled()) {
873
+ throw new Error(
874
+ `XAA is not enabled (set CLAUDE_CODE_ENABLE_XAA=1). Remove 'oauth.xaa' from server '${serverName}' to use the standard consent flow.`,
875
+ )
876
+ }
877
+ logEvent('tengu_mcp_oauth_flow_start', {
878
+ isOAuthFlow: true,
879
+ authMethod:
880
+ 'xaa' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
881
+ transportType:
882
+ serverConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
883
+ ...(getLoggingSafeMcpBaseUrl(serverConfig)
884
+ ? {
885
+ mcpServerBaseUrl: getLoggingSafeMcpBaseUrl(
886
+ serverConfig,
887
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
888
+ }
889
+ : {}),
890
+ })
891
+ // performMCPXaaAuth logs its own success/failure events (with
892
+ // idTokenCacheHit + xaaFailureStage).
893
+ await performMCPXaaAuth(
894
+ serverName,
895
+ serverConfig,
896
+ onAuthorizationUrl,
897
+ abortSignal,
898
+ options?.skipBrowserOpen,
899
+ )
900
+ return
901
+ }
902
+
903
+ // Check for cached step-up scope and resource metadata URL before clearing
904
+ // tokens. The transport-attached auth provider persists scope when it receives
905
+ // a step-up 401, so we can use it here instead of making an extra probe request.
906
+ const storage = getSecureStorage()
907
+ const serverKey = getServerKey(serverName, serverConfig)
908
+ const cachedEntry = storage.read()?.mcpOAuth?.[serverKey]
909
+ const cachedStepUpScope = cachedEntry?.stepUpScope
910
+ const cachedResourceMetadataUrl =
911
+ cachedEntry?.discoveryState?.resourceMetadataUrl
912
+
913
+ // Clear any existing stored credentials to ensure fresh client registration.
914
+ // Note: this deletes the entire entry (including discoveryState/stepUpScope),
915
+ // but we already read the cached values above.
916
+ clearServerTokensFromLocalStorage(serverName, serverConfig)
917
+
918
+ // Use cached step-up scope and resource metadata URL if available.
919
+ // The transport-attached auth provider caches these when it receives a
920
+ // step-up 401, so we don't need to probe the server again.
921
+ let resourceMetadataUrl: URL | undefined
922
+ if (cachedResourceMetadataUrl) {
923
+ try {
924
+ resourceMetadataUrl = new URL(cachedResourceMetadataUrl)
925
+ } catch {
926
+ logMCPDebug(
927
+ serverName,
928
+ `Invalid cached resourceMetadataUrl: ${cachedResourceMetadataUrl}`,
929
+ )
930
+ }
931
+ }
932
+ const wwwAuthParams: WWWAuthenticateParams = {
933
+ scope: cachedStepUpScope,
934
+ resourceMetadataUrl,
935
+ }
936
+
937
+ const flowAttemptId = randomUUID()
938
+
939
+ logEvent('tengu_mcp_oauth_flow_start', {
940
+ flowAttemptId:
941
+ flowAttemptId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
942
+ isOAuthFlow: true,
943
+ transportType:
944
+ serverConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
945
+ ...(getLoggingSafeMcpBaseUrl(serverConfig)
946
+ ? {
947
+ mcpServerBaseUrl: getLoggingSafeMcpBaseUrl(
948
+ serverConfig,
949
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
950
+ }
951
+ : {}),
952
+ })
953
+
954
+ // Track whether we reached the token-exchange phase so the catch block can
955
+ // attribute the failure reason correctly.
956
+ let authorizationCodeObtained = false
957
+
958
+ try {
959
+ // Use configured callback port for pre-configured OAuth, otherwise find an available port
960
+ const configuredCallbackPort = serverConfig.oauth?.callbackPort
961
+ const port = configuredCallbackPort ?? (await findAvailablePort())
962
+ const redirectUri = buildRedirectUri(port)
963
+ logMCPDebug(
964
+ serverName,
965
+ `Using redirect port: ${port}${configuredCallbackPort ? ' (from config)' : ''}`,
966
+ )
967
+
968
+ const provider = new ClaudeAuthProvider(
969
+ serverName,
970
+ serverConfig,
971
+ redirectUri,
972
+ true,
973
+ onAuthorizationUrl,
974
+ options?.skipBrowserOpen,
975
+ )
976
+
977
+ // Fetch and store OAuth metadata for scope information
978
+ try {
979
+ const metadata = await fetchAuthServerMetadata(
980
+ serverName,
981
+ serverConfig.url,
982
+ serverConfig.oauth?.authServerMetadataUrl,
983
+ undefined,
984
+ wwwAuthParams.resourceMetadataUrl,
985
+ )
986
+ if (metadata) {
987
+ // Store metadata in provider for scope information
988
+ provider.setMetadata(metadata)
989
+ logMCPDebug(
990
+ serverName,
991
+ `Fetched OAuth metadata with scope: ${getScopeFromMetadata(metadata) || 'NONE'}`,
992
+ )
993
+ }
994
+ } catch (error) {
995
+ logMCPDebug(
996
+ serverName,
997
+ `Failed to fetch OAuth metadata: ${errorMessage(error)}`,
998
+ )
999
+ }
1000
+
1001
+ // Get the OAuth state from the provider for validation
1002
+ const oauthState = await provider.state()
1003
+
1004
+ // Store the server, timeout, and abort listener references for cleanup
1005
+ let server: Server | null = null
1006
+ let timeoutId: NodeJS.Timeout | null = null
1007
+ let abortHandler: (() => void) | null = null
1008
+
1009
+ const cleanup = () => {
1010
+ if (server) {
1011
+ server.removeAllListeners()
1012
+ // Defensive: removeAllListeners() strips the error handler, so swallow any late error during close
1013
+ server.on('error', () => {})
1014
+ server.close()
1015
+ server = null
1016
+ }
1017
+ if (timeoutId) {
1018
+ clearTimeout(timeoutId)
1019
+ timeoutId = null
1020
+ }
1021
+ if (abortSignal && abortHandler) {
1022
+ abortSignal.removeEventListener('abort', abortHandler)
1023
+ abortHandler = null
1024
+ }
1025
+ logMCPDebug(serverName, `MCP OAuth server cleaned up`)
1026
+ }
1027
+
1028
+ // Setup a server to receive the callback
1029
+ const authorizationCode = await new Promise<string>((resolve, reject) => {
1030
+ let resolved = false
1031
+ const resolveOnce = (code: string) => {
1032
+ if (resolved) return
1033
+ resolved = true
1034
+ resolve(code)
1035
+ }
1036
+ const rejectOnce = (error: Error) => {
1037
+ if (resolved) return
1038
+ resolved = true
1039
+ reject(error)
1040
+ }
1041
+
1042
+ if (abortSignal) {
1043
+ abortHandler = () => {
1044
+ cleanup()
1045
+ rejectOnce(new AuthenticationCancelledError())
1046
+ }
1047
+ if (abortSignal.aborted) {
1048
+ abortHandler()
1049
+ return
1050
+ }
1051
+ abortSignal.addEventListener('abort', abortHandler)
1052
+ }
1053
+
1054
+ // Allow manual callback URL paste for remote/browser-based environments
1055
+ // where localhost is not reachable from the user's browser.
1056
+ if (options?.onWaitingForCallback) {
1057
+ options.onWaitingForCallback((callbackUrl: string) => {
1058
+ try {
1059
+ const parsed = new URL(callbackUrl)
1060
+ const code = parsed.searchParams.get('code')
1061
+ const state = parsed.searchParams.get('state')
1062
+ const error = parsed.searchParams.get('error')
1063
+
1064
+ if (error) {
1065
+ const errorDescription =
1066
+ parsed.searchParams.get('error_description') || ''
1067
+ cleanup()
1068
+ rejectOnce(
1069
+ new Error(`OAuth error: ${error} - ${errorDescription}`),
1070
+ )
1071
+ return
1072
+ }
1073
+
1074
+ if (!code) {
1075
+ // Not a valid callback URL, ignore so the user can try again
1076
+ return
1077
+ }
1078
+
1079
+ if (state !== oauthState) {
1080
+ cleanup()
1081
+ rejectOnce(
1082
+ new Error('OAuth state mismatch - possible CSRF attack'),
1083
+ )
1084
+ return
1085
+ }
1086
+
1087
+ logMCPDebug(
1088
+ serverName,
1089
+ `Received auth code via manual callback URL`,
1090
+ )
1091
+ cleanup()
1092
+ resolveOnce(code)
1093
+ } catch {
1094
+ // Invalid URL, ignore so the user can try again
1095
+ }
1096
+ })
1097
+ }
1098
+
1099
+ server = createServer((req, res) => {
1100
+ const parsedUrl = parse(req.url || '', true)
1101
+
1102
+ if (parsedUrl.pathname === '/callback') {
1103
+ const code = parsedUrl.query.code as string
1104
+ const state = parsedUrl.query.state as string
1105
+ const error = parsedUrl.query.error
1106
+ const errorDescription = parsedUrl.query.error_description as string
1107
+ const errorUri = parsedUrl.query.error_uri as string
1108
+
1109
+ // Validate OAuth state to prevent CSRF attacks
1110
+ if (!error && state !== oauthState) {
1111
+ res.writeHead(400, { 'Content-Type': 'text/html' })
1112
+ res.end(
1113
+ `<h1>Authentication Error</h1><p>Invalid state parameter. Please try again.</p><p>You can close this window.</p>`,
1114
+ )
1115
+ cleanup()
1116
+ rejectOnce(new Error('OAuth state mismatch - possible CSRF attack'))
1117
+ return
1118
+ }
1119
+
1120
+ if (error) {
1121
+ res.writeHead(200, { 'Content-Type': 'text/html' })
1122
+ // Sanitize error messages to prevent XSS
1123
+ const sanitizedError = xss(String(error))
1124
+ const sanitizedErrorDescription = errorDescription
1125
+ ? xss(String(errorDescription))
1126
+ : ''
1127
+ res.end(
1128
+ `<h1>Authentication Error</h1><p>${sanitizedError}: ${sanitizedErrorDescription}</p><p>You can close this window.</p>`,
1129
+ )
1130
+ cleanup()
1131
+ let errorMessage = `OAuth error: ${error}`
1132
+ if (errorDescription) {
1133
+ errorMessage += ` - ${errorDescription}`
1134
+ }
1135
+ if (errorUri) {
1136
+ errorMessage += ` (See: ${errorUri})`
1137
+ }
1138
+ rejectOnce(new Error(errorMessage))
1139
+ return
1140
+ }
1141
+
1142
+ if (code) {
1143
+ res.writeHead(200, { 'Content-Type': 'text/html' })
1144
+ res.end(
1145
+ `<h1>Authentication Successful</h1><p>You can close this window. Return to Claude Code.</p>`,
1146
+ )
1147
+ cleanup()
1148
+ resolveOnce(code)
1149
+ }
1150
+ }
1151
+ })
1152
+
1153
+ server.on('error', (err: NodeJS.ErrnoException) => {
1154
+ cleanup()
1155
+ if (err.code === 'EADDRINUSE') {
1156
+ const findCmd =
1157
+ getPlatform() === 'windows'
1158
+ ? `netstat -ano | findstr :${port}`
1159
+ : `lsof -ti:${port} -sTCP:LISTEN`
1160
+ rejectOnce(
1161
+ new Error(
1162
+ `OAuth callback port ${port} is already in use — another process may be holding it. ` +
1163
+ `Run \`${findCmd}\` to find it.`,
1164
+ ),
1165
+ )
1166
+ } else {
1167
+ rejectOnce(new Error(`OAuth callback server failed: ${err.message}`))
1168
+ }
1169
+ })
1170
+
1171
+ server.listen(port, '127.0.0.1', async () => {
1172
+ try {
1173
+ logMCPDebug(serverName, `Starting SDK auth`)
1174
+ logMCPDebug(serverName, `Server URL: ${serverConfig.url}`)
1175
+
1176
+ // First call to start the auth flow - should redirect
1177
+ // Pass the scope and resource_metadata from WWW-Authenticate header if available
1178
+ const result = await sdkAuth(provider, {
1179
+ serverUrl: serverConfig.url,
1180
+ scope: wwwAuthParams.scope,
1181
+ resourceMetadataUrl: wwwAuthParams.resourceMetadataUrl,
1182
+ })
1183
+ logMCPDebug(serverName, `Initial auth result: ${result}`)
1184
+
1185
+ if (result !== 'REDIRECT') {
1186
+ logMCPDebug(
1187
+ serverName,
1188
+ `Unexpected auth result, expected REDIRECT: ${result}`,
1189
+ )
1190
+ }
1191
+ } catch (error) {
1192
+ logMCPDebug(serverName, `SDK auth error: ${error}`)
1193
+ cleanup()
1194
+ rejectOnce(new Error(`SDK auth failed: ${errorMessage(error)}`))
1195
+ }
1196
+ })
1197
+
1198
+ // Don't let the callback server or timeout pin the event loop — if the UI
1199
+ // component unmounts without aborting (e.g. parent intercepts Esc), we'd
1200
+ // rather let the process exit than stay alive for 5 minutes holding the
1201
+ // port. The abortSignal is the intended lifecycle management.
1202
+ server.unref()
1203
+
1204
+ timeoutId = setTimeout(
1205
+ (cleanup, rejectOnce) => {
1206
+ cleanup()
1207
+ rejectOnce(new Error('Authentication timeout'))
1208
+ },
1209
+ 5 * 60 * 1000, // 5 minutes
1210
+ cleanup,
1211
+ rejectOnce,
1212
+ )
1213
+ timeoutId.unref()
1214
+ })
1215
+
1216
+ authorizationCodeObtained = true
1217
+
1218
+ // Now complete the auth flow with the received code
1219
+ logMCPDebug(serverName, `Completing auth flow with authorization code`)
1220
+ const result = await sdkAuth(provider, {
1221
+ serverUrl: serverConfig.url,
1222
+ authorizationCode,
1223
+ resourceMetadataUrl: wwwAuthParams.resourceMetadataUrl,
1224
+ })
1225
+
1226
+ logMCPDebug(serverName, `Auth result: ${result}`)
1227
+
1228
+ if (result === 'AUTHORIZED') {
1229
+ // Debug: Check if tokens were properly saved
1230
+ const savedTokens = await provider.tokens()
1231
+ logMCPDebug(
1232
+ serverName,
1233
+ `Tokens after auth: ${savedTokens ? 'Present' : 'Missing'}`,
1234
+ )
1235
+ if (savedTokens) {
1236
+ logMCPDebug(
1237
+ serverName,
1238
+ `Token access_token length: ${savedTokens.access_token?.length}`,
1239
+ )
1240
+ logMCPDebug(serverName, `Token expires_in: ${savedTokens.expires_in}`)
1241
+ }
1242
+
1243
+ logEvent('tengu_mcp_oauth_flow_success', {
1244
+ flowAttemptId:
1245
+ flowAttemptId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1246
+ transportType:
1247
+ serverConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1248
+ ...(getLoggingSafeMcpBaseUrl(serverConfig)
1249
+ ? {
1250
+ mcpServerBaseUrl: getLoggingSafeMcpBaseUrl(
1251
+ serverConfig,
1252
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1253
+ }
1254
+ : {}),
1255
+ })
1256
+ } else {
1257
+ throw new Error('Unexpected auth result: ' + result)
1258
+ }
1259
+ } catch (error) {
1260
+ logMCPDebug(serverName, `Error during auth completion: ${error}`)
1261
+
1262
+ // Determine failure reason for attribution telemetry. The try block covers
1263
+ // port acquisition, the callback server, the redirect flow, and token
1264
+ // exchange. Map known failure paths to stable reason codes.
1265
+ let reason: MCPOAuthFlowErrorReason = 'unknown'
1266
+ let oauthErrorCode: string | undefined
1267
+ let httpStatus: number | undefined
1268
+
1269
+ if (error instanceof AuthenticationCancelledError) {
1270
+ reason = 'cancelled'
1271
+ } else if (authorizationCodeObtained) {
1272
+ reason = 'token_exchange_failed'
1273
+ } else {
1274
+ const msg = errorMessage(error)
1275
+ if (msg.includes('Authentication timeout')) {
1276
+ reason = 'timeout'
1277
+ } else if (msg.includes('OAuth state mismatch')) {
1278
+ reason = 'state_mismatch'
1279
+ } else if (msg.includes('OAuth error:')) {
1280
+ reason = 'provider_denied'
1281
+ } else if (
1282
+ msg.includes('already in use') ||
1283
+ msg.includes('EADDRINUSE') ||
1284
+ msg.includes('callback server failed') ||
1285
+ msg.includes('No available port')
1286
+ ) {
1287
+ reason = 'port_unavailable'
1288
+ } else if (msg.includes('SDK auth failed')) {
1289
+ reason = 'sdk_auth_failed'
1290
+ }
1291
+ }
1292
+
1293
+ // sdkAuth uses native fetch and throws OAuthError subclasses (InvalidGrantError,
1294
+ // ServerError, InvalidClientError, etc.) via parseErrorResponse. Extract the
1295
+ // OAuth error code directly from the SDK error instance.
1296
+ if (error instanceof OAuthError) {
1297
+ oauthErrorCode = error.errorCode
1298
+ // SDK does not attach HTTP status as a property, but the fallback ServerError
1299
+ // embeds it in the message as "HTTP {status}:" when the response body was
1300
+ // unparseable. Best-effort extraction.
1301
+ const statusMatch = error.message.match(/^HTTP (\d{3}):/)
1302
+ if (statusMatch) {
1303
+ httpStatus = Number(statusMatch[1])
1304
+ }
1305
+ // If client not found, clear the stored client ID and suggest retry
1306
+ if (
1307
+ error.errorCode === 'invalid_client' &&
1308
+ error.message.includes('Client not found')
1309
+ ) {
1310
+ const storage = getSecureStorage()
1311
+ const existingData = storage.read() || {}
1312
+ const serverKey = getServerKey(serverName, serverConfig)
1313
+ if (existingData.mcpOAuth?.[serverKey]) {
1314
+ delete existingData.mcpOAuth[serverKey].clientId
1315
+ delete existingData.mcpOAuth[serverKey].clientSecret
1316
+ storage.update(existingData)
1317
+ }
1318
+ }
1319
+ }
1320
+
1321
+ logEvent('tengu_mcp_oauth_flow_error', {
1322
+ flowAttemptId:
1323
+ flowAttemptId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1324
+ reason:
1325
+ reason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1326
+ error_code:
1327
+ oauthErrorCode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1328
+ http_status:
1329
+ httpStatus?.toString() as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1330
+ transportType:
1331
+ serverConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1332
+ ...(getLoggingSafeMcpBaseUrl(serverConfig)
1333
+ ? {
1334
+ mcpServerBaseUrl: getLoggingSafeMcpBaseUrl(
1335
+ serverConfig,
1336
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1337
+ }
1338
+ : {}),
1339
+ })
1340
+ throw error
1341
+ }
1342
+ }
1343
+
1344
+ /**
1345
+ * Wraps fetch to detect 403 insufficient_scope responses and mark step-up
1346
+ * pending on the provider BEFORE the SDK's 403 handler calls auth(). Without
1347
+ * this, the SDK's authInternal sees refresh_token → refreshes (uselessly, since
1348
+ * RFC 6749 §6 forbids scope elevation via refresh) → returns 'AUTHORIZED' →
1349
+ * retry → 403 again → aborts with "Server returned 403 after trying upscoping",
1350
+ * never reaching redirectToAuthorization where step-up scope is persisted.
1351
+ * With this flag set, tokens() omits refresh_token so the SDK falls through
1352
+ * to the PKCE flow. See github.com/anthropics/claude-code/issues/28258.
1353
+ */
1354
+ export function wrapFetchWithStepUpDetection(
1355
+ baseFetch: FetchLike,
1356
+ provider: ClaudeAuthProvider,
1357
+ ): FetchLike {
1358
+ return async (url, init) => {
1359
+ const response = await baseFetch(url, init)
1360
+ if (response.status === 403) {
1361
+ const wwwAuth = response.headers.get('WWW-Authenticate')
1362
+ if (wwwAuth?.includes('insufficient_scope')) {
1363
+ // Match both quoted and unquoted values (RFC 6750 §3 allows either).
1364
+ // Same pattern as the SDK's extractFieldFromWwwAuth.
1365
+ const match = wwwAuth.match(/scope=(?:"([^"]+)"|([^\s,]+))/)
1366
+ const scope = match?.[1] ?? match?.[2]
1367
+ if (scope) {
1368
+ provider.markStepUpPending(scope)
1369
+ }
1370
+ }
1371
+ }
1372
+ return response
1373
+ }
1374
+ }
1375
+
1376
+ export class ClaudeAuthProvider implements OAuthClientProvider {
1377
+ private serverName: string
1378
+ private serverConfig: McpSSEServerConfig | McpHTTPServerConfig
1379
+ private redirectUri: string
1380
+ private handleRedirection: boolean
1381
+ private _codeVerifier?: string
1382
+ private _authorizationUrl?: string
1383
+ private _state?: string
1384
+ private _scopes?: string
1385
+ private _metadata?: Awaited<
1386
+ ReturnType<typeof discoverAuthorizationServerMetadata>
1387
+ >
1388
+ private _refreshInProgress?: Promise<OAuthTokens | undefined>
1389
+ private _pendingStepUpScope?: string
1390
+ private onAuthorizationUrlCallback?: (url: string) => void
1391
+ private skipBrowserOpen: boolean
1392
+
1393
+ constructor(
1394
+ serverName: string,
1395
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
1396
+ redirectUri: string = buildRedirectUri(),
1397
+ handleRedirection = false,
1398
+ onAuthorizationUrl?: (url: string) => void,
1399
+ skipBrowserOpen?: boolean,
1400
+ ) {
1401
+ this.serverName = serverName
1402
+ this.serverConfig = serverConfig
1403
+ this.redirectUri = redirectUri
1404
+ this.handleRedirection = handleRedirection
1405
+ this.onAuthorizationUrlCallback = onAuthorizationUrl
1406
+ this.skipBrowserOpen = skipBrowserOpen ?? false
1407
+ }
1408
+
1409
+ get redirectUrl(): string {
1410
+ return this.redirectUri
1411
+ }
1412
+
1413
+ get authorizationUrl(): string | undefined {
1414
+ return this._authorizationUrl
1415
+ }
1416
+
1417
+ get clientMetadata(): OAuthClientMetadata {
1418
+ const metadata: OAuthClientMetadata = {
1419
+ client_name: `Claude Code (${this.serverName})`,
1420
+ redirect_uris: [this.redirectUri],
1421
+ grant_types: ['authorization_code', 'refresh_token'],
1422
+ response_types: ['code'],
1423
+ token_endpoint_auth_method: 'none', // Public client
1424
+ }
1425
+
1426
+ // Include scope from metadata if available
1427
+ const metadataScope = getScopeFromMetadata(this._metadata)
1428
+ if (metadataScope) {
1429
+ metadata.scope = metadataScope
1430
+ logMCPDebug(
1431
+ this.serverName,
1432
+ `Using scope from metadata: ${metadata.scope}`,
1433
+ )
1434
+ }
1435
+
1436
+ return metadata
1437
+ }
1438
+
1439
+ /**
1440
+ * CIMD (SEP-991): URL-based client_id. When the auth server advertises
1441
+ * client_id_metadata_document_supported: true, the SDK uses this URL as the
1442
+ * client_id instead of performing Dynamic Client Registration.
1443
+ * Override via MCP_OAUTH_CLIENT_METADATA_URL env var (e.g. for testing, FedStart).
1444
+ */
1445
+ get clientMetadataUrl(): string | undefined {
1446
+ const override = process.env.MCP_OAUTH_CLIENT_METADATA_URL
1447
+ if (override) {
1448
+ logMCPDebug(this.serverName, `Using CIMD URL from env: ${override}`)
1449
+ return override
1450
+ }
1451
+ return MCP_CLIENT_METADATA_URL
1452
+ }
1453
+
1454
+ setMetadata(
1455
+ metadata: Awaited<ReturnType<typeof discoverAuthorizationServerMetadata>>,
1456
+ ): void {
1457
+ this._metadata = metadata
1458
+ }
1459
+
1460
+ /**
1461
+ * Called by the fetch wrapper when a 403 insufficient_scope response is
1462
+ * detected. Setting this causes tokens() to omit refresh_token, forcing
1463
+ * the SDK's authInternal to skip its (useless) refresh path and fall through
1464
+ * to startAuthorization → redirectToAuthorization → step-up persistence.
1465
+ * RFC 6749 §6 forbids scope elevation via refresh, so refreshing would just
1466
+ * return the same-scoped token and the retry would 403 again.
1467
+ */
1468
+ markStepUpPending(scope: string): void {
1469
+ this._pendingStepUpScope = scope
1470
+ logMCPDebug(this.serverName, `Marked step-up pending: ${scope}`)
1471
+ }
1472
+
1473
+ async state(): Promise<string> {
1474
+ // Generate state if not already generated for this instance
1475
+ if (!this._state) {
1476
+ this._state = randomBytes(32).toString('base64url')
1477
+ logMCPDebug(this.serverName, 'Generated new OAuth state')
1478
+ }
1479
+ return this._state
1480
+ }
1481
+
1482
+ async clientInformation(): Promise<OAuthClientInformation | undefined> {
1483
+ const storage = getSecureStorage()
1484
+ const data = storage.read()
1485
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1486
+
1487
+ // Check session credentials first (from DCR or previous auth)
1488
+ const storedInfo = data?.mcpOAuth?.[serverKey]
1489
+ if (storedInfo?.clientId) {
1490
+ logMCPDebug(this.serverName, `Found client info`)
1491
+ return {
1492
+ client_id: storedInfo.clientId,
1493
+ client_secret: storedInfo.clientSecret,
1494
+ }
1495
+ }
1496
+
1497
+ // Fallback: pre-configured client ID from server config
1498
+ const configClientId = this.serverConfig.oauth?.clientId
1499
+ if (configClientId) {
1500
+ const clientConfig = data?.mcpOAuthClientConfig?.[serverKey]
1501
+ logMCPDebug(this.serverName, `Using pre-configured client ID`)
1502
+ return {
1503
+ client_id: configClientId,
1504
+ client_secret: clientConfig?.clientSecret,
1505
+ }
1506
+ }
1507
+
1508
+ // If we don't have stored client info, return undefined to trigger registration
1509
+ logMCPDebug(this.serverName, `No client info found`)
1510
+ return undefined
1511
+ }
1512
+
1513
+ async saveClientInformation(
1514
+ clientInformation: OAuthClientInformationFull,
1515
+ ): Promise<void> {
1516
+ const storage = getSecureStorage()
1517
+ const existingData = storage.read() || {}
1518
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1519
+
1520
+ const updatedData: SecureStorageData = {
1521
+ ...existingData,
1522
+ mcpOAuth: {
1523
+ ...existingData.mcpOAuth,
1524
+ [serverKey]: {
1525
+ ...existingData.mcpOAuth?.[serverKey],
1526
+ serverName: this.serverName,
1527
+ serverUrl: this.serverConfig.url,
1528
+ clientId: clientInformation.client_id,
1529
+ clientSecret: clientInformation.client_secret,
1530
+ // Provide default values for required fields if not present
1531
+ accessToken: existingData.mcpOAuth?.[serverKey]?.accessToken || '',
1532
+ expiresAt: existingData.mcpOAuth?.[serverKey]?.expiresAt || 0,
1533
+ },
1534
+ },
1535
+ }
1536
+
1537
+ storage.update(updatedData)
1538
+ }
1539
+
1540
+ async tokens(): Promise<OAuthTokens | undefined> {
1541
+ // Cross-process token changes (another CC instance refreshed or invalidated)
1542
+ // are picked up via the keychain cache TTL (see macOsKeychainStorage.ts).
1543
+ // In-process writes already invalidate the cache via storage.update().
1544
+ // We do NOT clearKeychainCache() here — tokens() is called by the MCP SDK's
1545
+ // _commonHeaders on every request, and forcing a cache miss would trigger
1546
+ // a blocking spawnSync(`security find-generic-password`) 30-40x/sec.
1547
+ // See CPU profile: spawnSync was 7.2% of total CPU after PR #19436.
1548
+ const storage = getSecureStorage()
1549
+ const data = await storage.readAsync()
1550
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1551
+
1552
+ const tokenData = data?.mcpOAuth?.[serverKey]
1553
+
1554
+ // XAA: a cached id_token plays the same UX role as a refresh_token — run
1555
+ // the silent exchange to get a fresh access_token without a browser. The
1556
+ // id_token does expire (we re-acquire via `xaa login` when it does); the
1557
+ // point is that while it's valid, re-auth is zero-interaction.
1558
+ //
1559
+ // Only fire when we don't have a refresh_token. If the AS returned one,
1560
+ // the normal refresh path (below) is cheaper — 1 request vs the 4-request
1561
+ // XAA chain. If that refresh is revoked, refreshAuthorization() clears it
1562
+ // (invalidateCredentials('tokens')), and the next tokens() falls through
1563
+ // to here.
1564
+ //
1565
+ // Fires on:
1566
+ // - never authed (!tokenData) → first connect, auto-auth
1567
+ // - SDK partial write {accessToken:''} → stale from past session
1568
+ // - expired/expiring, no refresh_token → proactive XAA re-auth
1569
+ //
1570
+ // No special-casing of {accessToken:'', expiresAt:0}. Yes, SDK auth()
1571
+ // writes that mid-flow (saveClientInformation defaults). But with this
1572
+ // auto-auth branch, the *first* tokens() call — before auth() writes
1573
+ // anything — fires xaaRefresh. If id_token is cached, SDK short-circuits
1574
+ // there and never reaches the write. If id_token isn't cached, xaaRefresh
1575
+ // returns undefined in ~1 keychain read, auth() proceeds, writes the
1576
+ // marker, calls tokens() again, xaaRefresh fails again identically.
1577
+ // Harmless redundancy, not a wasted exchange. And guarding on `!==''`
1578
+ // permanently bricks auto-auth when a *prior* session left that marker
1579
+ // in keychain — real bug seen with xaa.dev.
1580
+ //
1581
+ // xaaRefresh() internally short-circuits to undefined when the id_token
1582
+ // isn't cached (or settings.xaaIdp is gone) → we fall through to the
1583
+ // existing needs-auth path → user runs `xaa login`.
1584
+ //
1585
+ if (
1586
+ isXaaEnabled() &&
1587
+ this.serverConfig.oauth?.xaa &&
1588
+ !tokenData?.refreshToken &&
1589
+ (!tokenData?.accessToken ||
1590
+ (tokenData.expiresAt - Date.now()) / 1000 <= 300)
1591
+ ) {
1592
+ if (!this._refreshInProgress) {
1593
+ logMCPDebug(
1594
+ this.serverName,
1595
+ tokenData
1596
+ ? `XAA: access_token expiring, attempting silent exchange`
1597
+ : `XAA: no access_token yet, attempting silent exchange`,
1598
+ )
1599
+ this._refreshInProgress = this.xaaRefresh().finally(() => {
1600
+ this._refreshInProgress = undefined
1601
+ })
1602
+ }
1603
+ try {
1604
+ const refreshed = await this._refreshInProgress
1605
+ if (refreshed) return refreshed
1606
+ } catch (e) {
1607
+ logMCPDebug(
1608
+ this.serverName,
1609
+ `XAA silent exchange failed: ${errorMessage(e)}`,
1610
+ )
1611
+ }
1612
+ // Fall through. Either id_token isn't cached (xaaRefresh returned
1613
+ // undefined) or the exchange errored. Normal path below handles both:
1614
+ // !tokenData → undefined → 401 → needs-auth; expired → undefined → same.
1615
+ }
1616
+
1617
+ if (!tokenData) {
1618
+ logMCPDebug(this.serverName, `No token data found`)
1619
+ return undefined
1620
+ }
1621
+
1622
+ // Check if token is expired
1623
+ const expiresIn = (tokenData.expiresAt - Date.now()) / 1000
1624
+
1625
+ // Step-up check: if a 403 insufficient_scope was detected and the current
1626
+ // token doesn't have the requested scope, omit refresh_token below so the
1627
+ // SDK skips refresh and falls through to the PKCE flow.
1628
+ const currentScopes = tokenData.scope?.split(' ') ?? []
1629
+ const needsStepUp =
1630
+ this._pendingStepUpScope !== undefined &&
1631
+ this._pendingStepUpScope.split(' ').some(s => !currentScopes.includes(s))
1632
+ if (needsStepUp) {
1633
+ logMCPDebug(
1634
+ this.serverName,
1635
+ `Step-up pending (${this._pendingStepUpScope}), omitting refresh_token`,
1636
+ )
1637
+ }
1638
+
1639
+ // If token is expired and we don't have a refresh token, return undefined
1640
+ if (expiresIn <= 0 && !tokenData.refreshToken) {
1641
+ logMCPDebug(this.serverName, `Token expired without refresh token`)
1642
+ return undefined
1643
+ }
1644
+
1645
+ // If token is expired or about to expire (within 5 minutes) and we have a refresh token, refresh it proactively.
1646
+ // This proactive refresh is a UX improvement - it avoids the latency of a failed request followed by token refresh.
1647
+ // While MCP servers should return 401 for expired tokens (which triggers SDK-level refresh), proactively refreshing
1648
+ // before expiry provides a smoother user experience.
1649
+ // Skip when step-up is pending — refreshing can't elevate scope (RFC 6749 §6).
1650
+ if (expiresIn <= 300 && tokenData.refreshToken && !needsStepUp) {
1651
+ // Reuse existing refresh promise if one is in progress to prevent concurrent refreshes
1652
+ if (!this._refreshInProgress) {
1653
+ logMCPDebug(
1654
+ this.serverName,
1655
+ `Token expires in ${Math.floor(expiresIn)}s, attempting proactive refresh`,
1656
+ )
1657
+ this._refreshInProgress = this.refreshAuthorization(
1658
+ tokenData.refreshToken,
1659
+ ).finally(() => {
1660
+ this._refreshInProgress = undefined
1661
+ })
1662
+ } else {
1663
+ logMCPDebug(
1664
+ this.serverName,
1665
+ `Token refresh already in progress, reusing existing promise`,
1666
+ )
1667
+ }
1668
+
1669
+ try {
1670
+ const refreshed = await this._refreshInProgress
1671
+ if (refreshed) {
1672
+ logMCPDebug(this.serverName, `Token refreshed successfully`)
1673
+ return refreshed
1674
+ }
1675
+ logMCPDebug(
1676
+ this.serverName,
1677
+ `Token refresh failed, returning current tokens`,
1678
+ )
1679
+ } catch (error) {
1680
+ logMCPDebug(
1681
+ this.serverName,
1682
+ `Token refresh error: ${errorMessage(error)}`,
1683
+ )
1684
+ }
1685
+ }
1686
+
1687
+ // Return current tokens (may be expired if refresh failed or not needed yet)
1688
+ const tokens = {
1689
+ access_token: tokenData.accessToken,
1690
+ refresh_token: needsStepUp ? undefined : tokenData.refreshToken,
1691
+ expires_in: expiresIn,
1692
+ scope: tokenData.scope,
1693
+ token_type: 'Bearer',
1694
+ }
1695
+
1696
+ logMCPDebug(this.serverName, `Returning tokens`)
1697
+ logMCPDebug(this.serverName, `Token length: ${tokens.access_token?.length}`)
1698
+ logMCPDebug(this.serverName, `Has refresh token: ${!!tokens.refresh_token}`)
1699
+ logMCPDebug(this.serverName, `Expires in: ${Math.floor(expiresIn)}s`)
1700
+
1701
+ return tokens
1702
+ }
1703
+
1704
+ async saveTokens(tokens: OAuthTokens): Promise<void> {
1705
+ this._pendingStepUpScope = undefined
1706
+ const storage = getSecureStorage()
1707
+ const existingData = storage.read() || {}
1708
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1709
+
1710
+ logMCPDebug(this.serverName, `Saving tokens`)
1711
+ logMCPDebug(this.serverName, `Token expires in: ${tokens.expires_in}`)
1712
+ logMCPDebug(this.serverName, `Has refresh token: ${!!tokens.refresh_token}`)
1713
+
1714
+ const updatedData: SecureStorageData = {
1715
+ ...existingData,
1716
+ mcpOAuth: {
1717
+ ...existingData.mcpOAuth,
1718
+ [serverKey]: {
1719
+ ...existingData.mcpOAuth?.[serverKey],
1720
+ serverName: this.serverName,
1721
+ serverUrl: this.serverConfig.url,
1722
+ accessToken: tokens.access_token,
1723
+ refreshToken: tokens.refresh_token,
1724
+ expiresAt: Date.now() + (tokens.expires_in || 3600) * 1000,
1725
+ scope: tokens.scope,
1726
+ },
1727
+ },
1728
+ }
1729
+
1730
+ storage.update(updatedData)
1731
+ }
1732
+
1733
+ /**
1734
+ * XAA silent refresh: cached id_token → Layer-2 exchange → new access_token.
1735
+ * No browser.
1736
+ *
1737
+ * Returns undefined if the id_token is gone from cache — caller treats this
1738
+ * as needs-interactive-reauth (transport will 401, CC surfaces it).
1739
+ *
1740
+ * On exchange failure, clears the id_token cache so the next interactive
1741
+ * auth does a fresh IdP login (the cached id_token is likely stale/revoked).
1742
+ *
1743
+ * TODO(xaa-ga): add cross-process lockfile before GA. `_refreshInProgress`
1744
+ * only dedupes within one process — two CC instances with expiring tokens
1745
+ * both fire the full 4-request XAA chain and race on storage.update().
1746
+ * Unlike inc-4829 the id_token is not single-use so both access_tokens
1747
+ * stay valid (wasted round-trips + keychain write race, not brickage),
1748
+ * but this is the shape CLAUDE.md flags under "Token/auth caching across
1749
+ * process boundaries". Mirror refreshAuthorization()'s lockfile pattern.
1750
+ */
1751
+ private async xaaRefresh(): Promise<OAuthTokens | undefined> {
1752
+ const idp = getXaaIdpSettings()
1753
+ if (!idp) return undefined // config was removed mid-session
1754
+
1755
+ const idToken = getCachedIdpIdToken(idp.issuer)
1756
+ if (!idToken) {
1757
+ logMCPDebug(
1758
+ this.serverName,
1759
+ 'XAA: id_token not cached, needs interactive re-auth',
1760
+ )
1761
+ return undefined
1762
+ }
1763
+
1764
+ const clientId = this.serverConfig.oauth?.clientId
1765
+ const clientConfig = getMcpClientConfig(this.serverName, this.serverConfig)
1766
+ if (!clientId || !clientConfig?.clientSecret) {
1767
+ logMCPDebug(
1768
+ this.serverName,
1769
+ 'XAA: missing clientId or clientSecret in config — skipping silent refresh',
1770
+ )
1771
+ return undefined // shouldn't happen if `mcp add` was correct
1772
+ }
1773
+
1774
+ const idpClientSecret = getIdpClientSecret(idp.issuer)
1775
+
1776
+ // Discover IdP token endpoint. Could cache (fetchCache.ts already
1777
+ // caches /.well-known/ requests), but OIDC metadata is cheap + idempotent.
1778
+ // xaaRefresh is the silent tokens() path — soft-fail to undefined so the
1779
+ // caller falls through to needs-authentication instead of throwing mid-connect.
1780
+ let oidc
1781
+ try {
1782
+ oidc = await discoverOidc(idp.issuer)
1783
+ } catch (e) {
1784
+ logMCPDebug(
1785
+ this.serverName,
1786
+ `XAA: OIDC discovery failed in silent refresh: ${errorMessage(e)}`,
1787
+ )
1788
+ return undefined
1789
+ }
1790
+
1791
+ try {
1792
+ const tokens = await performCrossAppAccess(
1793
+ this.serverConfig.url,
1794
+ {
1795
+ clientId,
1796
+ clientSecret: clientConfig.clientSecret,
1797
+ idpClientId: idp.clientId,
1798
+ idpClientSecret,
1799
+ idpIdToken: idToken,
1800
+ idpTokenEndpoint: oidc.token_endpoint,
1801
+ },
1802
+ this.serverName,
1803
+ )
1804
+ // Write directly (not via saveTokens) so clientId + clientSecret land in
1805
+ // storage even when this is the first write for serverKey. saveTokens
1806
+ // only spreads existing data; if no prior performMCPXaaAuth ran,
1807
+ // revokeServerTokens would later read tokenData.clientId as undefined
1808
+ // and send a client_id-less RFC 7009 request that strict ASes reject.
1809
+ const storage = getSecureStorage()
1810
+ const existingData = storage.read() || {}
1811
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1812
+ const prev = existingData.mcpOAuth?.[serverKey]
1813
+ storage.update({
1814
+ ...existingData,
1815
+ mcpOAuth: {
1816
+ ...existingData.mcpOAuth,
1817
+ [serverKey]: {
1818
+ ...prev,
1819
+ serverName: this.serverName,
1820
+ serverUrl: this.serverConfig.url,
1821
+ accessToken: tokens.access_token,
1822
+ refreshToken: tokens.refresh_token ?? prev?.refreshToken,
1823
+ expiresAt: Date.now() + (tokens.expires_in || 3600) * 1000,
1824
+ scope: tokens.scope,
1825
+ clientId,
1826
+ clientSecret: clientConfig.clientSecret,
1827
+ discoveryState: {
1828
+ authorizationServerUrl: tokens.authorizationServerUrl,
1829
+ },
1830
+ },
1831
+ },
1832
+ })
1833
+ return {
1834
+ access_token: tokens.access_token,
1835
+ token_type: 'Bearer',
1836
+ expires_in: tokens.expires_in,
1837
+ scope: tokens.scope,
1838
+ refresh_token: tokens.refresh_token,
1839
+ }
1840
+ } catch (e) {
1841
+ if (e instanceof XaaTokenExchangeError && e.shouldClearIdToken) {
1842
+ clearIdpIdToken(idp.issuer)
1843
+ logMCPDebug(
1844
+ this.serverName,
1845
+ 'XAA: cleared id_token after exchange failure',
1846
+ )
1847
+ }
1848
+ throw e
1849
+ }
1850
+ }
1851
+
1852
+ async redirectToAuthorization(authorizationUrl: URL): Promise<void> {
1853
+ // Store the authorization URL
1854
+ this._authorizationUrl = authorizationUrl.toString()
1855
+
1856
+ // Extract and store scopes from the authorization URL for later use in token exchange
1857
+ const scopes = authorizationUrl.searchParams.get('scope')
1858
+ logMCPDebug(
1859
+ this.serverName,
1860
+ `Authorization URL: ${redactSensitiveUrlParams(authorizationUrl.toString())}`,
1861
+ )
1862
+ logMCPDebug(this.serverName, `Scopes in URL: ${scopes || 'NOT FOUND'}`)
1863
+
1864
+ if (scopes) {
1865
+ this._scopes = scopes
1866
+ logMCPDebug(
1867
+ this.serverName,
1868
+ `Captured scopes from authorization URL: ${scopes}`,
1869
+ )
1870
+ } else {
1871
+ // If no scope in URL, try to get it from metadata
1872
+ const metadataScope = getScopeFromMetadata(this._metadata)
1873
+ if (metadataScope) {
1874
+ this._scopes = metadataScope
1875
+ logMCPDebug(
1876
+ this.serverName,
1877
+ `Using scopes from metadata: ${metadataScope}`,
1878
+ )
1879
+ } else {
1880
+ logMCPDebug(this.serverName, `No scopes available from URL or metadata`)
1881
+ }
1882
+ }
1883
+
1884
+ // Persist scope for step-up auth: only when the transport-attached provider
1885
+ // (handleRedirection=false) receives a step-up 401. The SDK calls auth()
1886
+ // which calls redirectToAuthorization with the new scope. We persist it
1887
+ // so the next performMCPOAuthFlow can use it without an extra probe request.
1888
+ // Guard with !handleRedirection to avoid persisting during normal auth flows
1889
+ // (where the scope may come from metadata scopes_supported rather than a 401).
1890
+ if (this._scopes && !this.handleRedirection) {
1891
+ const storage = getSecureStorage()
1892
+ const existingData = storage.read() || {}
1893
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1894
+ const existing = existingData.mcpOAuth?.[serverKey]
1895
+ if (existing) {
1896
+ existing.stepUpScope = this._scopes
1897
+ storage.update(existingData)
1898
+ logMCPDebug(this.serverName, `Persisted step-up scope: ${this._scopes}`)
1899
+ }
1900
+ }
1901
+
1902
+ if (!this.handleRedirection) {
1903
+ logMCPDebug(
1904
+ this.serverName,
1905
+ `Redirection handling is disabled, skipping redirect`,
1906
+ )
1907
+ return
1908
+ }
1909
+
1910
+ // Validate URL scheme for security
1911
+ const urlString = authorizationUrl.toString()
1912
+ if (!urlString.startsWith('http://') && !urlString.startsWith('https://')) {
1913
+ throw new Error(
1914
+ 'Invalid authorization URL: must use http:// or https:// scheme',
1915
+ )
1916
+ }
1917
+
1918
+ logMCPDebug(this.serverName, `Redirecting to authorization URL`)
1919
+ const redactedUrl = redactSensitiveUrlParams(urlString)
1920
+ logMCPDebug(this.serverName, `Authorization URL: ${redactedUrl}`)
1921
+
1922
+ // Notify the UI about the authorization URL BEFORE opening the browser,
1923
+ // so users can see the URL as a fallback if the browser fails to open
1924
+ if (this.onAuthorizationUrlCallback) {
1925
+ this.onAuthorizationUrlCallback(urlString)
1926
+ }
1927
+
1928
+ if (!this.skipBrowserOpen) {
1929
+ logMCPDebug(this.serverName, `Opening authorization URL: ${redactedUrl}`)
1930
+
1931
+ const success = await openBrowser(urlString)
1932
+ if (!success) {
1933
+ logMCPDebug(
1934
+ this.serverName,
1935
+ `Browser didn't open automatically. URL is shown in UI.`,
1936
+ )
1937
+ }
1938
+ } else {
1939
+ logMCPDebug(
1940
+ this.serverName,
1941
+ `Skipping browser open (skipBrowserOpen=true). URL: ${redactedUrl}`,
1942
+ )
1943
+ }
1944
+ }
1945
+
1946
+ async saveCodeVerifier(codeVerifier: string): Promise<void> {
1947
+ logMCPDebug(this.serverName, `Saving code verifier`)
1948
+ this._codeVerifier = codeVerifier
1949
+ }
1950
+
1951
+ async codeVerifier(): Promise<string> {
1952
+ if (!this._codeVerifier) {
1953
+ logMCPDebug(this.serverName, `No code verifier saved`)
1954
+ throw new Error('No code verifier saved')
1955
+ }
1956
+ logMCPDebug(this.serverName, `Returning code verifier`)
1957
+ return this._codeVerifier
1958
+ }
1959
+
1960
+ async invalidateCredentials(
1961
+ scope: 'all' | 'client' | 'tokens' | 'verifier' | 'discovery',
1962
+ ): Promise<void> {
1963
+ const storage = getSecureStorage()
1964
+ const existingData = storage.read()
1965
+ if (!existingData?.mcpOAuth) return
1966
+
1967
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
1968
+ const tokenData = existingData.mcpOAuth[serverKey]
1969
+ if (!tokenData) return
1970
+
1971
+ switch (scope) {
1972
+ case 'all':
1973
+ delete existingData.mcpOAuth[serverKey]
1974
+ break
1975
+ case 'client':
1976
+ tokenData.clientId = undefined
1977
+ tokenData.clientSecret = undefined
1978
+ break
1979
+ case 'tokens':
1980
+ tokenData.accessToken = ''
1981
+ tokenData.refreshToken = undefined
1982
+ tokenData.expiresAt = 0
1983
+ break
1984
+ case 'verifier':
1985
+ this._codeVerifier = undefined
1986
+ return
1987
+ case 'discovery':
1988
+ tokenData.discoveryState = undefined
1989
+ tokenData.stepUpScope = undefined
1990
+ break
1991
+ }
1992
+
1993
+ storage.update(existingData)
1994
+ logMCPDebug(this.serverName, `Invalidated credentials (scope: ${scope})`)
1995
+ }
1996
+
1997
+ async saveDiscoveryState(state: OAuthDiscoveryState): Promise<void> {
1998
+ const storage = getSecureStorage()
1999
+ const existingData = storage.read() || {}
2000
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
2001
+
2002
+ logMCPDebug(
2003
+ this.serverName,
2004
+ `Saving discovery state (authServer: ${state.authorizationServerUrl})`,
2005
+ )
2006
+
2007
+ // Persist only the URLs, NOT the full metadata blobs.
2008
+ // authorizationServerMetadata alone is ~1.5-2KB per MCP server (every
2009
+ // grant type, PKCE method, endpoint the IdP supports). On macOS the
2010
+ // keychain write goes through `security -i` which has a 4096-byte stdin
2011
+ // line limit — with hex encoding that's ~2013 bytes of JSON total. Two
2012
+ // OAuth MCP servers persisting full metadata overflows it, corrupting
2013
+ // the credential store (#30337). The SDK re-fetches missing metadata
2014
+ // with one HTTP GET on the next auth — see node_modules/.../auth.js
2015
+ // `cachedState.authorizationServerMetadata ?? await discover...`.
2016
+ const updatedData: SecureStorageData = {
2017
+ ...existingData,
2018
+ mcpOAuth: {
2019
+ ...existingData.mcpOAuth,
2020
+ [serverKey]: {
2021
+ ...existingData.mcpOAuth?.[serverKey],
2022
+ serverName: this.serverName,
2023
+ serverUrl: this.serverConfig.url,
2024
+ accessToken: existingData.mcpOAuth?.[serverKey]?.accessToken || '',
2025
+ expiresAt: existingData.mcpOAuth?.[serverKey]?.expiresAt || 0,
2026
+ discoveryState: {
2027
+ authorizationServerUrl: state.authorizationServerUrl,
2028
+ resourceMetadataUrl: state.resourceMetadataUrl,
2029
+ },
2030
+ },
2031
+ },
2032
+ }
2033
+
2034
+ storage.update(updatedData)
2035
+ }
2036
+
2037
+ async discoveryState(): Promise<OAuthDiscoveryState | undefined> {
2038
+ const storage = getSecureStorage()
2039
+ const data = storage.read()
2040
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
2041
+
2042
+ const cached = data?.mcpOAuth?.[serverKey]?.discoveryState
2043
+ if (cached?.authorizationServerUrl) {
2044
+ logMCPDebug(
2045
+ this.serverName,
2046
+ `Returning cached discovery state (authServer: ${cached.authorizationServerUrl})`,
2047
+ )
2048
+
2049
+ return {
2050
+ authorizationServerUrl: cached.authorizationServerUrl,
2051
+ resourceMetadataUrl: cached.resourceMetadataUrl,
2052
+ resourceMetadata:
2053
+ cached.resourceMetadata as OAuthDiscoveryState['resourceMetadata'],
2054
+ authorizationServerMetadata:
2055
+ cached.authorizationServerMetadata as OAuthDiscoveryState['authorizationServerMetadata'],
2056
+ }
2057
+ }
2058
+
2059
+ // Check config hint for direct metadata URL
2060
+ const metadataUrl = this.serverConfig.oauth?.authServerMetadataUrl
2061
+ if (metadataUrl) {
2062
+ logMCPDebug(
2063
+ this.serverName,
2064
+ `Fetching metadata from configured URL: ${metadataUrl}`,
2065
+ )
2066
+ try {
2067
+ const metadata = await fetchAuthServerMetadata(
2068
+ this.serverName,
2069
+ this.serverConfig.url,
2070
+ metadataUrl,
2071
+ )
2072
+ if (metadata) {
2073
+ return {
2074
+ authorizationServerUrl: metadata.issuer,
2075
+ authorizationServerMetadata:
2076
+ metadata as OAuthDiscoveryState['authorizationServerMetadata'],
2077
+ }
2078
+ }
2079
+ } catch (error) {
2080
+ logMCPDebug(
2081
+ this.serverName,
2082
+ `Failed to fetch from configured metadata URL: ${errorMessage(error)}`,
2083
+ )
2084
+ }
2085
+ }
2086
+
2087
+ return undefined
2088
+ }
2089
+
2090
+ async refreshAuthorization(
2091
+ refreshToken: string,
2092
+ ): Promise<OAuthTokens | undefined> {
2093
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
2094
+ const claudeDir = getClaudeConfigHomeDir()
2095
+ await mkdir(claudeDir, { recursive: true })
2096
+ const sanitizedKey = serverKey.replace(/[^a-zA-Z0-9]/g, '_')
2097
+ const lockfilePath = join(claudeDir, `mcp-refresh-${sanitizedKey}.lock`)
2098
+
2099
+ let release: (() => Promise<void>) | undefined
2100
+ for (let retry = 0; retry < MAX_LOCK_RETRIES; retry++) {
2101
+ try {
2102
+ logMCPDebug(
2103
+ this.serverName,
2104
+ `Acquiring refresh lock (attempt ${retry + 1})`,
2105
+ )
2106
+ release = await lockfile.lock(lockfilePath, {
2107
+ realpath: false,
2108
+ onCompromised: () => {
2109
+ logMCPDebug(this.serverName, `Refresh lock was compromised`)
2110
+ },
2111
+ })
2112
+ logMCPDebug(this.serverName, `Acquired refresh lock`)
2113
+ break
2114
+ } catch (e: unknown) {
2115
+ const code = getErrnoCode(e)
2116
+ if (code === 'ELOCKED') {
2117
+ logMCPDebug(
2118
+ this.serverName,
2119
+ `Refresh lock held by another process, waiting (attempt ${retry + 1}/${MAX_LOCK_RETRIES})`,
2120
+ )
2121
+ await sleep(1000 + Math.random() * 1000)
2122
+ continue
2123
+ }
2124
+ logMCPDebug(
2125
+ this.serverName,
2126
+ `Failed to acquire refresh lock: ${code}, proceeding without lock`,
2127
+ )
2128
+ break
2129
+ }
2130
+ }
2131
+ if (!release) {
2132
+ logMCPDebug(
2133
+ this.serverName,
2134
+ `Could not acquire refresh lock after ${MAX_LOCK_RETRIES} retries, proceeding without lock`,
2135
+ )
2136
+ }
2137
+
2138
+ try {
2139
+ // Re-read tokens after acquiring lock — another process may have refreshed
2140
+ clearKeychainCache()
2141
+ const storage = getSecureStorage()
2142
+ const data = storage.read()
2143
+ const tokenData = data?.mcpOAuth?.[serverKey]
2144
+ if (tokenData) {
2145
+ const expiresIn = (tokenData.expiresAt - Date.now()) / 1000
2146
+ if (expiresIn > 300) {
2147
+ logMCPDebug(
2148
+ this.serverName,
2149
+ `Another process already refreshed tokens (expires in ${Math.floor(expiresIn)}s)`,
2150
+ )
2151
+ return {
2152
+ access_token: tokenData.accessToken,
2153
+ refresh_token: tokenData.refreshToken,
2154
+ expires_in: expiresIn,
2155
+ scope: tokenData.scope,
2156
+ token_type: 'Bearer',
2157
+ }
2158
+ }
2159
+ // Use the freshest refresh token from storage
2160
+ if (tokenData.refreshToken) {
2161
+ refreshToken = tokenData.refreshToken
2162
+ }
2163
+ }
2164
+ return await this._doRefresh(refreshToken)
2165
+ } finally {
2166
+ if (release) {
2167
+ try {
2168
+ await release()
2169
+ logMCPDebug(this.serverName, `Released refresh lock`)
2170
+ } catch {
2171
+ logMCPDebug(this.serverName, `Failed to release refresh lock`)
2172
+ }
2173
+ }
2174
+ }
2175
+ }
2176
+
2177
+ private async _doRefresh(
2178
+ refreshToken: string,
2179
+ ): Promise<OAuthTokens | undefined> {
2180
+ const MAX_ATTEMPTS = 3
2181
+
2182
+ const mcpServerBaseUrl = getLoggingSafeMcpBaseUrl(this.serverConfig)
2183
+ const emitRefreshEvent = (
2184
+ outcome: 'success' | 'failure',
2185
+ reason?: MCPRefreshFailureReason,
2186
+ ): void => {
2187
+ logEvent(
2188
+ outcome === 'success'
2189
+ ? 'tengu_mcp_oauth_refresh_success'
2190
+ : 'tengu_mcp_oauth_refresh_failure',
2191
+ {
2192
+ transportType: this.serverConfig
2193
+ .type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2194
+ ...(mcpServerBaseUrl
2195
+ ? {
2196
+ mcpServerBaseUrl:
2197
+ mcpServerBaseUrl as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2198
+ }
2199
+ : {}),
2200
+ ...(reason
2201
+ ? {
2202
+ reason:
2203
+ reason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2204
+ }
2205
+ : {}),
2206
+ },
2207
+ )
2208
+ }
2209
+
2210
+ for (let attempt = 1; attempt <= MAX_ATTEMPTS; attempt++) {
2211
+ try {
2212
+ logMCPDebug(this.serverName, `Starting token refresh`)
2213
+ const authFetch = createAuthFetch()
2214
+
2215
+ // Reuse cached metadata from the initial OAuth flow if available,
2216
+ // since metadata (token endpoint URL, etc.) is static per auth server.
2217
+ // Priority:
2218
+ // 1. In-memory cache (same-session refreshes)
2219
+ // 2. Persisted discovery state from initial auth (cross-session) —
2220
+ // avoids re-running RFC 9728 discovery on every refresh.
2221
+ // 3. Full RFC 9728 → RFC 8414 re-discovery via fetchAuthServerMetadata.
2222
+ let metadata = this._metadata
2223
+ if (!metadata) {
2224
+ const cached = await this.discoveryState()
2225
+ if (cached?.authorizationServerMetadata) {
2226
+ logMCPDebug(
2227
+ this.serverName,
2228
+ `Using persisted auth server metadata for refresh`,
2229
+ )
2230
+ metadata = cached.authorizationServerMetadata
2231
+ } else if (cached?.authorizationServerUrl) {
2232
+ logMCPDebug(
2233
+ this.serverName,
2234
+ `Re-discovering metadata from persisted auth server URL: ${cached.authorizationServerUrl}`,
2235
+ )
2236
+ metadata = await discoverAuthorizationServerMetadata(
2237
+ cached.authorizationServerUrl,
2238
+ { fetchFn: authFetch },
2239
+ )
2240
+ }
2241
+ }
2242
+ if (!metadata) {
2243
+ metadata = await fetchAuthServerMetadata(
2244
+ this.serverName,
2245
+ this.serverConfig.url,
2246
+ this.serverConfig.oauth?.authServerMetadataUrl,
2247
+ authFetch,
2248
+ )
2249
+ }
2250
+ if (!metadata) {
2251
+ logMCPDebug(this.serverName, `Failed to discover OAuth metadata`)
2252
+ emitRefreshEvent('failure', 'metadata_discovery_failed')
2253
+ return undefined
2254
+ }
2255
+ // Cache for future refreshes
2256
+ this._metadata = metadata
2257
+
2258
+ const clientInfo = await this.clientInformation()
2259
+ if (!clientInfo) {
2260
+ logMCPDebug(this.serverName, `No client information available`)
2261
+ emitRefreshEvent('failure', 'no_client_info')
2262
+ return undefined
2263
+ }
2264
+
2265
+ const newTokens = await sdkRefreshAuthorization(
2266
+ new URL(this.serverConfig.url),
2267
+ {
2268
+ metadata,
2269
+ clientInformation: clientInfo,
2270
+ refreshToken,
2271
+ resource: new URL(this.serverConfig.url),
2272
+ fetchFn: authFetch,
2273
+ },
2274
+ )
2275
+
2276
+ if (newTokens) {
2277
+ logMCPDebug(this.serverName, `Token refresh successful`)
2278
+ await this.saveTokens(newTokens)
2279
+ emitRefreshEvent('success')
2280
+ return newTokens
2281
+ }
2282
+
2283
+ logMCPDebug(this.serverName, `Token refresh returned no tokens`)
2284
+ emitRefreshEvent('failure', 'no_tokens_returned')
2285
+ return undefined
2286
+ } catch (error) {
2287
+ // Invalid grant means the refresh token itself is invalid/revoked/expired.
2288
+ // But another process may have already refreshed successfully — check first.
2289
+ if (error instanceof InvalidGrantError) {
2290
+ logMCPDebug(
2291
+ this.serverName,
2292
+ `Token refresh failed with invalid_grant: ${error.message}`,
2293
+ )
2294
+ clearKeychainCache()
2295
+ const storage = getSecureStorage()
2296
+ const data = storage.read()
2297
+ const serverKey = getServerKey(this.serverName, this.serverConfig)
2298
+ const tokenData = data?.mcpOAuth?.[serverKey]
2299
+ if (tokenData) {
2300
+ const expiresIn = (tokenData.expiresAt - Date.now()) / 1000
2301
+ if (expiresIn > 300) {
2302
+ logMCPDebug(
2303
+ this.serverName,
2304
+ `Another process refreshed tokens, using those`,
2305
+ )
2306
+ // Not emitted as success: this process did not perform a
2307
+ // refresh, and the winning process already emitted its own
2308
+ // success event. Emitting here would double-count.
2309
+ return {
2310
+ access_token: tokenData.accessToken,
2311
+ refresh_token: tokenData.refreshToken,
2312
+ expires_in: expiresIn,
2313
+ scope: tokenData.scope,
2314
+ token_type: 'Bearer',
2315
+ }
2316
+ }
2317
+ }
2318
+ logMCPDebug(
2319
+ this.serverName,
2320
+ `No valid tokens in storage, clearing stored tokens`,
2321
+ )
2322
+ await this.invalidateCredentials('tokens')
2323
+ emitRefreshEvent('failure', 'invalid_grant')
2324
+ return undefined
2325
+ }
2326
+
2327
+ // Retry on timeouts or transient server errors
2328
+ const isTimeoutError =
2329
+ error instanceof Error &&
2330
+ /timeout|timed out|etimedout|econnreset/i.test(error.message)
2331
+ const isTransientServerError =
2332
+ error instanceof ServerError ||
2333
+ error instanceof TemporarilyUnavailableError ||
2334
+ error instanceof TooManyRequestsError
2335
+ const isRetryable = isTimeoutError || isTransientServerError
2336
+
2337
+ if (!isRetryable || attempt >= MAX_ATTEMPTS) {
2338
+ logMCPDebug(
2339
+ this.serverName,
2340
+ `Token refresh failed: ${errorMessage(error)}`,
2341
+ )
2342
+ emitRefreshEvent(
2343
+ 'failure',
2344
+ isRetryable ? 'transient_retries_exhausted' : 'request_failed',
2345
+ )
2346
+ return undefined
2347
+ }
2348
+
2349
+ const delayMs = 1000 * Math.pow(2, attempt - 1) // 1s, 2s, 4s
2350
+ logMCPDebug(
2351
+ this.serverName,
2352
+ `Token refresh failed, retrying in ${delayMs}ms (attempt ${attempt}/${MAX_ATTEMPTS})`,
2353
+ )
2354
+ await sleep(delayMs)
2355
+ }
2356
+ }
2357
+
2358
+ return undefined
2359
+ }
2360
+ }
2361
+
2362
+ export async function readClientSecret(): Promise<string> {
2363
+ const envSecret = process.env.MCP_CLIENT_SECRET
2364
+ if (envSecret) {
2365
+ return envSecret
2366
+ }
2367
+
2368
+ if (!process.stdin.isTTY) {
2369
+ throw new Error(
2370
+ 'No TTY available to prompt for client secret. Set MCP_CLIENT_SECRET env var instead.',
2371
+ )
2372
+ }
2373
+
2374
+ return new Promise((resolve, reject) => {
2375
+ process.stderr.write('Enter OAuth client secret: ')
2376
+ process.stdin.setRawMode?.(true)
2377
+ let secret = ''
2378
+ const onData = (ch: Buffer) => {
2379
+ const c = ch.toString()
2380
+ if (c === '\n' || c === '\r') {
2381
+ process.stdin.setRawMode?.(false)
2382
+ process.stdin.removeListener('data', onData)
2383
+ process.stderr.write('\n')
2384
+ resolve(secret)
2385
+ } else if (c === '\u0003') {
2386
+ process.stdin.setRawMode?.(false)
2387
+ process.stdin.removeListener('data', onData)
2388
+ reject(new Error('Cancelled'))
2389
+ } else if (c === '\u007F' || c === '\b') {
2390
+ secret = secret.slice(0, -1)
2391
+ } else {
2392
+ secret += c
2393
+ }
2394
+ }
2395
+ process.stdin.on('data', onData)
2396
+ })
2397
+ }
2398
+
2399
+ export function saveMcpClientSecret(
2400
+ serverName: string,
2401
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
2402
+ clientSecret: string,
2403
+ ): void {
2404
+ const storage = getSecureStorage()
2405
+ const existingData = storage.read() || {}
2406
+ const serverKey = getServerKey(serverName, serverConfig)
2407
+ storage.update({
2408
+ ...existingData,
2409
+ mcpOAuthClientConfig: {
2410
+ ...existingData.mcpOAuthClientConfig,
2411
+ [serverKey]: { clientSecret },
2412
+ },
2413
+ })
2414
+ }
2415
+
2416
+ export function clearMcpClientConfig(
2417
+ serverName: string,
2418
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
2419
+ ): void {
2420
+ const storage = getSecureStorage()
2421
+ const existingData = storage.read()
2422
+ if (!existingData?.mcpOAuthClientConfig) return
2423
+ const serverKey = getServerKey(serverName, serverConfig)
2424
+ if (existingData.mcpOAuthClientConfig[serverKey]) {
2425
+ delete existingData.mcpOAuthClientConfig[serverKey]
2426
+ storage.update(existingData)
2427
+ }
2428
+ }
2429
+
2430
+ export function getMcpClientConfig(
2431
+ serverName: string,
2432
+ serverConfig: McpSSEServerConfig | McpHTTPServerConfig,
2433
+ ): { clientSecret?: string } | undefined {
2434
+ const storage = getSecureStorage()
2435
+ const data = storage.read()
2436
+ const serverKey = getServerKey(serverName, serverConfig)
2437
+ return data?.mcpOAuthClientConfig?.[serverKey]
2438
+ }
2439
+
2440
+ /**
2441
+ * Safely extracts scope information from AuthorizationServerMetadata.
2442
+ * The metadata can be either OAuthMetadata or OpenIdProviderDiscoveryMetadata,
2443
+ * and different providers use different fields for scope information.
2444
+ */
2445
+ function getScopeFromMetadata(
2446
+ metadata: AuthorizationServerMetadata | undefined,
2447
+ ): string | undefined {
2448
+ if (!metadata) return undefined
2449
+ // Try 'scope' first (non-standard but used by some providers)
2450
+ if ('scope' in metadata && typeof metadata.scope === 'string') {
2451
+ return metadata.scope
2452
+ }
2453
+ // Try 'default_scope' (non-standard but used by some providers)
2454
+ if (
2455
+ 'default_scope' in metadata &&
2456
+ typeof metadata.default_scope === 'string'
2457
+ ) {
2458
+ return metadata.default_scope
2459
+ }
2460
+ // Fall back to scopes_supported (standard OAuth 2.0 field)
2461
+ if (metadata.scopes_supported && Array.isArray(metadata.scopes_supported)) {
2462
+ return metadata.scopes_supported.join(' ')
2463
+ }
2464
+ return undefined
2465
+ }