@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,2643 @@
1
+ /**
2
+ * Marketplace manager for Claude Code plugins
3
+ *
4
+ * This module provides functionality to:
5
+ * - Manage known marketplace sources (URLs, GitHub repos, npm packages, local files)
6
+ * - Cache marketplace manifests locally for offline access
7
+ * - Install plugins from marketplace entries
8
+ * - Track and update marketplace configurations
9
+ *
10
+ * File structure managed by this module:
11
+ * ~/.claude/
12
+ * └── plugins/
13
+ * ├── known_marketplaces.json # Configuration of all known marketplaces
14
+ * └── marketplaces/ # Cache directory for marketplace data
15
+ * ├── my-marketplace.json # Cached marketplace from URL source
16
+ * └── github-marketplace/ # Cloned repository for GitHub source
17
+ * └── .claude-plugin/
18
+ * └── marketplace.json
19
+ */
20
+
21
+ import axios from 'axios'
22
+ import { writeFile } from 'fs/promises'
23
+ import isEqual from 'lodash-es/isEqual.js'
24
+ import memoize from 'lodash-es/memoize.js'
25
+ import { basename, dirname, isAbsolute, join, resolve, sep } from 'path'
26
+ import { getFeatureValue_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js'
27
+ import { logForDebugging } from '../debug.js'
28
+ import { isEnvTruthy } from '../envUtils.js'
29
+ import {
30
+ ConfigParseError,
31
+ errorMessage,
32
+ getErrnoCode,
33
+ isENOENT,
34
+ toError,
35
+ } from '../errors.js'
36
+ import { execFileNoThrow, execFileNoThrowWithCwd } from '../execFileNoThrow.js'
37
+ import { getFsImplementation } from '../fsOperations.js'
38
+ import { gitExe } from '../git.js'
39
+ import { logError } from '../log.js'
40
+ import {
41
+ getInitialSettings,
42
+ getSettingsForSource,
43
+ updateSettingsForSource,
44
+ } from '../settings/settings.js'
45
+ import type { SettingsJson } from '../settings/types.js'
46
+ import {
47
+ jsonParse,
48
+ jsonStringify,
49
+ writeFileSync_DEPRECATED,
50
+ } from '../slowOperations.js'
51
+ import {
52
+ getAddDirEnabledPlugins,
53
+ getAddDirExtraMarketplaces,
54
+ } from './addDirPluginSettings.js'
55
+ import { markPluginVersionOrphaned } from './cacheUtils.js'
56
+ import { classifyFetchError, logPluginFetch } from './fetchTelemetry.js'
57
+ import { removeAllPluginsForMarketplace } from './installedPluginsManager.js'
58
+ import {
59
+ extractHostFromSource,
60
+ formatSourceForDisplay,
61
+ getHostPatternsFromAllowlist,
62
+ getStrictKnownMarketplaces,
63
+ isSourceAllowedByPolicy,
64
+ isSourceInBlocklist,
65
+ } from './marketplaceHelpers.js'
66
+ import {
67
+ OFFICIAL_MARKETPLACE_NAME,
68
+ OFFICIAL_MARKETPLACE_SOURCE,
69
+ } from './officialMarketplace.js'
70
+ import { fetchOfficialMarketplaceFromGcs } from './officialMarketplaceGcs.js'
71
+ import {
72
+ deletePluginDataDir,
73
+ getPluginSeedDirs,
74
+ getPluginsDirectory,
75
+ } from './pluginDirectories.js'
76
+ import { parsePluginIdentifier } from './pluginIdentifier.js'
77
+ import { deletePluginOptions } from './pluginOptionsStorage.js'
78
+ import {
79
+ isLocalMarketplaceSource,
80
+ type KnownMarketplace,
81
+ type KnownMarketplacesFile,
82
+ KnownMarketplacesFileSchema,
83
+ type MarketplaceSource,
84
+ type PluginMarketplace,
85
+ type PluginMarketplaceEntry,
86
+ PluginMarketplaceSchema,
87
+ validateOfficialNameSource,
88
+ } from './schemas.js'
89
+
90
+ /**
91
+ * Result of loading and caching a marketplace
92
+ */
93
+ type LoadedPluginMarketplace = {
94
+ marketplace: PluginMarketplace
95
+ cachePath: string
96
+ }
97
+
98
+ /**
99
+ * Get the path to the known marketplaces configuration file
100
+ * Using a function instead of a constant allows proper mocking in tests
101
+ */
102
+ function getKnownMarketplacesFile(): string {
103
+ return join(getPluginsDirectory(), 'known_marketplaces.json')
104
+ }
105
+
106
+ /**
107
+ * Get the path to the marketplaces cache directory
108
+ * Using a function instead of a constant allows proper mocking in tests
109
+ */
110
+ export function getMarketplacesCacheDir(): string {
111
+ return join(getPluginsDirectory(), 'marketplaces')
112
+ }
113
+
114
+ /**
115
+ * Memoized inner function to get marketplace data.
116
+ * This caches the marketplace in memory after loading from disk or network.
117
+ */
118
+
119
+ /**
120
+ * Clear all cached marketplace data (for testing)
121
+ */
122
+ export function clearMarketplacesCache(): void {
123
+ getMarketplace.cache?.clear?.()
124
+ }
125
+
126
+ /**
127
+ * Configuration for known marketplaces
128
+ */
129
+ export type KnownMarketplacesConfig = KnownMarketplacesFile
130
+
131
+ /**
132
+ * Declared marketplace entry (intent layer).
133
+ *
134
+ * Structurally compatible with settings `extraKnownMarketplaces` entries, but
135
+ * adds `sourceIsFallback` for implicit built-in declarations. This is NOT a
136
+ * settings-schema field — it's only ever set in code (never parsed from JSON).
137
+ */
138
+ export type DeclaredMarketplace = {
139
+ source: MarketplaceSource
140
+ installLocation?: string
141
+ autoUpdate?: boolean
142
+ /**
143
+ * Presence suffices. When set, diffMarketplaces treats an already-materialized
144
+ * entry as upToDate regardless of source shape — never reports sourceChanged.
145
+ *
146
+ * Used for the implicit official-marketplace declaration: we want "clone from
147
+ * GitHub if missing", not "replace with GitHub if present under a different
148
+ * source". Without this, a seed dir that registers the official marketplace
149
+ * under e.g. an internal-mirror source would be stomped by a GitHub re-clone.
150
+ */
151
+ sourceIsFallback?: boolean
152
+ }
153
+
154
+ /**
155
+ * Get declared marketplace intent from merged settings and --add-dir sources.
156
+ * This is what SHOULD exist — used by the reconciler to find gaps.
157
+ *
158
+ * The official marketplace is implicitly declared with `sourceIsFallback: true`
159
+ * when any enabled plugin references it.
160
+ */
161
+ export function getDeclaredMarketplaces(): Record<string, DeclaredMarketplace> {
162
+ const implicit: Record<string, DeclaredMarketplace> = {}
163
+
164
+ // Only the official marketplace can be implicitly declared — it's the one
165
+ // built-in source we know. Other marketplaces have no default source to inject.
166
+ // Explicitly-disabled entries (value: false) don't count.
167
+ const enabledPlugins = {
168
+ ...getAddDirEnabledPlugins(),
169
+ ...(getInitialSettings().enabledPlugins ?? {}),
170
+ }
171
+ for (const [pluginId, value] of Object.entries(enabledPlugins)) {
172
+ if (
173
+ value &&
174
+ parsePluginIdentifier(pluginId).marketplace === OFFICIAL_MARKETPLACE_NAME
175
+ ) {
176
+ implicit[OFFICIAL_MARKETPLACE_NAME] = {
177
+ source: OFFICIAL_MARKETPLACE_SOURCE,
178
+ sourceIsFallback: true,
179
+ }
180
+ break
181
+ }
182
+ }
183
+
184
+ // Lowest precedence: implicit < --add-dir < merged settings.
185
+ // An explicit extraKnownMarketplaces entry for claude-plugins-official
186
+ // in --add-dir or settings wins.
187
+ return {
188
+ ...implicit,
189
+ ...getAddDirExtraMarketplaces(),
190
+ ...(getInitialSettings().extraKnownMarketplaces ?? {}),
191
+ }
192
+ }
193
+
194
+ /**
195
+ * Find which editable settings source declared a marketplace.
196
+ * Checks in reverse precedence order (highest priority last) so the
197
+ * result is the source that "wins" in the merged view.
198
+ * Returns null if the marketplace isn't declared in any editable source.
199
+ */
200
+ export function getMarketplaceDeclaringSource(
201
+ name: string,
202
+ ): 'userSettings' | 'projectSettings' | 'localSettings' | null {
203
+ // Check highest-precedence editable sources first — the one that wins
204
+ // in the merged view is the one we should write back to.
205
+ const editableSources: Array<
206
+ 'localSettings' | 'projectSettings' | 'userSettings'
207
+ > = ['localSettings', 'projectSettings', 'userSettings']
208
+
209
+ for (const source of editableSources) {
210
+ const settings = getSettingsForSource(source)
211
+ if (settings?.extraKnownMarketplaces?.[name]) {
212
+ return source
213
+ }
214
+ }
215
+ return null
216
+ }
217
+
218
+ /**
219
+ * Save a marketplace entry to settings (intent layer).
220
+ * Does NOT touch known_marketplaces.json (state layer).
221
+ *
222
+ * @param name - The marketplace name
223
+ * @param entry - The marketplace config
224
+ * @param settingSource - Which settings source to write to (defaults to userSettings)
225
+ */
226
+ export function saveMarketplaceToSettings(
227
+ name: string,
228
+ entry: DeclaredMarketplace,
229
+ settingSource:
230
+ | 'userSettings'
231
+ | 'projectSettings'
232
+ | 'localSettings' = 'userSettings',
233
+ ): void {
234
+ const existing = getSettingsForSource(settingSource) ?? {}
235
+ const current = { ...existing.extraKnownMarketplaces }
236
+ current[name] = entry
237
+ updateSettingsForSource(settingSource, { extraKnownMarketplaces: current })
238
+ }
239
+
240
+ /**
241
+ * Load known marketplaces configuration from disk
242
+ *
243
+ * Reads the configuration file at ~/.claude/plugins/known_marketplaces.json
244
+ * which contains a mapping of marketplace names to their sources and metadata.
245
+ *
246
+ * Example configuration file content:
247
+ * ```json
248
+ * {
249
+ * "official-marketplace": {
250
+ * "source": { "source": "url", "url": "https://example.com/marketplace.json" },
251
+ * "installLocation": "/Users/me/.claude/plugins/marketplaces/official-marketplace.json",
252
+ * "lastUpdated": "2024-01-15T10:30:00.000Z"
253
+ * },
254
+ * "company-plugins": {
255
+ * "source": { "source": "github", "repo": "mycompany/plugins" },
256
+ * "installLocation": "/Users/me/.claude/plugins/marketplaces/company-plugins",
257
+ * "lastUpdated": "2024-01-14T15:45:00.000Z"
258
+ * }
259
+ * }
260
+ * ```
261
+ *
262
+ * @returns Configuration object mapping marketplace names to their metadata
263
+ */
264
+ export async function loadKnownMarketplacesConfig(): Promise<KnownMarketplacesConfig> {
265
+ const fs = getFsImplementation()
266
+ const configFile = getKnownMarketplacesFile()
267
+
268
+ try {
269
+ const content = await fs.readFile(configFile, {
270
+ encoding: 'utf-8',
271
+ })
272
+ const data = jsonParse(content)
273
+ // Validate against schema
274
+ const parsed = KnownMarketplacesFileSchema().safeParse(data)
275
+ if (!parsed.success) {
276
+ const errorMsg = `Marketplace configuration file is corrupted: ${parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`).join(', ')}`
277
+ logForDebugging(errorMsg, {
278
+ level: 'error',
279
+ })
280
+ throw new ConfigParseError(errorMsg, configFile, data)
281
+ }
282
+ return parsed.data
283
+ } catch (error) {
284
+ if (isENOENT(error)) {
285
+ return {}
286
+ }
287
+ // If it's already a ConfigParseError, re-throw it
288
+ if (error instanceof ConfigParseError) {
289
+ throw error
290
+ }
291
+ // For JSON parse errors or I/O errors, throw with helpful message
292
+ const errorMsg = `Failed to load marketplace configuration: ${errorMessage(error)}`
293
+ logForDebugging(errorMsg, {
294
+ level: 'error',
295
+ })
296
+ throw new Error(errorMsg)
297
+ }
298
+ }
299
+
300
+ /**
301
+ * Load known marketplaces config, returning {} on any error instead of throwing.
302
+ *
303
+ * Use this on read-only paths (plugin loading, feature checks) where a corrupted
304
+ * config should degrade gracefully rather than crash. DO NOT use on load→mutate→save
305
+ * paths — returning {} there would cause the save to overwrite the corrupted file
306
+ * with just the new entry, permanently destroying the user's other entries. The
307
+ * throwing variant preserves the file so the user can fix the corruption and recover.
308
+ */
309
+ export async function loadKnownMarketplacesConfigSafe(): Promise<KnownMarketplacesConfig> {
310
+ try {
311
+ return await loadKnownMarketplacesConfig()
312
+ } catch {
313
+ // Inner function already logged via logForDebugging. Don't logError here —
314
+ // corrupted user config isn't a Claude Code bug, shouldn't hit the error file.
315
+ return {}
316
+ }
317
+ }
318
+
319
+ /**
320
+ * Save known marketplaces configuration to disk
321
+ *
322
+ * Writes the configuration to ~/.claude/plugins/known_marketplaces.json,
323
+ * creating the directory structure if it doesn't exist.
324
+ *
325
+ * @param config - The marketplace configuration to save
326
+ */
327
+ export async function saveKnownMarketplacesConfig(
328
+ config: KnownMarketplacesConfig,
329
+ ): Promise<void> {
330
+ // Validate before saving
331
+ const parsed = KnownMarketplacesFileSchema().safeParse(config)
332
+ const configFile = getKnownMarketplacesFile()
333
+
334
+ if (!parsed.success) {
335
+ throw new ConfigParseError(
336
+ `Invalid marketplace config: ${parsed.error.message}`,
337
+ configFile,
338
+ config,
339
+ )
340
+ }
341
+
342
+ const fs = getFsImplementation()
343
+ // Get directory from config file path to ensure consistency
344
+ const dir = join(configFile, '..')
345
+ await fs.mkdir(dir)
346
+ writeFileSync_DEPRECATED(configFile, jsonStringify(parsed.data, null, 2), {
347
+ encoding: 'utf-8',
348
+ flush: true,
349
+ })
350
+ }
351
+
352
+ /**
353
+ * Register marketplaces from the read-only seed directories into the primary
354
+ * known_marketplaces.json.
355
+ *
356
+ * The seed's known_marketplaces.json contains installLocation paths pointing
357
+ * into the seed dir itself. Registering those entries into the primary JSON
358
+ * makes them visible to all marketplace readers (getMarketplaceCacheOnly,
359
+ * getPluginByIdCacheOnly, etc.) without any loader changes — they just follow
360
+ * the installLocation wherever it points.
361
+ *
362
+ * Seed entries always win for marketplaces declared in the seed — the seed is
363
+ * admin-managed (baked into the container image). If admin updates the seed
364
+ * in a new image, those changes propagate on next boot. Users opt out of seed
365
+ * plugins via `plugin disable`, not by removing the marketplace.
366
+ *
367
+ * With multiple seed dirs (path-delimiter-separated), first-seed-wins: a
368
+ * marketplace name claimed by an earlier seed is skipped by later seeds.
369
+ *
370
+ * autoUpdate is forced to false since the seed is read-only and git-pull would
371
+ * fail. installLocation is computed from the runtime seedDir, not trusted from
372
+ * the seed's JSON (handles multi-stage Docker mount-path drift).
373
+ *
374
+ * Idempotent: second call with unchanged seed writes nothing.
375
+ *
376
+ * @returns true if any marketplace entries were written/changed (caller should
377
+ * clear caches so earlier plugin-load passes don't keep stale "marketplace
378
+ * not found" state)
379
+ */
380
+ export async function registerSeedMarketplaces(): Promise<boolean> {
381
+ const seedDirs = getPluginSeedDirs()
382
+ if (seedDirs.length === 0) return false
383
+
384
+ const primary = await loadKnownMarketplacesConfig()
385
+ // First-seed-wins across this registration pass. Can't use the isEqual check
386
+ // alone — two seeds with the same name will have different installLocations.
387
+ const claimed = new Set<string>()
388
+ let changed = 0
389
+
390
+ for (const seedDir of seedDirs) {
391
+ const seedConfig = await readSeedKnownMarketplaces(seedDir)
392
+ if (!seedConfig) continue
393
+
394
+ for (const [name, seedEntry] of Object.entries(seedConfig)) {
395
+ if (claimed.has(name)) continue
396
+
397
+ // Compute installLocation relative to THIS seedDir, not the build-time
398
+ // path baked into the seed's JSON. Handles multi-stage Docker builds
399
+ // where the seed is mounted at a different path than where it was built.
400
+ const resolvedLocation = await findSeedMarketplaceLocation(seedDir, name)
401
+ if (!resolvedLocation) {
402
+ // Seed content missing (incomplete build) — leave primary alone, but
403
+ // don't claim the name either: a later seed may have working content.
404
+ logForDebugging(
405
+ `Seed marketplace '${name}' not found under ${seedDir}/marketplaces/, skipping`,
406
+ { level: 'warn' },
407
+ )
408
+ continue
409
+ }
410
+ claimed.add(name)
411
+
412
+ const desired: KnownMarketplace = {
413
+ source: seedEntry.source,
414
+ installLocation: resolvedLocation,
415
+ lastUpdated: seedEntry.lastUpdated,
416
+ autoUpdate: false,
417
+ }
418
+
419
+ // Skip if primary already matches — idempotent no-op, no write.
420
+ if (isEqual(primary[name], desired)) continue
421
+
422
+ // Seed wins — admin-managed. Overwrite any existing primary entry.
423
+ primary[name] = desired
424
+ changed++
425
+ }
426
+ }
427
+
428
+ if (changed > 0) {
429
+ await saveKnownMarketplacesConfig(primary)
430
+ logForDebugging(`Synced ${changed} marketplace(s) from seed dir(s)`)
431
+ return true
432
+ }
433
+ return false
434
+ }
435
+
436
+ async function readSeedKnownMarketplaces(
437
+ seedDir: string,
438
+ ): Promise<KnownMarketplacesConfig | null> {
439
+ const seedJsonPath = join(seedDir, 'known_marketplaces.json')
440
+ try {
441
+ const content = await getFsImplementation().readFile(seedJsonPath, {
442
+ encoding: 'utf-8',
443
+ })
444
+ const parsed = KnownMarketplacesFileSchema().safeParse(jsonParse(content))
445
+ if (!parsed.success) {
446
+ logForDebugging(
447
+ `Seed known_marketplaces.json invalid at ${seedDir}: ${parsed.error.message}`,
448
+ { level: 'warn' },
449
+ )
450
+ return null
451
+ }
452
+ return parsed.data
453
+ } catch (e) {
454
+ if (!isENOENT(e)) {
455
+ logForDebugging(
456
+ `Failed to read seed known_marketplaces.json at ${seedDir}: ${e}`,
457
+ { level: 'warn' },
458
+ )
459
+ }
460
+ return null
461
+ }
462
+ }
463
+
464
+ /**
465
+ * Locate a marketplace in the seed directory by name.
466
+ *
467
+ * Probes the canonical locations under seedDir/marketplaces/ rather than
468
+ * trusting the seed's stored installLocation (which may have a stale absolute
469
+ * path from a different build-time mount point).
470
+ *
471
+ * @returns Readable location, or null if neither format exists/validates
472
+ */
473
+ async function findSeedMarketplaceLocation(
474
+ seedDir: string,
475
+ name: string,
476
+ ): Promise<string | null> {
477
+ const dirCandidate = join(seedDir, 'marketplaces', name)
478
+ const jsonCandidate = join(seedDir, 'marketplaces', `${name}.json`)
479
+ for (const candidate of [dirCandidate, jsonCandidate]) {
480
+ try {
481
+ await readCachedMarketplace(candidate)
482
+ return candidate
483
+ } catch {
484
+ // Try next candidate
485
+ }
486
+ }
487
+ return null
488
+ }
489
+
490
+ /**
491
+ * If installLocation points into a configured seed directory, return that seed
492
+ * directory. Seed-managed entries are admin-controlled — users can't
493
+ * remove/refresh/modify them (they'd be overwritten by registerSeedMarketplaces
494
+ * on next startup). Returning the specific seed lets error messages name it.
495
+ */
496
+ function seedDirFor(installLocation: string): string | undefined {
497
+ return getPluginSeedDirs().find(
498
+ d => installLocation === d || installLocation.startsWith(d + sep),
499
+ )
500
+ }
501
+
502
+ /**
503
+ * Git pull operation (exported for testing)
504
+ *
505
+ * Pulls latest changes with a configurable timeout (default 120s, override via CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS).
506
+ * Provides helpful error messages for common failure scenarios.
507
+ * If a ref is specified, fetches and checks out that specific branch or tag.
508
+ */
509
+ // Environment variables to prevent git from prompting for credentials
510
+ const GIT_NO_PROMPT_ENV = {
511
+ GIT_TERMINAL_PROMPT: '0', // Prevent terminal credential prompts
512
+ GIT_ASKPASS: '', // Disable askpass GUI programs
513
+ }
514
+
515
+ const DEFAULT_PLUGIN_GIT_TIMEOUT_MS = 120 * 1000
516
+
517
+ function getPluginGitTimeoutMs(): number {
518
+ const envValue = process.env.CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS
519
+ if (envValue) {
520
+ const parsed = parseInt(envValue, 10)
521
+ if (!isNaN(parsed) && parsed > 0) {
522
+ return parsed
523
+ }
524
+ }
525
+ return DEFAULT_PLUGIN_GIT_TIMEOUT_MS
526
+ }
527
+
528
+ export async function gitPull(
529
+ cwd: string,
530
+ ref?: string,
531
+ options?: { disableCredentialHelper?: boolean; sparsePaths?: string[] },
532
+ ): Promise<{ code: number; stderr: string }> {
533
+ logForDebugging(`git pull: cwd=${cwd} ref=${ref ?? 'default'}`)
534
+ const env = { ...process.env, ...GIT_NO_PROMPT_ENV }
535
+ const credentialArgs = options?.disableCredentialHelper
536
+ ? ['-c', 'credential.helper=']
537
+ : []
538
+
539
+ if (ref) {
540
+ const fetchResult = await execFileNoThrowWithCwd(
541
+ gitExe(),
542
+ [...credentialArgs, 'fetch', 'origin', ref],
543
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
544
+ )
545
+
546
+ if (fetchResult.code !== 0) {
547
+ return enhanceGitPullErrorMessages(fetchResult)
548
+ }
549
+
550
+ const checkoutResult = await execFileNoThrowWithCwd(
551
+ gitExe(),
552
+ [...credentialArgs, 'checkout', ref],
553
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
554
+ )
555
+
556
+ if (checkoutResult.code !== 0) {
557
+ return enhanceGitPullErrorMessages(checkoutResult)
558
+ }
559
+
560
+ const pullResult = await execFileNoThrowWithCwd(
561
+ gitExe(),
562
+ [...credentialArgs, 'pull', 'origin', ref],
563
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
564
+ )
565
+ if (pullResult.code !== 0) {
566
+ return enhanceGitPullErrorMessages(pullResult)
567
+ }
568
+ await gitSubmoduleUpdate(cwd, credentialArgs, env, options?.sparsePaths)
569
+ return pullResult
570
+ }
571
+
572
+ const result = await execFileNoThrowWithCwd(
573
+ gitExe(),
574
+ [...credentialArgs, 'pull', 'origin', 'HEAD'],
575
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
576
+ )
577
+ if (result.code !== 0) {
578
+ return enhanceGitPullErrorMessages(result)
579
+ }
580
+ await gitSubmoduleUpdate(cwd, credentialArgs, env, options?.sparsePaths)
581
+ return result
582
+ }
583
+
584
+ /**
585
+ * Sync submodule working dirs after a successful pull. gitClone() uses
586
+ * --recurse-submodules, but gitPull() didn't — the parent repo's submodule
587
+ * pointer would advance while the working dir stayed at the old commit,
588
+ * making plugin sources in submodules unresolvable after marketplace update.
589
+ * Non-fatal: a failed submodule update logs a warning; most marketplaces
590
+ * don't use submodules at all. (gh-30696)
591
+ *
592
+ * Skipped for sparse clones — gitClone's sparse path intentionally omits
593
+ * --recurse-submodules to preserve partial-clone bandwidth savings, and
594
+ * .gitmodules is a root file that cone-mode sparse-checkout always
595
+ * materializes, so the .gitmodules gate alone can't distinguish sparse repos.
596
+ *
597
+ * Perf: git-submodule is a bash script that spawns ~20 subprocesses (~35ms+)
598
+ * even when no submodules exist. .gitmodules is a tracked file — pull
599
+ * materializes it iff the repo has submodules — so gate on its presence to
600
+ * skip the spawn for the common case.
601
+ *
602
+ * --init performs first-contact clone of newly-added submodules, so maintain
603
+ * parity with gitClone's non-sparse path: StrictHostKeyChecking=yes for
604
+ * fail-closed SSH (unknown hosts reject rather than silently populate
605
+ * known_hosts), and --depth 1 for shallow clone (matching --shallow-submodules).
606
+ * --depth only affects not-yet-initialized submodules; existing shallow
607
+ * submodules are unaffected.
608
+ */
609
+ async function gitSubmoduleUpdate(
610
+ cwd: string,
611
+ credentialArgs: string[],
612
+ env: NodeJS.ProcessEnv,
613
+ sparsePaths: string[] | undefined,
614
+ ): Promise<void> {
615
+ if (sparsePaths && sparsePaths.length > 0) return
616
+ const hasGitmodules = await getFsImplementation()
617
+ .stat(join(cwd, '.gitmodules'))
618
+ .then(
619
+ () => true,
620
+ () => false,
621
+ )
622
+ if (!hasGitmodules) return
623
+ const result = await execFileNoThrowWithCwd(
624
+ gitExe(),
625
+ [
626
+ '-c',
627
+ 'core.sshCommand=ssh -o BatchMode=yes -o StrictHostKeyChecking=yes',
628
+ ...credentialArgs,
629
+ 'submodule',
630
+ 'update',
631
+ '--init',
632
+ '--recursive',
633
+ '--depth',
634
+ '1',
635
+ ],
636
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
637
+ )
638
+ if (result.code !== 0) {
639
+ logForDebugging(
640
+ `git submodule update failed (non-fatal): ${result.stderr}`,
641
+ { level: 'warn' },
642
+ )
643
+ }
644
+ }
645
+
646
+ /**
647
+ * Enhance error messages for git pull failures
648
+ */
649
+ function enhanceGitPullErrorMessages(result: {
650
+ code: number
651
+ stderr: string
652
+ error?: string
653
+ }): { code: number; stderr: string } {
654
+ if (result.code === 0) {
655
+ return result
656
+ }
657
+
658
+ // Detect execa timeout kills via the error field (stderr won't contain "timed out"
659
+ // when the process is killed by SIGTERM — the timeout info is only in error)
660
+ if (result.error?.includes('timed out')) {
661
+ const timeoutSec = Math.round(getPluginGitTimeoutMs() / 1000)
662
+ return {
663
+ ...result,
664
+ stderr: `Git pull timed out after ${timeoutSec}s. Try increasing the timeout via CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS environment variable.\n\nOriginal error: ${result.stderr}`,
665
+ }
666
+ }
667
+
668
+ // Detect SSH host key verification failures (check before the generic
669
+ // 'Could not read from remote' catch — that string appears in both cases).
670
+ // OpenSSH emits "Host key verification failed" for BOTH host-not-in-known_hosts
671
+ // and host-key-has-changed — the latter also includes the "REMOTE HOST
672
+ // IDENTIFICATION HAS CHANGED" banner, which needs different remediation.
673
+ if (result.stderr.includes('REMOTE HOST IDENTIFICATION HAS CHANGED')) {
674
+ return {
675
+ ...result,
676
+ stderr: `SSH host key for this marketplace's git host has changed (server key rotation or possible MITM). Remove the stale entry with: ssh-keygen -R <host>\nThen connect once manually to accept the new key.\n\nOriginal error: ${result.stderr}`,
677
+ }
678
+ }
679
+ if (result.stderr.includes('Host key verification failed')) {
680
+ return {
681
+ ...result,
682
+ stderr: `SSH host key verification failed while updating marketplace. The host key is not in your known_hosts file. Connect once manually to add it (e.g., ssh -T git@<host>), or remove and re-add the marketplace with an HTTPS URL.\n\nOriginal error: ${result.stderr}`,
683
+ }
684
+ }
685
+
686
+ // Detect SSH authentication failures
687
+ if (
688
+ result.stderr.includes('Permission denied (publickey)') ||
689
+ result.stderr.includes('Could not read from remote repository')
690
+ ) {
691
+ return {
692
+ ...result,
693
+ stderr: `SSH authentication failed while updating marketplace. Please ensure your SSH keys are configured.\n\nOriginal error: ${result.stderr}`,
694
+ }
695
+ }
696
+
697
+ // Detect network issues
698
+ if (
699
+ result.stderr.includes('timed out') ||
700
+ result.stderr.includes('Could not resolve host')
701
+ ) {
702
+ return {
703
+ ...result,
704
+ stderr: `Network error while updating marketplace. Please check your internet connection.\n\nOriginal error: ${result.stderr}`,
705
+ }
706
+ }
707
+
708
+ return result
709
+ }
710
+
711
+ /**
712
+ * Check if SSH is likely to work for GitHub
713
+ * This is a quick heuristic check that avoids the full clone timeout
714
+ *
715
+ * Uses StrictHostKeyChecking=yes (not accept-new) so an unknown github.com
716
+ * host key fails closed rather than being silently added to known_hosts.
717
+ * This prevents a network-level MITM from poisoning known_hosts on first
718
+ * contact. Users who already have github.com in known_hosts see no change;
719
+ * users who don't are routed to the HTTPS clone path.
720
+ *
721
+ * @returns true if SSH auth succeeds and github.com is already trusted
722
+ */
723
+ async function isGitHubSshLikelyConfigured(): Promise<boolean> {
724
+ try {
725
+ // Quick SSH connection test with 2 second timeout
726
+ // This fails fast if SSH isn't configured
727
+ const result = await execFileNoThrow(
728
+ 'ssh',
729
+ [
730
+ '-T',
731
+ '-o',
732
+ 'BatchMode=yes',
733
+ '-o',
734
+ 'ConnectTimeout=2',
735
+ '-o',
736
+ 'StrictHostKeyChecking=yes',
737
+ 'git@github.com',
738
+ ],
739
+ {
740
+ timeout: 3000, // 3 second total timeout
741
+ },
742
+ )
743
+
744
+ // SSH to github.com always returns exit code 1 with "successfully authenticated"
745
+ // or exit code 255 with "Permission denied" - we want the former
746
+ const configured =
747
+ result.code === 1 &&
748
+ (result.stderr?.includes('successfully authenticated') ||
749
+ result.stdout?.includes('successfully authenticated'))
750
+ logForDebugging(
751
+ `SSH config check: code=${result.code} configured=${configured}`,
752
+ )
753
+ return configured
754
+ } catch (error) {
755
+ // Any error means SSH isn't configured properly
756
+ logForDebugging(`SSH configuration check failed: ${errorMessage(error)}`, {
757
+ level: 'warn',
758
+ })
759
+ return false
760
+ }
761
+ }
762
+
763
+ /**
764
+ * Check if a git error indicates authentication failure.
765
+ * Used to provide enhanced error messages for auth failures.
766
+ */
767
+ function isAuthenticationError(stderr: string): boolean {
768
+ return (
769
+ stderr.includes('Authentication failed') ||
770
+ stderr.includes('could not read Username') ||
771
+ stderr.includes('terminal prompts disabled') ||
772
+ stderr.includes('403') ||
773
+ stderr.includes('401')
774
+ )
775
+ }
776
+
777
+ /**
778
+ * Extract the SSH host from a git URL for error messaging.
779
+ * Matches the SSH format user@host:path (e.g., git@github.com:owner/repo.git).
780
+ */
781
+ function extractSshHost(gitUrl: string): string | null {
782
+ const match = gitUrl.match(/^[^@]+@([^:]+):/)
783
+ return match?.[1] ?? null
784
+ }
785
+
786
+ /**
787
+ * Git clone operation (exported for testing)
788
+ *
789
+ * Clones a git repository with a configurable timeout (default 120s, override via CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS)
790
+ * and larger repositories. Provides helpful error messages for common failure scenarios.
791
+ * Optionally checks out a specific branch or tag.
792
+ *
793
+ * Does NOT disable credential helpers — this allows the user's existing auth setup
794
+ * (gh auth, keychain, git-credential-store, etc.) to work natively for private repos.
795
+ * Interactive prompts are still prevented via GIT_TERMINAL_PROMPT=0, GIT_ASKPASS='',
796
+ * stdin: 'ignore', and BatchMode=yes for SSH.
797
+ *
798
+ * Uses StrictHostKeyChecking=yes (not accept-new): unknown SSH hosts fail closed
799
+ * with a clear message rather than being silently trusted on first contact. For
800
+ * the github source type, the preflight check routes unknown-host users to HTTPS
801
+ * automatically; for explicit git@host:… URLs, users see an actionable error.
802
+ */
803
+ export async function gitClone(
804
+ gitUrl: string,
805
+ targetPath: string,
806
+ ref?: string,
807
+ sparsePaths?: string[],
808
+ ): Promise<{ code: number; stderr: string }> {
809
+ const useSparse = sparsePaths && sparsePaths.length > 0
810
+ const args = [
811
+ '-c',
812
+ 'core.sshCommand=ssh -o BatchMode=yes -o StrictHostKeyChecking=yes',
813
+ 'clone',
814
+ '--depth',
815
+ '1',
816
+ ]
817
+
818
+ if (useSparse) {
819
+ // Partial clone: skip blob download until checkout, defer checkout until
820
+ // after sparse-checkout is configured. Submodules are intentionally dropped
821
+ // for sparse clones — sparse monorepos rarely need them, and recursing
822
+ // submodules would defeat the partial-clone bandwidth savings.
823
+ args.push('--filter=blob:none', '--no-checkout')
824
+ } else {
825
+ args.push('--recurse-submodules', '--shallow-submodules')
826
+ }
827
+
828
+ if (ref) {
829
+ args.push('--branch', ref)
830
+ }
831
+
832
+ args.push(gitUrl, targetPath)
833
+
834
+ const timeoutMs = getPluginGitTimeoutMs()
835
+ logForDebugging(
836
+ `git clone: url=${redactUrlCredentials(gitUrl)} ref=${ref ?? 'default'} timeout=${timeoutMs}ms`,
837
+ )
838
+
839
+ const result = await execFileNoThrowWithCwd(gitExe(), args, {
840
+ timeout: timeoutMs,
841
+ stdin: 'ignore',
842
+ env: { ...process.env, ...GIT_NO_PROMPT_ENV },
843
+ })
844
+
845
+ // Scrub credentials from execa's error/stderr fields before any logging or
846
+ // returning. execa's shortMessage embeds the full command line (including
847
+ // the credentialed URL), and result.stderr may also contain it on some git
848
+ // versions.
849
+ const redacted = redactUrlCredentials(gitUrl)
850
+ if (gitUrl !== redacted) {
851
+ if (result.error) result.error = result.error.replaceAll(gitUrl, redacted)
852
+ if (result.stderr)
853
+ result.stderr = result.stderr.replaceAll(gitUrl, redacted)
854
+ }
855
+
856
+ if (result.code === 0) {
857
+ if (useSparse) {
858
+ // Configure the sparse cone, then materialize only those paths.
859
+ // `sparse-checkout set --cone` handles both init and path selection
860
+ // in a single step on git >= 2.25.
861
+ const sparseResult = await execFileNoThrowWithCwd(
862
+ gitExe(),
863
+ ['sparse-checkout', 'set', '--cone', '--', ...sparsePaths],
864
+ {
865
+ cwd: targetPath,
866
+ timeout: timeoutMs,
867
+ stdin: 'ignore',
868
+ env: { ...process.env, ...GIT_NO_PROMPT_ENV },
869
+ },
870
+ )
871
+ if (sparseResult.code !== 0) {
872
+ return {
873
+ code: sparseResult.code,
874
+ stderr: `git sparse-checkout set failed: ${sparseResult.stderr}`,
875
+ }
876
+ }
877
+
878
+ const checkoutResult = await execFileNoThrowWithCwd(
879
+ gitExe(),
880
+ // ref was already passed to clone via --branch, so HEAD points to it;
881
+ // if no ref, HEAD points to the remote's default branch.
882
+ ['checkout', 'HEAD'],
883
+ {
884
+ cwd: targetPath,
885
+ timeout: timeoutMs,
886
+ stdin: 'ignore',
887
+ env: { ...process.env, ...GIT_NO_PROMPT_ENV },
888
+ },
889
+ )
890
+ if (checkoutResult.code !== 0) {
891
+ return {
892
+ code: checkoutResult.code,
893
+ stderr: `git checkout after sparse-checkout failed: ${checkoutResult.stderr}`,
894
+ }
895
+ }
896
+ }
897
+ logForDebugging(`git clone succeeded: ${redactUrlCredentials(gitUrl)}`)
898
+ return result
899
+ }
900
+
901
+ logForDebugging(
902
+ `git clone failed: url=${redactUrlCredentials(gitUrl)} code=${result.code} error=${result.error ?? 'none'} stderr=${result.stderr}`,
903
+ { level: 'warn' },
904
+ )
905
+
906
+ // Detect timeout kills — when execFileNoThrowWithCwd kills the process via SIGTERM,
907
+ // stderr may only contain partial output (e.g. "Cloning into '...'") with no
908
+ // "timed out" string. Check the error field from execa which contains the
909
+ // timeout message.
910
+ if (result.error?.includes('timed out')) {
911
+ return {
912
+ ...result,
913
+ stderr: `Git clone timed out after ${Math.round(timeoutMs / 1000)}s. The repository may be too large for the current timeout. Set CLAUDE_CODE_PLUGIN_GIT_TIMEOUT_MS to increase it (e.g., 300000 for 5 minutes).\n\nOriginal error: ${result.stderr}`,
914
+ }
915
+ }
916
+
917
+ // Enhance error messages for common scenarios
918
+ if (result.stderr) {
919
+ // Host key verification failure — check FIRST, before the generic
920
+ // 'Could not read from remote repository' catch (that string appears
921
+ // in both stderr outputs, so order matters). OpenSSH emits
922
+ // "Host key verification failed" for BOTH host-not-in-known_hosts and
923
+ // host-key-has-changed; distinguish them by the key-change banner.
924
+ if (result.stderr.includes('REMOTE HOST IDENTIFICATION HAS CHANGED')) {
925
+ const host = extractSshHost(gitUrl)
926
+ const removeHint = host ? `ssh-keygen -R ${host}` : 'ssh-keygen -R <host>'
927
+ return {
928
+ ...result,
929
+ stderr: `SSH host key has changed (server key rotation or possible MITM). Remove the stale known_hosts entry:\n ${removeHint}\nThen connect once manually to verify and accept the new key.\n\nOriginal error: ${result.stderr}`,
930
+ }
931
+ }
932
+ if (result.stderr.includes('Host key verification failed')) {
933
+ const host = extractSshHost(gitUrl)
934
+ const connectHint = host ? `ssh -T git@${host}` : 'ssh -T git@<host>'
935
+ return {
936
+ ...result,
937
+ stderr: `SSH host key is not in your known_hosts file. To add it, connect once manually (this will show the fingerprint for you to verify):\n ${connectHint}\n\nOr use an HTTPS URL instead (recommended for public repos).\n\nOriginal error: ${result.stderr}`,
938
+ }
939
+ }
940
+
941
+ if (
942
+ result.stderr.includes('Permission denied (publickey)') ||
943
+ result.stderr.includes('Could not read from remote repository')
944
+ ) {
945
+ return {
946
+ ...result,
947
+ stderr: `SSH authentication failed. Please ensure your SSH keys are configured for GitHub, or use an HTTPS URL instead.\n\nOriginal error: ${result.stderr}`,
948
+ }
949
+ }
950
+
951
+ if (isAuthenticationError(result.stderr)) {
952
+ return {
953
+ ...result,
954
+ stderr: `HTTPS authentication failed. Please ensure your credential helper is configured (e.g., gh auth login).\n\nOriginal error: ${result.stderr}`,
955
+ }
956
+ }
957
+
958
+ if (
959
+ result.stderr.includes('timed out') ||
960
+ result.stderr.includes('timeout') ||
961
+ result.stderr.includes('Could not resolve host')
962
+ ) {
963
+ return {
964
+ ...result,
965
+ stderr: `Network error or timeout while cloning repository. Please check your internet connection and try again.\n\nOriginal error: ${result.stderr}`,
966
+ }
967
+ }
968
+ }
969
+
970
+ // Fallback for empty stderr — gh-28373: user saw "Failed to clone
971
+ // marketplace repository:" with nothing after the colon. Git CAN fail
972
+ // without writing to stderr (stdout instead, or output swallowed by
973
+ // credential helper / signal). execa's error field has the execa-level
974
+ // message (command, exit code, signal); exit code is the minimum.
975
+ if (!result.stderr) {
976
+ return {
977
+ code: result.code,
978
+ stderr:
979
+ result.error ||
980
+ `git clone exited with code ${result.code} (no stderr output). Run with --debug to see the full command.`,
981
+ }
982
+ }
983
+
984
+ return result
985
+ }
986
+
987
+ /**
988
+ * Progress callback for marketplace operations.
989
+ *
990
+ * This callback is invoked at various stages during marketplace operations
991
+ * (downloading, git operations, validation, etc.) to provide user feedback.
992
+ *
993
+ * IMPORTANT: Implementations should handle errors internally and not throw exceptions.
994
+ * If a callback throws, it will be caught and logged but won't abort the operation.
995
+ *
996
+ * @param message - Human-readable progress message to display to the user
997
+ */
998
+ export type MarketplaceProgressCallback = (message: string) => void
999
+
1000
+ /**
1001
+ * Safely invoke a progress callback, catching and logging any errors.
1002
+ * Prevents callback errors from aborting marketplace operations.
1003
+ *
1004
+ * @param onProgress - The progress callback to invoke
1005
+ * @param message - Progress message to pass to the callback
1006
+ */
1007
+ function safeCallProgress(
1008
+ onProgress: MarketplaceProgressCallback | undefined,
1009
+ message: string,
1010
+ ): void {
1011
+ if (!onProgress) return
1012
+ try {
1013
+ onProgress(message)
1014
+ } catch (callbackError) {
1015
+ logForDebugging(`Progress callback error: ${errorMessage(callbackError)}`, {
1016
+ level: 'warn',
1017
+ })
1018
+ }
1019
+ }
1020
+
1021
+ /**
1022
+ * Reconcile the on-disk sparse-checkout state with the desired config.
1023
+ *
1024
+ * Runs before gitPull to handle transitions:
1025
+ * - Full→Sparse or SparseA→SparseB: run `sparse-checkout set --cone` (idempotent)
1026
+ * - Sparse→Full: return non-zero so caller falls back to rm+reclone. Avoids
1027
+ * `sparse-checkout disable` on a --filter=blob:none partial clone, which would
1028
+ * trigger a lazy fetch of every blob in the monorepo.
1029
+ * - Full→Full (common case): single local `git config --get` check, no-op.
1030
+ *
1031
+ * Failures here (ENOENT, not a repo) are harmless — gitPull will also fail and
1032
+ * trigger the clone path, which establishes the correct state from scratch.
1033
+ */
1034
+ export async function reconcileSparseCheckout(
1035
+ cwd: string,
1036
+ sparsePaths: string[] | undefined,
1037
+ ): Promise<{ code: number; stderr: string }> {
1038
+ const env = { ...process.env, ...GIT_NO_PROMPT_ENV }
1039
+
1040
+ if (sparsePaths && sparsePaths.length > 0) {
1041
+ return execFileNoThrowWithCwd(
1042
+ gitExe(),
1043
+ ['sparse-checkout', 'set', '--cone', '--', ...sparsePaths],
1044
+ { cwd, timeout: getPluginGitTimeoutMs(), stdin: 'ignore', env },
1045
+ )
1046
+ }
1047
+
1048
+ const check = await execFileNoThrowWithCwd(
1049
+ gitExe(),
1050
+ ['config', '--get', 'core.sparseCheckout'],
1051
+ { cwd, stdin: 'ignore', env },
1052
+ )
1053
+ if (check.code === 0 && check.stdout.trim() === 'true') {
1054
+ return {
1055
+ code: 1,
1056
+ stderr:
1057
+ 'sparsePaths removed from config but repository is sparse; re-cloning for full checkout',
1058
+ }
1059
+ }
1060
+ return { code: 0, stderr: '' }
1061
+ }
1062
+
1063
+ /**
1064
+ * Cache a marketplace from a git repository
1065
+ *
1066
+ * Clones or updates a git repository containing marketplace data.
1067
+ * If the repository already exists at cachePath, pulls the latest changes.
1068
+ * If pulling fails, removes the directory and re-clones.
1069
+ *
1070
+ * Example repository structure:
1071
+ * ```
1072
+ * my-marketplace/
1073
+ * ├── .claude-plugin/
1074
+ * │ └── marketplace.json # Default location for marketplace manifest
1075
+ * ├── plugins/ # Plugin implementations
1076
+ * └── README.md
1077
+ * ```
1078
+ *
1079
+ * @param gitUrl - The git URL to clone (https or ssh)
1080
+ * @param cachePath - Local directory path to clone/update the repository
1081
+ * @param ref - Optional git branch or tag to checkout
1082
+ * @param onProgress - Optional callback to report progress
1083
+ */
1084
+ async function cacheMarketplaceFromGit(
1085
+ gitUrl: string,
1086
+ cachePath: string,
1087
+ ref?: string,
1088
+ sparsePaths?: string[],
1089
+ onProgress?: MarketplaceProgressCallback,
1090
+ options?: { disableCredentialHelper?: boolean },
1091
+ ): Promise<void> {
1092
+ const fs = getFsImplementation()
1093
+
1094
+ // Attempt incremental update; fall back to re-clone if the repo is absent,
1095
+ // stale, or otherwise not updatable. Using pull-first avoids a stat-before-operate
1096
+ // TOCTOU check: gitPull returns non-zero when cachePath is missing or has no .git.
1097
+ const timeoutSec = Math.round(getPluginGitTimeoutMs() / 1000)
1098
+ safeCallProgress(
1099
+ onProgress,
1100
+ `Refreshing marketplace cache (timeout: ${timeoutSec}s)…`,
1101
+ )
1102
+
1103
+ // Reconcile sparse-checkout config before pulling. If this requires a re-clone
1104
+ // (Sparse→Full transition) or fails (missing dir, not a repo), skip straight
1105
+ // to the rm+clone fallback.
1106
+ const reconcileResult = await reconcileSparseCheckout(cachePath, sparsePaths)
1107
+ if (reconcileResult.code === 0) {
1108
+ const pullStarted = performance.now()
1109
+ const pullResult = await gitPull(cachePath, ref, {
1110
+ disableCredentialHelper: options?.disableCredentialHelper,
1111
+ sparsePaths,
1112
+ })
1113
+ logPluginFetch(
1114
+ 'marketplace_pull',
1115
+ gitUrl,
1116
+ pullResult.code === 0 ? 'success' : 'failure',
1117
+ performance.now() - pullStarted,
1118
+ pullResult.code === 0 ? undefined : classifyFetchError(pullResult.stderr),
1119
+ )
1120
+ if (pullResult.code === 0) return
1121
+ logForDebugging(`git pull failed, will re-clone: ${pullResult.stderr}`, {
1122
+ level: 'warn',
1123
+ })
1124
+ } else {
1125
+ logForDebugging(
1126
+ `sparse-checkout reconcile requires re-clone: ${reconcileResult.stderr}`,
1127
+ )
1128
+ }
1129
+
1130
+ try {
1131
+ await fs.rm(cachePath, { recursive: true })
1132
+ // rm succeeded — a stale or partially-cloned directory existed; log for diagnostics
1133
+ logForDebugging(
1134
+ `Found stale marketplace directory at ${cachePath}, cleaning up to allow re-clone`,
1135
+ { level: 'warn' },
1136
+ )
1137
+ safeCallProgress(
1138
+ onProgress,
1139
+ 'Found stale directory, cleaning up and re-cloning…',
1140
+ )
1141
+ } catch (rmError) {
1142
+ if (!isENOENT(rmError)) {
1143
+ const rmErrorMsg = errorMessage(rmError)
1144
+ throw new Error(
1145
+ `Failed to clean up existing marketplace directory. Please manually delete the directory at ${cachePath} and try again.\n\nTechnical details: ${rmErrorMsg}`,
1146
+ )
1147
+ }
1148
+ // ENOENT — cachePath didn't exist, this is a fresh install, nothing to clean up
1149
+ }
1150
+
1151
+ // Clone the repository (one attempt — no internal retry loop)
1152
+ const refMessage = ref ? ` (ref: ${ref})` : ''
1153
+ safeCallProgress(
1154
+ onProgress,
1155
+ `Cloning repository (timeout: ${timeoutSec}s): ${redactUrlCredentials(gitUrl)}${refMessage}`,
1156
+ )
1157
+ const cloneStarted = performance.now()
1158
+ const result = await gitClone(gitUrl, cachePath, ref, sparsePaths)
1159
+ logPluginFetch(
1160
+ 'marketplace_clone',
1161
+ gitUrl,
1162
+ result.code === 0 ? 'success' : 'failure',
1163
+ performance.now() - cloneStarted,
1164
+ result.code === 0 ? undefined : classifyFetchError(result.stderr),
1165
+ )
1166
+ if (result.code !== 0) {
1167
+ // Clean up any partial directory created by the failed clone so the next
1168
+ // attempt starts fresh. Best-effort: if this fails, the stale dir will be
1169
+ // auto-detected and removed at the top of the next call.
1170
+ try {
1171
+ await fs.rm(cachePath, { recursive: true, force: true })
1172
+ } catch {
1173
+ // ignore
1174
+ }
1175
+ throw new Error(`Failed to clone marketplace repository: ${result.stderr}`)
1176
+ }
1177
+ safeCallProgress(onProgress, 'Clone complete, validating marketplace…')
1178
+ }
1179
+
1180
+ /**
1181
+ * Redact header values for safe logging
1182
+ *
1183
+ * @param headers - Headers to redact
1184
+ * @returns Headers with values replaced by '***REDACTED***'
1185
+ */
1186
+ function redactHeaders(
1187
+ headers: Record<string, string>,
1188
+ ): Record<string, string> {
1189
+ return Object.fromEntries(
1190
+ Object.entries(headers).map(([key]) => [key, '***REDACTED***']),
1191
+ )
1192
+ }
1193
+
1194
+ /**
1195
+ * Redact userinfo (username:password) in a URL to avoid logging credentials.
1196
+ *
1197
+ * Marketplace URLs may embed credentials (e.g. GitHub PATs in
1198
+ * `https://user:token@github.com/org/repo`). Debug logs and progress output
1199
+ * are written to disk and may be included in bug reports, so credentials must
1200
+ * be redacted before logging.
1201
+ *
1202
+ * Redacts all credentials from http(s) URLs:
1203
+ * https://user:token@github.com/repo → https://***:***@github.com/repo
1204
+ * https://:token@github.com/repo → https://:***@github.com/repo
1205
+ * https://token@github.com/repo → https://***@github.com/repo
1206
+ *
1207
+ * Both username and password are redacted unconditionally on http(s) because
1208
+ * it is impossible to distinguish `placeholder:secret` (e.g. x-access-token:ghp_...)
1209
+ * from `secret:placeholder` (e.g. ghp_...:x-oauth-basic) by parsing alone.
1210
+ * Non-http(s) schemes (ssh://git@...) and non-URL inputs (`owner/repo` shorthand)
1211
+ * pass through unchanged.
1212
+ */
1213
+ function redactUrlCredentials(urlString: string): string {
1214
+ try {
1215
+ const parsed = new URL(urlString)
1216
+ const isHttp = parsed.protocol === 'http:' || parsed.protocol === 'https:'
1217
+ if (isHttp && (parsed.username || parsed.password)) {
1218
+ if (parsed.username) parsed.username = '***'
1219
+ if (parsed.password) parsed.password = '***'
1220
+ return parsed.toString()
1221
+ }
1222
+ } catch {
1223
+ // Not a valid URL — safe as-is
1224
+ }
1225
+ return urlString
1226
+ }
1227
+
1228
+ /**
1229
+ * Cache a marketplace from a URL
1230
+ *
1231
+ * Downloads a marketplace.json file from a URL and saves it locally.
1232
+ * Creates the cache directory structure if it doesn't exist.
1233
+ *
1234
+ * Example marketplace.json structure:
1235
+ * ```json
1236
+ * {
1237
+ * "name": "my-marketplace",
1238
+ * "owner": { "name": "John Doe", "email": "john@example.com" },
1239
+ * "plugins": [
1240
+ * {
1241
+ * "id": "my-plugin",
1242
+ * "name": "My Plugin",
1243
+ * "source": "./plugins/my-plugin.json",
1244
+ * "category": "productivity",
1245
+ * "description": "A helpful plugin"
1246
+ * }
1247
+ * ]
1248
+ * }
1249
+ * ```
1250
+ *
1251
+ * @param url - The URL to download the marketplace.json from
1252
+ * @param cachePath - Local file path to save the downloaded marketplace
1253
+ * @param customHeaders - Optional custom HTTP headers for authentication
1254
+ * @param onProgress - Optional callback to report progress
1255
+ */
1256
+ async function cacheMarketplaceFromUrl(
1257
+ url: string,
1258
+ cachePath: string,
1259
+ customHeaders?: Record<string, string>,
1260
+ onProgress?: MarketplaceProgressCallback,
1261
+ ): Promise<void> {
1262
+ const fs = getFsImplementation()
1263
+
1264
+ const redactedUrl = redactUrlCredentials(url)
1265
+ safeCallProgress(onProgress, `Downloading marketplace from ${redactedUrl}`)
1266
+ logForDebugging(`Downloading marketplace from URL: ${redactedUrl}`)
1267
+ if (customHeaders && Object.keys(customHeaders).length > 0) {
1268
+ logForDebugging(
1269
+ `Using custom headers: ${jsonStringify(redactHeaders(customHeaders))}`,
1270
+ )
1271
+ }
1272
+
1273
+ const headers = {
1274
+ ...customHeaders,
1275
+ // User-Agent must come last to prevent override (for consistency with WebFetch)
1276
+ 'User-Agent': 'Claude-Code-Plugin-Manager',
1277
+ }
1278
+
1279
+ let response
1280
+ const fetchStarted = performance.now()
1281
+ try {
1282
+ response = await axios.get(url, {
1283
+ timeout: 10000,
1284
+ headers,
1285
+ })
1286
+ } catch (error) {
1287
+ logPluginFetch(
1288
+ 'marketplace_url',
1289
+ url,
1290
+ 'failure',
1291
+ performance.now() - fetchStarted,
1292
+ classifyFetchError(error),
1293
+ )
1294
+ if (axios.isAxiosError(error)) {
1295
+ if (error.code === 'ECONNREFUSED' || error.code === 'ENOTFOUND') {
1296
+ throw new Error(
1297
+ `Could not connect to ${redactedUrl}. Please check your internet connection and verify the URL is correct.\n\nTechnical details: ${error.message}`,
1298
+ )
1299
+ }
1300
+ if (error.code === 'ETIMEDOUT') {
1301
+ throw new Error(
1302
+ `Request timed out while downloading marketplace from ${redactedUrl}. The server may be slow or unreachable.\n\nTechnical details: ${error.message}`,
1303
+ )
1304
+ }
1305
+ if (error.response) {
1306
+ throw new Error(
1307
+ `HTTP ${error.response.status} error while downloading marketplace from ${redactedUrl}. The marketplace file may not exist at this URL.\n\nTechnical details: ${error.message}`,
1308
+ )
1309
+ }
1310
+ }
1311
+ throw new Error(
1312
+ `Failed to download marketplace from ${redactedUrl}: ${errorMessage(error)}`,
1313
+ )
1314
+ }
1315
+
1316
+ safeCallProgress(onProgress, 'Validating marketplace data')
1317
+ // Validate the response is a valid marketplace
1318
+ const result = PluginMarketplaceSchema().safeParse(response.data)
1319
+ if (!result.success) {
1320
+ logPluginFetch(
1321
+ 'marketplace_url',
1322
+ url,
1323
+ 'failure',
1324
+ performance.now() - fetchStarted,
1325
+ 'invalid_schema',
1326
+ )
1327
+ throw new ConfigParseError(
1328
+ `Invalid marketplace schema from URL: ${result.error.issues.map(e => `${e.path.join('.')}: ${e.message}`).join(', ')}`,
1329
+ redactedUrl,
1330
+ response.data,
1331
+ )
1332
+ }
1333
+ logPluginFetch(
1334
+ 'marketplace_url',
1335
+ url,
1336
+ 'success',
1337
+ performance.now() - fetchStarted,
1338
+ )
1339
+
1340
+ safeCallProgress(onProgress, 'Saving marketplace to cache')
1341
+ // Ensure cache directory exists
1342
+ const cacheDir = join(cachePath, '..')
1343
+ await fs.mkdir(cacheDir)
1344
+
1345
+ // Write the validated marketplace file
1346
+ writeFileSync_DEPRECATED(cachePath, jsonStringify(result.data, null, 2), {
1347
+ encoding: 'utf-8',
1348
+ flush: true,
1349
+ })
1350
+ }
1351
+
1352
+ /**
1353
+ * Generate a cache path for a marketplace source
1354
+ */
1355
+ function getCachePathForSource(source: MarketplaceSource): string {
1356
+ const tempName =
1357
+ source.source === 'github'
1358
+ ? source.repo.replace('/', '-')
1359
+ : source.source === 'npm'
1360
+ ? source.package.replace('@', '').replace('/', '-')
1361
+ : source.source === 'file'
1362
+ ? basename(source.path).replace('.json', '')
1363
+ : source.source === 'directory'
1364
+ ? basename(source.path)
1365
+ : 'temp_' + Date.now()
1366
+ return tempName
1367
+ }
1368
+
1369
+ /**
1370
+ * Parse and validate JSON file with a Zod schema
1371
+ */
1372
+ async function parseFileWithSchema<T>(
1373
+ filePath: string,
1374
+ schema: {
1375
+ safeParse: (data: unknown) => {
1376
+ success: boolean
1377
+ data?: T
1378
+ error?: {
1379
+ issues: Array<{ path: PropertyKey[]; message: string }>
1380
+ }
1381
+ }
1382
+ },
1383
+ ): Promise<T> {
1384
+ const fs = getFsImplementation()
1385
+ const content = await fs.readFile(filePath, { encoding: 'utf-8' })
1386
+ let data: unknown
1387
+ try {
1388
+ data = jsonParse(content)
1389
+ } catch (error) {
1390
+ throw new ConfigParseError(
1391
+ `Invalid JSON in ${filePath}: ${errorMessage(error)}`,
1392
+ filePath,
1393
+ content,
1394
+ )
1395
+ }
1396
+ const result = schema.safeParse(data)
1397
+ if (!result.success) {
1398
+ throw new ConfigParseError(
1399
+ `Invalid schema: ${filePath} ${result.error?.issues.map(e => `${e.path.join('.')}: ${e.message}`).join(', ')}`,
1400
+ filePath,
1401
+ data,
1402
+ )
1403
+ }
1404
+ return result.data!
1405
+ }
1406
+
1407
+ /**
1408
+ * Load and cache a marketplace from its source
1409
+ *
1410
+ * Handles different source types:
1411
+ * - URL: Downloads marketplace.json directly
1412
+ * - GitHub: Clones repo and looks for .claude-plugin/marketplace.json
1413
+ * - Git: Clones repository from git URL
1414
+ * - NPM: (Not yet implemented) Would fetch from npm package
1415
+ * - File: Reads from local filesystem
1416
+ *
1417
+ * After loading, validates the marketplace schema and renames the cache
1418
+ * to match the marketplace's actual name from the manifest.
1419
+ *
1420
+ * Cache structure:
1421
+ * ~/.claude/plugins/marketplaces/
1422
+ * ├── official-marketplace.json # From URL source
1423
+ * ├── github-marketplace/ # From GitHub/Git source
1424
+ * │ └── .claude-plugin/
1425
+ * │ └── marketplace.json
1426
+ * └── local-marketplace.json # From file source
1427
+ *
1428
+ * @param source - The marketplace source to load from
1429
+ * @param onProgress - Optional callback to report progress
1430
+ * @returns Object containing the validated marketplace and its cache path
1431
+ * @throws If marketplace file not found or validation fails
1432
+ */
1433
+ async function loadAndCacheMarketplace(
1434
+ source: MarketplaceSource,
1435
+ onProgress?: MarketplaceProgressCallback,
1436
+ ): Promise<LoadedPluginMarketplace> {
1437
+ const fs = getFsImplementation()
1438
+ const cacheDir = getMarketplacesCacheDir()
1439
+
1440
+ // Ensure cache directory exists
1441
+ await fs.mkdir(cacheDir)
1442
+
1443
+ let temporaryCachePath: string
1444
+ let marketplacePath: string
1445
+ let cleanupNeeded = false
1446
+
1447
+ // Generate a temp name for the cache path
1448
+ const tempName = getCachePathForSource(source)
1449
+
1450
+ try {
1451
+ switch (source.source) {
1452
+ case 'url': {
1453
+ // Direct URL to marketplace.json
1454
+ temporaryCachePath = join(cacheDir, `${tempName}.json`)
1455
+ cleanupNeeded = true
1456
+ await cacheMarketplaceFromUrl(
1457
+ source.url,
1458
+ temporaryCachePath,
1459
+ source.headers,
1460
+ onProgress,
1461
+ )
1462
+ marketplacePath = temporaryCachePath
1463
+ break
1464
+ }
1465
+
1466
+ case 'github': {
1467
+ // Smart SSH/HTTPS selection: check if SSH is configured before trying it
1468
+ // This avoids waiting for timeout on SSH when it's not configured
1469
+ const sshUrl = `git@github.com:${source.repo}.git`
1470
+ const httpsUrl = `https://github.com/${source.repo}.git`
1471
+ temporaryCachePath = join(cacheDir, tempName)
1472
+ cleanupNeeded = true
1473
+
1474
+ let lastError: Error | null = null
1475
+
1476
+ // Quick check if SSH is likely to work
1477
+ const sshConfigured = await isGitHubSshLikelyConfigured()
1478
+
1479
+ if (sshConfigured) {
1480
+ // SSH looks good, try it first
1481
+ safeCallProgress(onProgress, `Cloning via SSH: ${sshUrl}`)
1482
+ try {
1483
+ await cacheMarketplaceFromGit(
1484
+ sshUrl,
1485
+ temporaryCachePath,
1486
+ source.ref,
1487
+ source.sparsePaths,
1488
+ onProgress,
1489
+ )
1490
+ } catch (err) {
1491
+ lastError = toError(err)
1492
+
1493
+ // Log SSH failure for monitoring
1494
+ logError(lastError)
1495
+
1496
+ // SSH failed despite being configured, try HTTPS fallback
1497
+ safeCallProgress(
1498
+ onProgress,
1499
+ `SSH clone failed, retrying with HTTPS: ${httpsUrl}`,
1500
+ )
1501
+
1502
+ logForDebugging(
1503
+ `SSH clone failed for ${source.repo} despite SSH being configured, falling back to HTTPS`,
1504
+ { level: 'info' },
1505
+ )
1506
+
1507
+ // Clean up failed SSH attempt if it created anything
1508
+ await fs.rm(temporaryCachePath, { recursive: true, force: true })
1509
+
1510
+ // Try HTTPS
1511
+ try {
1512
+ await cacheMarketplaceFromGit(
1513
+ httpsUrl,
1514
+ temporaryCachePath,
1515
+ source.ref,
1516
+ source.sparsePaths,
1517
+ onProgress,
1518
+ )
1519
+ lastError = null // Success!
1520
+ } catch (httpsErr) {
1521
+ // HTTPS also failed - use HTTPS error as the final error
1522
+ lastError = toError(httpsErr)
1523
+
1524
+ // Log HTTPS failure for monitoring (both SSH and HTTPS failed)
1525
+ logError(lastError)
1526
+ }
1527
+ }
1528
+ } else {
1529
+ // SSH not configured, go straight to HTTPS
1530
+ safeCallProgress(
1531
+ onProgress,
1532
+ `SSH not configured, cloning via HTTPS: ${httpsUrl}`,
1533
+ )
1534
+
1535
+ logForDebugging(
1536
+ `SSH not configured for GitHub, using HTTPS for ${source.repo}`,
1537
+ { level: 'info' },
1538
+ )
1539
+
1540
+ try {
1541
+ await cacheMarketplaceFromGit(
1542
+ httpsUrl,
1543
+ temporaryCachePath,
1544
+ source.ref,
1545
+ source.sparsePaths,
1546
+ onProgress,
1547
+ )
1548
+ } catch (err) {
1549
+ lastError = toError(err)
1550
+
1551
+ // Always try SSH as fallback for ANY HTTPS failure
1552
+ // Log HTTPS failure for monitoring
1553
+ logError(lastError)
1554
+
1555
+ // HTTPS failed, try SSH as fallback
1556
+ safeCallProgress(
1557
+ onProgress,
1558
+ `HTTPS clone failed, retrying with SSH: ${sshUrl}`,
1559
+ )
1560
+
1561
+ logForDebugging(
1562
+ `HTTPS clone failed for ${source.repo} (${lastError.message}), falling back to SSH`,
1563
+ { level: 'info' },
1564
+ )
1565
+
1566
+ // Clean up failed HTTPS attempt if it created anything
1567
+ await fs.rm(temporaryCachePath, { recursive: true, force: true })
1568
+
1569
+ // Try SSH
1570
+ try {
1571
+ await cacheMarketplaceFromGit(
1572
+ sshUrl,
1573
+ temporaryCachePath,
1574
+ source.ref,
1575
+ source.sparsePaths,
1576
+ onProgress,
1577
+ )
1578
+ lastError = null // Success!
1579
+ } catch (sshErr) {
1580
+ // SSH also failed - use SSH error as the final error
1581
+ lastError = toError(sshErr)
1582
+
1583
+ // Log SSH failure for monitoring (both HTTPS and SSH failed)
1584
+ logError(lastError)
1585
+ }
1586
+ }
1587
+ }
1588
+
1589
+ // If we still have an error, throw it
1590
+ if (lastError) {
1591
+ throw lastError
1592
+ }
1593
+
1594
+ marketplacePath = join(
1595
+ temporaryCachePath,
1596
+ source.path || '.claude-plugin/marketplace.json',
1597
+ )
1598
+ break
1599
+ }
1600
+
1601
+ case 'git': {
1602
+ temporaryCachePath = join(cacheDir, tempName)
1603
+ cleanupNeeded = true
1604
+ await cacheMarketplaceFromGit(
1605
+ source.url,
1606
+ temporaryCachePath,
1607
+ source.ref,
1608
+ source.sparsePaths,
1609
+ onProgress,
1610
+ )
1611
+ marketplacePath = join(
1612
+ temporaryCachePath,
1613
+ source.path || '.claude-plugin/marketplace.json',
1614
+ )
1615
+ break
1616
+ }
1617
+
1618
+ case 'npm': {
1619
+ // TODO: Implement npm package support
1620
+ throw new Error('NPM marketplace sources not yet implemented')
1621
+ }
1622
+
1623
+ case 'file': {
1624
+ // For local files, resolve paths relative to marketplace root directory
1625
+ // File sources point to .claude-plugin/marketplace.json, so the marketplace
1626
+ // root is two directories up (parent of .claude-plugin/)
1627
+ // Resolve to absolute so error messages show the actual path checked
1628
+ // (legacy known_marketplaces.json entries may have relative paths)
1629
+ const absPath = resolve(source.path)
1630
+ marketplacePath = absPath
1631
+ temporaryCachePath = dirname(dirname(absPath))
1632
+ cleanupNeeded = false
1633
+ break
1634
+ }
1635
+
1636
+ case 'directory': {
1637
+ // For directories, look for .claude-plugin/marketplace.json
1638
+ // Resolve to absolute so error messages show the actual path checked
1639
+ // (legacy known_marketplaces.json entries may have relative paths)
1640
+ const absPath = resolve(source.path)
1641
+ marketplacePath = join(absPath, '.claude-plugin', 'marketplace.json')
1642
+ temporaryCachePath = absPath
1643
+ cleanupNeeded = false
1644
+ break
1645
+ }
1646
+
1647
+ case 'settings': {
1648
+ // Inline manifest from settings.json — no fetch. Synthesize the
1649
+ // marketplace.json on disk so getMarketplaceCacheOnly reads it
1650
+ // like any other source. The plugins array already passed
1651
+ // PluginMarketplaceEntrySchema validation when settings were parsed;
1652
+ // the post-switch parseFileWithSchema re-validates the full
1653
+ // PluginMarketplaceSchema (catches schema drift between the two).
1654
+ //
1655
+ // Writing to source.name up front means the rename below is a no-op
1656
+ // (temporaryCachePath === finalCachePath). known_marketplaces.json
1657
+ // stores this source object including the plugins array, so
1658
+ // diffMarketplaces detects settings edits via isEqual — no special
1659
+ // dirty-tracking needed.
1660
+ temporaryCachePath = join(cacheDir, source.name)
1661
+ marketplacePath = join(
1662
+ temporaryCachePath,
1663
+ '.claude-plugin',
1664
+ 'marketplace.json',
1665
+ )
1666
+ cleanupNeeded = false
1667
+ await fs.mkdir(dirname(marketplacePath))
1668
+ // No `satisfies PluginMarketplace` here: source.plugins is the narrow
1669
+ // SettingsMarketplacePlugin type (no strict/.default(), no manifest
1670
+ // fields). The parseFileWithSchema(PluginMarketplaceSchema()) call
1671
+ // below widens and validates — that's the real check.
1672
+ await writeFile(
1673
+ marketplacePath,
1674
+ jsonStringify(
1675
+ {
1676
+ name: source.name,
1677
+ owner: source.owner ?? { name: 'settings' },
1678
+ plugins: source.plugins,
1679
+ },
1680
+ null,
1681
+ 2,
1682
+ ),
1683
+ )
1684
+ break
1685
+ }
1686
+
1687
+ default:
1688
+ throw new Error(`Unsupported marketplace source type`)
1689
+ }
1690
+
1691
+ // Load and validate the marketplace
1692
+ logForDebugging(`Reading marketplace from ${marketplacePath}`)
1693
+ let marketplace: PluginMarketplace
1694
+ try {
1695
+ marketplace = await parseFileWithSchema(
1696
+ marketplacePath,
1697
+ PluginMarketplaceSchema(),
1698
+ )
1699
+ } catch (e) {
1700
+ if (isENOENT(e)) {
1701
+ throw new Error(`Marketplace file not found at ${marketplacePath}`)
1702
+ }
1703
+ throw new Error(
1704
+ `Failed to parse marketplace file at ${marketplacePath}: ${errorMessage(e)}`,
1705
+ )
1706
+ }
1707
+
1708
+ // Now rename the cache path to use the marketplace's actual name
1709
+ const finalCachePath = join(cacheDir, marketplace.name)
1710
+ // Defense-in-depth: the schema rejects path separators, .., and . in marketplace.name,
1711
+ // but verify the computed path is a strict subdirectory of cacheDir before fs.rm.
1712
+ // A malicious marketplace.json with a crafted name must never cause us to rm outside
1713
+ // cacheDir, nor rm cacheDir itself (e.g. name "." → join normalizes to cacheDir).
1714
+ const resolvedFinal = resolve(finalCachePath)
1715
+ const resolvedCacheDir = resolve(cacheDir)
1716
+ if (!resolvedFinal.startsWith(resolvedCacheDir + sep)) {
1717
+ throw new Error(
1718
+ `Marketplace name '${marketplace.name}' resolves to a path outside the cache directory`,
1719
+ )
1720
+ }
1721
+ // Don't rename if it's a local file or directory, or already has the right name
1722
+ if (
1723
+ temporaryCachePath !== finalCachePath &&
1724
+ !isLocalMarketplaceSource(source)
1725
+ ) {
1726
+ try {
1727
+ // Remove the destination if it already exists, then rename
1728
+ try {
1729
+ onProgress?.('Cleaning up old marketplace cache…')
1730
+ } catch (callbackError) {
1731
+ logForDebugging(
1732
+ `Progress callback error: ${errorMessage(callbackError)}`,
1733
+ { level: 'warn' },
1734
+ )
1735
+ }
1736
+ await fs.rm(finalCachePath, { recursive: true, force: true })
1737
+ // Rename temp cache to final name
1738
+ await fs.rename(temporaryCachePath, finalCachePath)
1739
+ temporaryCachePath = finalCachePath
1740
+ cleanupNeeded = false // Successfully renamed, no cleanup needed
1741
+ } catch (error) {
1742
+ const errorMsg = errorMessage(error)
1743
+ throw new Error(
1744
+ `Failed to finalize marketplace cache. Please manually delete the directory at ${finalCachePath} if it exists and try again.\n\nTechnical details: ${errorMsg}`,
1745
+ )
1746
+ }
1747
+ }
1748
+
1749
+ return { marketplace, cachePath: temporaryCachePath }
1750
+ } catch (error) {
1751
+ // Clean up any temporary files/directories on error
1752
+ if (
1753
+ cleanupNeeded &&
1754
+ temporaryCachePath! &&
1755
+ !isLocalMarketplaceSource(source)
1756
+ ) {
1757
+ try {
1758
+ await fs.rm(temporaryCachePath!, { recursive: true, force: true })
1759
+ } catch (cleanupError) {
1760
+ logForDebugging(
1761
+ `Warning: Failed to clean up temporary marketplace cache at ${temporaryCachePath}: ${errorMessage(cleanupError)}`,
1762
+ { level: 'warn' },
1763
+ )
1764
+ }
1765
+ }
1766
+ throw error
1767
+ }
1768
+ }
1769
+
1770
+ /**
1771
+ * Add a marketplace source to the known marketplaces
1772
+ *
1773
+ * The marketplace is fetched, validated, and cached locally.
1774
+ * The configuration is saved to ~/.claude/plugins/known_marketplaces.json.
1775
+ *
1776
+ * @param source - MarketplaceSource object representing the marketplace source.
1777
+ * Callers should parse user input into MarketplaceSource format
1778
+ * (see AddMarketplace.parseMarketplaceInput for handling shortcuts like "owner/repo").
1779
+ * @param onProgress - Optional callback for progress updates during marketplace installation
1780
+ * @throws If source format is invalid or marketplace cannot be loaded
1781
+ */
1782
+ export async function addMarketplaceSource(
1783
+ source: MarketplaceSource,
1784
+ onProgress?: MarketplaceProgressCallback,
1785
+ ): Promise<{
1786
+ name: string
1787
+ alreadyMaterialized: boolean
1788
+ resolvedSource: MarketplaceSource
1789
+ }> {
1790
+ // Resolve relative directory/file paths to absolute so state is cwd-independent
1791
+ let resolvedSource = source
1792
+ if (isLocalMarketplaceSource(source) && !isAbsolute(source.path)) {
1793
+ resolvedSource = { ...source, path: resolve(source.path) }
1794
+ }
1795
+
1796
+ // Check policy FIRST, before any network/filesystem operations
1797
+ // This prevents downloading/cloning when the source is blocked
1798
+ if (!isSourceAllowedByPolicy(resolvedSource)) {
1799
+ // Check if explicitly blocked vs not in allowlist for better error messages
1800
+ if (isSourceInBlocklist(resolvedSource)) {
1801
+ throw new Error(
1802
+ `Marketplace source '${formatSourceForDisplay(resolvedSource)}' is blocked by enterprise policy.`,
1803
+ )
1804
+ }
1805
+ // Not in allowlist - build helpful error message
1806
+ const allowlist = getStrictKnownMarketplaces() || []
1807
+ const hostPatterns = getHostPatternsFromAllowlist()
1808
+ const sourceHost = extractHostFromSource(resolvedSource)
1809
+
1810
+ let errorMessage = `Marketplace source '${formatSourceForDisplay(resolvedSource)}'`
1811
+ if (sourceHost) {
1812
+ errorMessage += ` (${sourceHost})`
1813
+ }
1814
+ errorMessage += ' is blocked by enterprise policy.'
1815
+
1816
+ if (allowlist.length > 0) {
1817
+ errorMessage += ` Allowed sources: ${allowlist.map(s => formatSourceForDisplay(s)).join(', ')}`
1818
+ } else {
1819
+ errorMessage += ' No external marketplaces are allowed.'
1820
+ }
1821
+
1822
+ // If source is a github shorthand and there are hostPatterns, suggest using full URL
1823
+ if (resolvedSource.source === 'github' && hostPatterns.length > 0) {
1824
+ errorMessage +=
1825
+ `\n\nTip: The shorthand "${resolvedSource.repo}" assumes github.com. ` +
1826
+ `For internal GitHub Enterprise, use the full URL:\n` +
1827
+ ` git@your-github-host.com:${resolvedSource.repo}.git`
1828
+ }
1829
+
1830
+ throw new Error(errorMessage)
1831
+ }
1832
+
1833
+ // Source-idempotency: if this exact source already exists, skip clone
1834
+ const existingConfig = await loadKnownMarketplacesConfig()
1835
+ for (const [existingName, existingEntry] of Object.entries(existingConfig)) {
1836
+ if (isEqual(existingEntry.source, resolvedSource)) {
1837
+ logForDebugging(
1838
+ `Source already materialized as '${existingName}', skipping clone`,
1839
+ )
1840
+ return { name: existingName, alreadyMaterialized: true, resolvedSource }
1841
+ }
1842
+ }
1843
+
1844
+ // Load and cache the marketplace to validate it and get its name
1845
+ const { marketplace, cachePath } = await loadAndCacheMarketplace(
1846
+ resolvedSource,
1847
+ onProgress,
1848
+ )
1849
+
1850
+ // Validate that reserved names come from official sources
1851
+ const sourceValidationError = validateOfficialNameSource(
1852
+ marketplace.name,
1853
+ resolvedSource,
1854
+ )
1855
+ if (sourceValidationError) {
1856
+ throw new Error(sourceValidationError)
1857
+ }
1858
+
1859
+ // Name collision with different source: overwrite (settings intent wins).
1860
+ // Seed-managed entries are admin-controlled and cannot be overwritten.
1861
+ // Re-read config after clone (may take a while; another process may have written).
1862
+ const config = await loadKnownMarketplacesConfig()
1863
+ const oldEntry = config[marketplace.name]
1864
+ if (oldEntry) {
1865
+ const seedDir = seedDirFor(oldEntry.installLocation)
1866
+ if (seedDir) {
1867
+ throw new Error(
1868
+ `Marketplace '${marketplace.name}' is seed-managed (${seedDir}). ` +
1869
+ `To use a different source, ask your admin to update the seed, ` +
1870
+ `or use a different marketplace name.`,
1871
+ )
1872
+ }
1873
+ logForDebugging(
1874
+ `Marketplace '${marketplace.name}' exists with different source — overwriting`,
1875
+ )
1876
+ // Clean up the old cache if it's not a user-owned local path AND it
1877
+ // actually differs from the new cachePath. loadAndCacheMarketplace writes
1878
+ // to cachePath BEFORE we get here — rm-ing the same dir deletes the fresh
1879
+ // write. Settings sources always land on the same dir (name → path);
1880
+ // git sources hit this latently when the source repo changes but the
1881
+ // fetched marketplace.json declares the same name. Only rm when locations
1882
+ // genuinely differ (the only case where there's a stale dir to clean).
1883
+ //
1884
+ // Defensively validate the stored path before rm: a corrupted
1885
+ // installLocation (gh-32793, gh-32661) could point at the user's project
1886
+ // dir. If it's outside the cache dir, skip cleanup — the stale dir (if
1887
+ // any) is harmless, and blocking the re-add would prevent the user from
1888
+ // fixing the corruption.
1889
+ if (!isLocalMarketplaceSource(oldEntry.source)) {
1890
+ const cacheDir = resolve(getMarketplacesCacheDir())
1891
+ const resolvedOld = resolve(oldEntry.installLocation)
1892
+ const resolvedNew = resolve(cachePath)
1893
+ if (resolvedOld === resolvedNew) {
1894
+ // Same dir — loadAndCacheMarketplace already overwrote in place.
1895
+ // Nothing to clean.
1896
+ } else if (
1897
+ resolvedOld === cacheDir ||
1898
+ resolvedOld.startsWith(cacheDir + sep)
1899
+ ) {
1900
+ const fs = getFsImplementation()
1901
+ await fs.rm(oldEntry.installLocation, { recursive: true, force: true })
1902
+ } else {
1903
+ logForDebugging(
1904
+ `Skipping cleanup of old installLocation (${oldEntry.installLocation}) — ` +
1905
+ `outside ${cacheDir}. The path is corrupted; leaving it alone and ` +
1906
+ `overwriting the config entry.`,
1907
+ { level: 'warn' },
1908
+ )
1909
+ }
1910
+ }
1911
+ }
1912
+
1913
+ // Update config using the marketplace's actual name
1914
+ config[marketplace.name] = {
1915
+ source: resolvedSource,
1916
+ installLocation: cachePath,
1917
+ lastUpdated: new Date().toISOString(),
1918
+ }
1919
+ await saveKnownMarketplacesConfig(config)
1920
+
1921
+ logForDebugging(`Added marketplace source: ${marketplace.name}`)
1922
+
1923
+ return { name: marketplace.name, alreadyMaterialized: false, resolvedSource }
1924
+ }
1925
+
1926
+ /**
1927
+ * Remove a marketplace source from known marketplaces
1928
+ *
1929
+ * Removes the marketplace configuration and cleans up cached files.
1930
+ * Deletes both directory caches (for git sources) and file caches (for URL sources).
1931
+ * Also cleans up the marketplace from settings.json (extraKnownMarketplaces) and
1932
+ * removes related plugin entries from enabledPlugins.
1933
+ *
1934
+ * @param name - The marketplace name to remove
1935
+ * @throws If marketplace with given name is not found
1936
+ */
1937
+ export async function removeMarketplaceSource(name: string): Promise<void> {
1938
+ const config = await loadKnownMarketplacesConfig()
1939
+
1940
+ if (!config[name]) {
1941
+ throw new Error(`Marketplace '${name}' not found`)
1942
+ }
1943
+
1944
+ // Seed-registered marketplaces are admin-baked into the container — removing
1945
+ // them is a category error. They'd resurrect on next startup anyway. Guide
1946
+ // the user to the right action instead.
1947
+ const entry = config[name]
1948
+ const seedDir = seedDirFor(entry.installLocation)
1949
+ if (seedDir) {
1950
+ throw new Error(
1951
+ `Marketplace '${name}' is registered from the read-only seed directory ` +
1952
+ `(${seedDir}) and will be re-registered on next startup. ` +
1953
+ `To stop using its plugins: claude plugin disable <plugin>@${name}`,
1954
+ )
1955
+ }
1956
+
1957
+ // Remove from config
1958
+ delete config[name]
1959
+ await saveKnownMarketplacesConfig(config)
1960
+
1961
+ // Clean up cached files (both directory and JSON formats)
1962
+ const fs = getFsImplementation()
1963
+ const cacheDir = getMarketplacesCacheDir()
1964
+ const cachePath = join(cacheDir, name)
1965
+ await fs.rm(cachePath, { recursive: true, force: true })
1966
+ const jsonCachePath = join(cacheDir, `${name}.json`)
1967
+ await fs.rm(jsonCachePath, { force: true })
1968
+
1969
+ // Clean up settings.json - remove marketplace from extraKnownMarketplaces
1970
+ // and remove related plugin entries from enabledPlugins
1971
+
1972
+ // Check each editable settings source
1973
+ const editableSources: Array<
1974
+ 'userSettings' | 'projectSettings' | 'localSettings'
1975
+ > = ['userSettings', 'projectSettings', 'localSettings']
1976
+
1977
+ for (const source of editableSources) {
1978
+ const settings = getSettingsForSource(source)
1979
+ if (!settings) continue
1980
+
1981
+ let needsUpdate = false
1982
+ const updates: {
1983
+ extraKnownMarketplaces?: typeof settings.extraKnownMarketplaces
1984
+ enabledPlugins?: typeof settings.enabledPlugins
1985
+ } = {}
1986
+
1987
+ // Remove from extraKnownMarketplaces if present
1988
+ if (settings.extraKnownMarketplaces?.[name]) {
1989
+ const updatedMarketplaces: Partial<
1990
+ SettingsJson['extraKnownMarketplaces']
1991
+ > = { ...settings.extraKnownMarketplaces }
1992
+ // Use undefined values (NOT delete) to signal key removal via mergeWith
1993
+ updatedMarketplaces[name] = undefined
1994
+ updates.extraKnownMarketplaces =
1995
+ updatedMarketplaces as SettingsJson['extraKnownMarketplaces']
1996
+ needsUpdate = true
1997
+ }
1998
+
1999
+ // Remove related plugins from enabledPlugins (format: "plugin@marketplace")
2000
+ if (settings.enabledPlugins) {
2001
+ const marketplaceSuffix = `@${name}`
2002
+ const updatedPlugins = { ...settings.enabledPlugins }
2003
+ let removedPlugins = false
2004
+
2005
+ for (const pluginId in updatedPlugins) {
2006
+ if (pluginId.endsWith(marketplaceSuffix)) {
2007
+ updatedPlugins[pluginId] = undefined
2008
+ removedPlugins = true
2009
+ }
2010
+ }
2011
+
2012
+ if (removedPlugins) {
2013
+ updates.enabledPlugins = updatedPlugins
2014
+ needsUpdate = true
2015
+ }
2016
+ }
2017
+
2018
+ // Update settings if changes were made
2019
+ if (needsUpdate) {
2020
+ const result = updateSettingsForSource(source, updates)
2021
+ if (result.error) {
2022
+ logError(result.error)
2023
+ logForDebugging(
2024
+ `Failed to clean up marketplace '${name}' from ${source} settings: ${result.error.message}`,
2025
+ )
2026
+ } else {
2027
+ logForDebugging(
2028
+ `Cleaned up marketplace '${name}' from ${source} settings`,
2029
+ )
2030
+ }
2031
+ }
2032
+ }
2033
+
2034
+ // Remove plugins from installed_plugins.json and mark orphaned paths.
2035
+ // Also wipe their stored options/secrets — after marketplace removal
2036
+ // zero installations remain, same "last scope gone" condition as
2037
+ // uninstallPluginOp.
2038
+ const { orphanedPaths, removedPluginIds } =
2039
+ removeAllPluginsForMarketplace(name)
2040
+ for (const installPath of orphanedPaths) {
2041
+ await markPluginVersionOrphaned(installPath)
2042
+ }
2043
+ for (const pluginId of removedPluginIds) {
2044
+ deletePluginOptions(pluginId)
2045
+ await deletePluginDataDir(pluginId)
2046
+ }
2047
+
2048
+ logForDebugging(`Removed marketplace source: ${name}`)
2049
+ }
2050
+
2051
+ /**
2052
+ * Read a cached marketplace from disk without updating it
2053
+ *
2054
+ * @param installLocation - Path to the cached marketplace
2055
+ * @returns The marketplace object
2056
+ * @throws If marketplace file not found or invalid
2057
+ */
2058
+ async function readCachedMarketplace(
2059
+ installLocation: string,
2060
+ ): Promise<PluginMarketplace> {
2061
+ // For git-sourced directories, the manifest lives at .claude-plugin/marketplace.json.
2062
+ // For url/file/directory sources it is the installLocation itself.
2063
+ // Try the nested path first; fall back to installLocation when it is a plain file
2064
+ // (ENOTDIR) or the nested file is simply missing (ENOENT).
2065
+ const nestedPath = join(installLocation, '.claude-plugin', 'marketplace.json')
2066
+ try {
2067
+ return await parseFileWithSchema(nestedPath, PluginMarketplaceSchema())
2068
+ } catch (e) {
2069
+ if (e instanceof ConfigParseError) throw e
2070
+ const code = getErrnoCode(e)
2071
+ if (code !== 'ENOENT' && code !== 'ENOTDIR') throw e
2072
+ }
2073
+ return await parseFileWithSchema(installLocation, PluginMarketplaceSchema())
2074
+ }
2075
+
2076
+ /**
2077
+ * Get a specific marketplace by name from cache only (no network).
2078
+ * Returns null if cache is missing or corrupted.
2079
+ * Use this for startup paths that should never block on network.
2080
+ */
2081
+ export async function getMarketplaceCacheOnly(
2082
+ name: string,
2083
+ ): Promise<PluginMarketplace | null> {
2084
+ const fs = getFsImplementation()
2085
+ const configFile = getKnownMarketplacesFile()
2086
+
2087
+ try {
2088
+ const content = await fs.readFile(configFile, { encoding: 'utf-8' })
2089
+ const config = jsonParse(content) as KnownMarketplacesConfig
2090
+ const entry = config[name]
2091
+
2092
+ if (!entry) {
2093
+ return null
2094
+ }
2095
+
2096
+ return await readCachedMarketplace(entry.installLocation)
2097
+ } catch (error) {
2098
+ if (isENOENT(error)) {
2099
+ return null
2100
+ }
2101
+ logForDebugging(
2102
+ `Failed to read cached marketplace ${name}: ${errorMessage(error)}`,
2103
+ { level: 'warn' },
2104
+ )
2105
+ return null
2106
+ }
2107
+ }
2108
+
2109
+ /**
2110
+ * Get a specific marketplace by name
2111
+ *
2112
+ * First attempts to read from cache. Only fetches from source if:
2113
+ * - No cached version exists
2114
+ * - Cache is invalid/corrupted
2115
+ *
2116
+ * This avoids unnecessary network/git operations on every access.
2117
+ * Use refreshMarketplace() to explicitly update from source.
2118
+ *
2119
+ * @param name - The marketplace name to fetch
2120
+ * @returns The marketplace object or null if not found/failed
2121
+ */
2122
+ export const getMarketplace = memoize(
2123
+ async (name: string): Promise<PluginMarketplace> => {
2124
+ const config = await loadKnownMarketplacesConfig()
2125
+ const entry = config[name]
2126
+
2127
+ if (!entry) {
2128
+ throw new Error(
2129
+ `Marketplace '${name}' not found in configuration. Available marketplaces: ${Object.keys(config).join(', ')}`,
2130
+ )
2131
+ }
2132
+
2133
+ // Legacy entries (pre-#19708) may have relative paths in global config.
2134
+ // These are meaningless outside the project that wrote them — resolving
2135
+ // against process.cwd() produces the wrong path. Give actionable guidance
2136
+ // instead of a misleading ENOENT.
2137
+ if (
2138
+ isLocalMarketplaceSource(entry.source) &&
2139
+ !isAbsolute(entry.source.path)
2140
+ ) {
2141
+ throw new Error(
2142
+ `Marketplace "${name}" has a relative source path (${entry.source.path}) ` +
2143
+ `in known_marketplaces.json — this is stale state from an older ` +
2144
+ `Claude Code version. Run 'claude marketplace remove ${name}' and ` +
2145
+ `re-add it from the original project directory.`,
2146
+ )
2147
+ }
2148
+
2149
+ // Try to read from disk cache
2150
+ try {
2151
+ return await readCachedMarketplace(entry.installLocation)
2152
+ } catch (error) {
2153
+ // Log cache corruption before re-fetching
2154
+ logForDebugging(
2155
+ `Cache corrupted or missing for marketplace ${name}, re-fetching from source: ${errorMessage(error)}`,
2156
+ {
2157
+ level: 'warn',
2158
+ },
2159
+ )
2160
+ }
2161
+
2162
+ // Cache doesn't exist or is invalid, fetch from source
2163
+ let marketplace: PluginMarketplace
2164
+ try {
2165
+ ;({ marketplace } = await loadAndCacheMarketplace(entry.source))
2166
+ } catch (error) {
2167
+ throw new Error(
2168
+ `Failed to load marketplace "${name}" from source (${entry.source.source}): ${errorMessage(error)}`,
2169
+ )
2170
+ }
2171
+
2172
+ // Update lastUpdated only when we actually fetch
2173
+ config[name]!.lastUpdated = new Date().toISOString()
2174
+ await saveKnownMarketplacesConfig(config)
2175
+
2176
+ return marketplace
2177
+ },
2178
+ )
2179
+
2180
+ /**
2181
+ * Get plugin by ID from cache only (no network calls).
2182
+ * Returns null if marketplace cache is missing or corrupted.
2183
+ * Use this for startup paths that should never block on network.
2184
+ *
2185
+ * @param pluginId - The plugin ID in format "name@marketplace"
2186
+ * @returns The plugin entry or null if not found/cache missing
2187
+ */
2188
+ export async function getPluginByIdCacheOnly(pluginId: string): Promise<{
2189
+ entry: PluginMarketplaceEntry
2190
+ marketplaceInstallLocation: string
2191
+ } | null> {
2192
+ const { name: pluginName, marketplace: marketplaceName } =
2193
+ parsePluginIdentifier(pluginId)
2194
+ if (!pluginName || !marketplaceName) {
2195
+ return null
2196
+ }
2197
+
2198
+ const fs = getFsImplementation()
2199
+ const configFile = getKnownMarketplacesFile()
2200
+
2201
+ try {
2202
+ const content = await fs.readFile(configFile, { encoding: 'utf-8' })
2203
+ const config = jsonParse(content) as KnownMarketplacesConfig
2204
+ const marketplaceConfig = config[marketplaceName]
2205
+
2206
+ if (!marketplaceConfig) {
2207
+ return null
2208
+ }
2209
+
2210
+ const marketplace = await getMarketplaceCacheOnly(marketplaceName)
2211
+ if (!marketplace) {
2212
+ return null
2213
+ }
2214
+
2215
+ const plugin = marketplace.plugins.find(p => p.name === pluginName)
2216
+ if (!plugin) {
2217
+ return null
2218
+ }
2219
+
2220
+ return {
2221
+ entry: plugin,
2222
+ marketplaceInstallLocation: marketplaceConfig.installLocation,
2223
+ }
2224
+ } catch {
2225
+ return null
2226
+ }
2227
+ }
2228
+
2229
+ /**
2230
+ * Get plugin by ID from a specific marketplace
2231
+ *
2232
+ * First tries cache-only lookup. If cache is missing/corrupted,
2233
+ * falls back to fetching from source.
2234
+ *
2235
+ * @param pluginId - The plugin ID in format "name@marketplace"
2236
+ * @returns The plugin entry or null if not found
2237
+ */
2238
+ export async function getPluginById(pluginId: string): Promise<{
2239
+ entry: PluginMarketplaceEntry
2240
+ marketplaceInstallLocation: string
2241
+ } | null> {
2242
+ // Try cache-only first (fast path)
2243
+ const cached = await getPluginByIdCacheOnly(pluginId)
2244
+ if (cached) {
2245
+ return cached
2246
+ }
2247
+
2248
+ // Cache miss - try fetching from source
2249
+ const { name: pluginName, marketplace: marketplaceName } =
2250
+ parsePluginIdentifier(pluginId)
2251
+ if (!pluginName || !marketplaceName) {
2252
+ return null
2253
+ }
2254
+
2255
+ try {
2256
+ const config = await loadKnownMarketplacesConfig()
2257
+ const marketplaceConfig = config[marketplaceName]
2258
+ if (!marketplaceConfig) {
2259
+ return null
2260
+ }
2261
+
2262
+ const marketplace = await getMarketplace(marketplaceName)
2263
+ const plugin = marketplace.plugins.find(p => p.name === pluginName)
2264
+
2265
+ if (!plugin) {
2266
+ return null
2267
+ }
2268
+
2269
+ return {
2270
+ entry: plugin,
2271
+ marketplaceInstallLocation: marketplaceConfig.installLocation,
2272
+ }
2273
+ } catch (error) {
2274
+ logForDebugging(
2275
+ `Could not find plugin ${pluginId}: ${errorMessage(error)}`,
2276
+ { level: 'debug' },
2277
+ )
2278
+ return null
2279
+ }
2280
+ }
2281
+
2282
+ /**
2283
+ * Refresh all marketplace caches
2284
+ *
2285
+ * Updates all configured marketplaces from their sources.
2286
+ * Continues refreshing even if some marketplaces fail.
2287
+ * Updates lastUpdated timestamps for successful refreshes.
2288
+ *
2289
+ * This is useful for:
2290
+ * - Periodic updates to get new plugins
2291
+ * - Syncing after network connectivity is restored
2292
+ * - Ensuring caches are up-to-date before browsing
2293
+ *
2294
+ * @returns Promise that resolves when all refresh attempts complete
2295
+ */
2296
+ export async function refreshAllMarketplaces(): Promise<void> {
2297
+ const config = await loadKnownMarketplacesConfig()
2298
+
2299
+ for (const [name, entry] of Object.entries(config)) {
2300
+ // Seed-managed marketplaces are controlled by the seed image — refreshing
2301
+ // them is pointless (registerSeedMarketplaces overwrites on next startup).
2302
+ if (seedDirFor(entry.installLocation)) {
2303
+ logForDebugging(
2304
+ `Skipping seed-managed marketplace '${name}' in bulk refresh`,
2305
+ )
2306
+ continue
2307
+ }
2308
+ // settings-sourced marketplaces have no upstream — see refreshMarketplace.
2309
+ if (entry.source.source === 'settings') {
2310
+ continue
2311
+ }
2312
+ // inc-5046: same GCS intercept as refreshMarketplace() — bulk update
2313
+ // hits this path on `claude plugin marketplace update` (no name arg).
2314
+ if (name === OFFICIAL_MARKETPLACE_NAME) {
2315
+ const sha = await fetchOfficialMarketplaceFromGcs(
2316
+ entry.installLocation,
2317
+ getMarketplacesCacheDir(),
2318
+ )
2319
+ if (sha !== null) {
2320
+ config[name]!.lastUpdated = new Date().toISOString()
2321
+ continue
2322
+ }
2323
+ if (
2324
+ !getFeatureValue_CACHED_MAY_BE_STALE(
2325
+ 'tengu_plugin_official_mkt_git_fallback',
2326
+ true,
2327
+ )
2328
+ ) {
2329
+ logForDebugging(
2330
+ `Skipping official marketplace bulk refresh: GCS failed, git fallback disabled`,
2331
+ )
2332
+ continue
2333
+ }
2334
+ // fall through to git
2335
+ }
2336
+ try {
2337
+ const { cachePath } = await loadAndCacheMarketplace(entry.source)
2338
+ config[name]!.lastUpdated = new Date().toISOString()
2339
+ config[name]!.installLocation = cachePath
2340
+ } catch (error) {
2341
+ logForDebugging(
2342
+ `Failed to refresh marketplace ${name}: ${errorMessage(error)}`,
2343
+ {
2344
+ level: 'error',
2345
+ },
2346
+ )
2347
+ }
2348
+ }
2349
+
2350
+ await saveKnownMarketplacesConfig(config)
2351
+ }
2352
+
2353
+ /**
2354
+ * Refresh a single marketplace cache
2355
+ *
2356
+ * Updates a specific marketplace from its source by doing an in-place update.
2357
+ * For git sources, runs git pull in the existing directory.
2358
+ * For URL sources, re-downloads to the existing file.
2359
+ * Clears the memoization cache and updates the lastUpdated timestamp.
2360
+ *
2361
+ * @param name - The name of the marketplace to refresh
2362
+ * @param onProgress - Optional callback to report progress
2363
+ * @throws If marketplace not found or refresh fails
2364
+ */
2365
+ export async function refreshMarketplace(
2366
+ name: string,
2367
+ onProgress?: MarketplaceProgressCallback,
2368
+ options?: { disableCredentialHelper?: boolean },
2369
+ ): Promise<void> {
2370
+ const config = await loadKnownMarketplacesConfig()
2371
+ const entry = config[name]
2372
+
2373
+ if (!entry) {
2374
+ throw new Error(
2375
+ `Marketplace '${name}' not found. Available marketplaces: ${Object.keys(config).join(', ')}`,
2376
+ )
2377
+ }
2378
+
2379
+ // Clear the memoization cache for this specific marketplace
2380
+ getMarketplace.cache?.delete?.(name)
2381
+
2382
+ // settings-sourced marketplaces have no upstream to pull. Edits to the
2383
+ // inline plugins array surface as sourceChanged in the reconciler, which
2384
+ // re-materializes via addMarketplaceSource — refresh is not the vehicle.
2385
+ if (entry.source.source === 'settings') {
2386
+ logForDebugging(
2387
+ `Skipping refresh for settings-sourced marketplace '${name}' — no upstream`,
2388
+ )
2389
+ return
2390
+ }
2391
+
2392
+ try {
2393
+ // For updates, use the existing installLocation directly (in-place update)
2394
+ const installLocation = entry.installLocation
2395
+ const source = entry.source
2396
+
2397
+ // Seed-managed marketplaces are controlled by the seed image. Refreshing
2398
+ // would be pointless — registerSeedMarketplaces() overwrites installLocation
2399
+ // back to seed on next startup. Error with guidance instead.
2400
+ const seedDir = seedDirFor(installLocation)
2401
+ if (seedDir) {
2402
+ throw new Error(
2403
+ `Marketplace '${name}' is seed-managed (${seedDir}) and its content is ` +
2404
+ `controlled by the seed image. To update: ask your admin to update the seed.`,
2405
+ )
2406
+ }
2407
+
2408
+ // For remote sources (github/git/url), installLocation must be inside the
2409
+ // marketplaces cache dir. A corrupted value (gh-32793, gh-32661 — e.g.
2410
+ // Windows path read on WSL, literal tilde, manual edit) can point at the
2411
+ // user's project. cacheMarketplaceFromGit would then run git ops with that
2412
+ // cwd (git walks up to the user's .git) and fs.rm it on pull failure.
2413
+ // Refuse instead of auto-fixing so the user knows their state is corrupted.
2414
+ if (!isLocalMarketplaceSource(source)) {
2415
+ const cacheDir = resolve(getMarketplacesCacheDir())
2416
+ const resolvedLoc = resolve(installLocation)
2417
+ if (resolvedLoc !== cacheDir && !resolvedLoc.startsWith(cacheDir + sep)) {
2418
+ throw new Error(
2419
+ `Marketplace '${name}' has a corrupted installLocation ` +
2420
+ `(${installLocation}) — expected a path inside ${cacheDir}. ` +
2421
+ `This can happen after cross-platform path writes or manual edits ` +
2422
+ `to known_marketplaces.json. ` +
2423
+ `Run: claude plugin marketplace remove "${name}" and re-add it.`,
2424
+ )
2425
+ }
2426
+ }
2427
+
2428
+ // inc-5046: official marketplace fetches from a GCS mirror instead of
2429
+ // git-cloning GitHub. Special-cased by NAME (not a new source type) so
2430
+ // no data migration is needed — existing known_marketplaces.json entries
2431
+ // still say source:'github', which is true (GCS is a mirror).
2432
+ if (name === OFFICIAL_MARKETPLACE_NAME) {
2433
+ const sha = await fetchOfficialMarketplaceFromGcs(
2434
+ installLocation,
2435
+ getMarketplacesCacheDir(),
2436
+ )
2437
+ if (sha !== null) {
2438
+ config[name] = { ...entry, lastUpdated: new Date().toISOString() }
2439
+ await saveKnownMarketplacesConfig(config)
2440
+ return
2441
+ }
2442
+ // GCS failed — fall through to git ONLY if the kill-switch allows.
2443
+ // Default true (backend write perms are pending as of inc-5046); flip
2444
+ // to false via GrowthBook once the backend is confirmed live so new
2445
+ // clients NEVER hit GitHub for the official marketplace.
2446
+ if (
2447
+ !getFeatureValue_CACHED_MAY_BE_STALE(
2448
+ 'tengu_plugin_official_mkt_git_fallback',
2449
+ true,
2450
+ )
2451
+ ) {
2452
+ // Throw, don't return — every other failure path in this function
2453
+ // throws, and callers like ManageMarketplaces.tsx:259 increment
2454
+ // updatedCount on any non-throwing return. A silent return would
2455
+ // report "Updated 1 marketplace" when nothing was refreshed.
2456
+ throw new Error(
2457
+ 'Official marketplace GCS fetch failed and git fallback is disabled',
2458
+ )
2459
+ }
2460
+ logForDebugging('Official marketplace GCS failed; falling back to git', {
2461
+ level: 'warn',
2462
+ })
2463
+ // ...falls through to source.source === 'github' branch below
2464
+ }
2465
+
2466
+ // Update based on source type
2467
+ if (source.source === 'github' || source.source === 'git') {
2468
+ // Git sources: do in-place git pull
2469
+ if (source.source === 'github') {
2470
+ // Same SSH/HTTPS fallback as loadAndCacheMarketplace: if the pull
2471
+ // succeeds the remote URL in .git/config is used, but a re-clone
2472
+ // needs a URL — pick the right protocol up-front and fall back.
2473
+ const sshUrl = `git@github.com:${source.repo}.git`
2474
+ const httpsUrl = `https://github.com/${source.repo}.git`
2475
+
2476
+ if (isEnvTruthy(process.env.CLAUDE_CODE_REMOTE)) {
2477
+ // CCR: always HTTPS (no SSH keys available)
2478
+ await cacheMarketplaceFromGit(
2479
+ httpsUrl,
2480
+ installLocation,
2481
+ source.ref,
2482
+ source.sparsePaths,
2483
+ onProgress,
2484
+ options,
2485
+ )
2486
+ } else {
2487
+ const sshConfigured = await isGitHubSshLikelyConfigured()
2488
+ const primaryUrl = sshConfigured ? sshUrl : httpsUrl
2489
+ const fallbackUrl = sshConfigured ? httpsUrl : sshUrl
2490
+
2491
+ try {
2492
+ await cacheMarketplaceFromGit(
2493
+ primaryUrl,
2494
+ installLocation,
2495
+ source.ref,
2496
+ source.sparsePaths,
2497
+ onProgress,
2498
+ options,
2499
+ )
2500
+ } catch {
2501
+ logForDebugging(
2502
+ `Marketplace refresh failed with ${sshConfigured ? 'SSH' : 'HTTPS'} for ${source.repo}, falling back to ${sshConfigured ? 'HTTPS' : 'SSH'}`,
2503
+ { level: 'info' },
2504
+ )
2505
+ await cacheMarketplaceFromGit(
2506
+ fallbackUrl,
2507
+ installLocation,
2508
+ source.ref,
2509
+ source.sparsePaths,
2510
+ onProgress,
2511
+ options,
2512
+ )
2513
+ }
2514
+ }
2515
+ } else {
2516
+ // Explicit git URL: use as-is (no fallback available)
2517
+ await cacheMarketplaceFromGit(
2518
+ source.url,
2519
+ installLocation,
2520
+ source.ref,
2521
+ source.sparsePaths,
2522
+ onProgress,
2523
+ options,
2524
+ )
2525
+ }
2526
+ // Validate that marketplace.json still exists after update
2527
+ // The repo may have been restructured or deprecated
2528
+ try {
2529
+ await readCachedMarketplace(installLocation)
2530
+ } catch {
2531
+ const sourceDisplay =
2532
+ source.source === 'github'
2533
+ ? source.repo
2534
+ : redactUrlCredentials(source.url)
2535
+ const reason =
2536
+ name === 'claude-code-plugins'
2537
+ ? `We've deprecated "claude-code-plugins" in favor of "claude-plugins-official".`
2538
+ : `This marketplace may have been deprecated or moved to a new location.`
2539
+ throw new Error(
2540
+ `The marketplace.json file is no longer present in this repository.\n\n` +
2541
+ `${reason}\n` +
2542
+ `Source: ${sourceDisplay}\n\n` +
2543
+ `You can remove this marketplace with: claude plugin marketplace remove "${name}"`,
2544
+ )
2545
+ }
2546
+ } else if (source.source === 'url') {
2547
+ // URL sources: re-download to existing file
2548
+ await cacheMarketplaceFromUrl(
2549
+ source.url,
2550
+ installLocation,
2551
+ source.headers,
2552
+ onProgress,
2553
+ )
2554
+ } else if (isLocalMarketplaceSource(source)) {
2555
+ // Local sources: no remote to update from, but validate the file still exists and is valid
2556
+ safeCallProgress(onProgress, 'Validating local marketplace')
2557
+ // Read and validate to ensure the marketplace file is still valid
2558
+ await readCachedMarketplace(installLocation)
2559
+ } else {
2560
+ throw new Error(`Unsupported marketplace source type for refresh`)
2561
+ }
2562
+
2563
+ // Update lastUpdated timestamp
2564
+ config[name]!.lastUpdated = new Date().toISOString()
2565
+ await saveKnownMarketplacesConfig(config)
2566
+
2567
+ logForDebugging(`Successfully refreshed marketplace: ${name}`)
2568
+ } catch (error) {
2569
+ const errorMessage = error instanceof Error ? error.message : String(error)
2570
+ logForDebugging(`Failed to refresh marketplace ${name}: ${errorMessage}`, {
2571
+ level: 'error',
2572
+ })
2573
+ throw new Error(`Failed to refresh marketplace '${name}': ${errorMessage}`)
2574
+ }
2575
+ }
2576
+
2577
+ /**
2578
+ * Set the autoUpdate flag for a marketplace
2579
+ *
2580
+ * When autoUpdate is enabled, the marketplace and its installed plugins
2581
+ * will be automatically updated on startup.
2582
+ *
2583
+ * @param name - The name of the marketplace to update
2584
+ * @param autoUpdate - Whether to enable auto-update
2585
+ * @throws If marketplace not found
2586
+ */
2587
+ export async function setMarketplaceAutoUpdate(
2588
+ name: string,
2589
+ autoUpdate: boolean,
2590
+ ): Promise<void> {
2591
+ const config = await loadKnownMarketplacesConfig()
2592
+ const entry = config[name]
2593
+
2594
+ if (!entry) {
2595
+ throw new Error(
2596
+ `Marketplace '${name}' not found. Available marketplaces: ${Object.keys(config).join(', ')}`,
2597
+ )
2598
+ }
2599
+
2600
+ // Seed-managed marketplaces always have autoUpdate: false (read-only, git-pull
2601
+ // would fail). Toggle appears to work but registerSeedMarketplaces overwrites
2602
+ // it on next startup. Error with guidance instead of silent revert.
2603
+ const seedDir = seedDirFor(entry.installLocation)
2604
+ if (seedDir) {
2605
+ throw new Error(
2606
+ `Marketplace '${name}' is seed-managed (${seedDir}) and ` +
2607
+ `auto-update is always disabled for seed content. ` +
2608
+ `To update: ask your admin to update the seed.`,
2609
+ )
2610
+ }
2611
+
2612
+ // Only update if the value is actually changing
2613
+ if (entry.autoUpdate === autoUpdate) {
2614
+ return
2615
+ }
2616
+
2617
+ config[name] = {
2618
+ ...entry,
2619
+ autoUpdate,
2620
+ }
2621
+ await saveKnownMarketplacesConfig(config)
2622
+
2623
+ // Also update intent in settings if declared there — write to the SAME
2624
+ // source that declared it to avoid creating duplicates at wrong scope
2625
+ const declaringSource = getMarketplaceDeclaringSource(name)
2626
+ if (declaringSource) {
2627
+ const declared =
2628
+ getSettingsForSource(declaringSource)?.extraKnownMarketplaces?.[name]
2629
+ if (declared) {
2630
+ saveMarketplaceToSettings(
2631
+ name,
2632
+ { source: declared.source, autoUpdate },
2633
+ declaringSource,
2634
+ )
2635
+ }
2636
+ }
2637
+
2638
+ logForDebugging(`Set autoUpdate=${autoUpdate} for marketplace: ${name}`)
2639
+ }
2640
+
2641
+ export const _test = {
2642
+ redactUrlCredentials,
2643
+ }