@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,3302 @@
1
+ /**
2
+ * Plugin Loader Module
3
+ *
4
+ * This module is responsible for discovering, loading, and validating Claude Code plugins
5
+ * from various sources including marketplaces and git repositories.
6
+ *
7
+ * NPM packages are also supported but must be referenced through marketplaces - the marketplace
8
+ * entry contains the NPM package information.
9
+ *
10
+ * Plugin Discovery Sources (in order of precedence):
11
+ * 1. Marketplace-based plugins (plugin@marketplace format in settings)
12
+ * 2. Session-only plugins (from --plugin-dir CLI flag or SDK plugins option)
13
+ *
14
+ * Plugin Directory Structure:
15
+ * ```
16
+ * my-plugin/
17
+ * ├── plugin.json # Optional manifest with metadata
18
+ * ├── commands/ # Custom slash commands
19
+ * │ ├── build.md
20
+ * │ └── deploy.md
21
+ * ├── agents/ # Custom AI agents
22
+ * │ └── test-runner.md
23
+ * └── hooks/ # Hook configurations
24
+ * └── hooks.json # Hook definitions
25
+ * ```
26
+ *
27
+ * The loader handles:
28
+ * - Plugin manifest validation
29
+ * - Hooks configuration loading and variable resolution
30
+ * - Duplicate name detection
31
+ * - Enable/disable state management
32
+ * - Error collection and reporting
33
+ */
34
+
35
+ import {
36
+ copyFile,
37
+ readdir,
38
+ readFile,
39
+ readlink,
40
+ realpath,
41
+ rename,
42
+ rm,
43
+ rmdir,
44
+ stat,
45
+ symlink,
46
+ } from 'fs/promises'
47
+ import memoize from 'lodash-es/memoize.js'
48
+ import { basename, dirname, join, relative, resolve, sep } from 'path'
49
+ import { getInlinePlugins } from '../../bootstrap/state.js'
50
+ import {
51
+ BUILTIN_MARKETPLACE_NAME,
52
+ getBuiltinPlugins,
53
+ } from '../../plugins/builtinPlugins.js'
54
+ import type {
55
+ LoadedPlugin,
56
+ PluginComponent,
57
+ PluginError,
58
+ PluginLoadResult,
59
+ PluginManifest,
60
+ } from '../../types/plugin.js'
61
+ import { logForDebugging } from '../debug.js'
62
+ import { isEnvTruthy } from '../envUtils.js'
63
+ import {
64
+ errorMessage,
65
+ getErrnoPath,
66
+ isENOENT,
67
+ isFsInaccessible,
68
+ toError,
69
+ } from '../errors.js'
70
+ import { execFileNoThrow, execFileNoThrowWithCwd } from '../execFileNoThrow.js'
71
+ import { pathExists } from '../file.js'
72
+ import { getFsImplementation } from '../fsOperations.js'
73
+ import { gitExe } from '../git.js'
74
+ import { lazySchema } from '../lazySchema.js'
75
+ import { logError } from '../log.js'
76
+ import { getSettings_DEPRECATED } from '../settings/settings.js'
77
+ import {
78
+ clearPluginSettingsBase,
79
+ getPluginSettingsBase,
80
+ resetSettingsCache,
81
+ setPluginSettingsBase,
82
+ } from '../settings/settingsCache.js'
83
+ import type { HooksSettings } from '../settings/types.js'
84
+ import { SettingsSchema } from '../settings/types.js'
85
+ import { jsonParse, jsonStringify } from '../slowOperations.js'
86
+ import { getAddDirEnabledPlugins } from './addDirPluginSettings.js'
87
+ import { verifyAndDemote } from './dependencyResolver.js'
88
+ import { classifyFetchError, logPluginFetch } from './fetchTelemetry.js'
89
+ import { checkGitAvailable } from './gitAvailability.js'
90
+ import { getInMemoryInstalledPlugins } from './installedPluginsManager.js'
91
+ import { getManagedPluginNames } from './managedPlugins.js'
92
+ import {
93
+ formatSourceForDisplay,
94
+ getBlockedMarketplaces,
95
+ getStrictKnownMarketplaces,
96
+ isSourceAllowedByPolicy,
97
+ isSourceInBlocklist,
98
+ } from './marketplaceHelpers.js'
99
+ import {
100
+ getMarketplaceCacheOnly,
101
+ getPluginByIdCacheOnly,
102
+ loadKnownMarketplacesConfigSafe,
103
+ } from './marketplaceManager.js'
104
+ import { getPluginSeedDirs, getPluginsDirectory } from './pluginDirectories.js'
105
+ import { parsePluginIdentifier } from './pluginIdentifier.js'
106
+ import { validatePathWithinBase } from './pluginInstallationHelpers.js'
107
+ import { calculatePluginVersion } from './pluginVersioning.js'
108
+ import {
109
+ type CommandMetadata,
110
+ PluginHooksSchema,
111
+ PluginIdSchema,
112
+ PluginManifestSchema,
113
+ type PluginMarketplaceEntry,
114
+ type PluginSource,
115
+ } from './schemas.js'
116
+ import {
117
+ convertDirectoryToZipInPlace,
118
+ extractZipToDirectory,
119
+ getSessionPluginCachePath,
120
+ isPluginZipCacheEnabled,
121
+ } from './zipCache.js'
122
+
123
+ /**
124
+ * Get the path where plugin cache is stored
125
+ */
126
+ export function getPluginCachePath(): string {
127
+ return join(getPluginsDirectory(), 'cache')
128
+ }
129
+
130
+ /**
131
+ * Compute the versioned cache path under a specific base plugins directory.
132
+ * Used to probe both primary and seed caches.
133
+ *
134
+ * @param baseDir - Base plugins directory (e.g. getPluginsDirectory() or seed dir)
135
+ * @param pluginId - Plugin identifier in format "name@marketplace"
136
+ * @param version - Version string (semver, git SHA, etc.)
137
+ * @returns Absolute path to versioned plugin directory under baseDir
138
+ */
139
+ export function getVersionedCachePathIn(
140
+ baseDir: string,
141
+ pluginId: string,
142
+ version: string,
143
+ ): string {
144
+ const { name: pluginName, marketplace } = parsePluginIdentifier(pluginId)
145
+ const sanitizedMarketplace = (marketplace || 'unknown').replace(
146
+ /[^a-zA-Z0-9\-_]/g,
147
+ '-',
148
+ )
149
+ const sanitizedPlugin = (pluginName || pluginId).replace(
150
+ /[^a-zA-Z0-9\-_]/g,
151
+ '-',
152
+ )
153
+ // Sanitize version to prevent path traversal attacks
154
+ const sanitizedVersion = version.replace(/[^a-zA-Z0-9\-_.]/g, '-')
155
+ return join(
156
+ baseDir,
157
+ 'cache',
158
+ sanitizedMarketplace,
159
+ sanitizedPlugin,
160
+ sanitizedVersion,
161
+ )
162
+ }
163
+
164
+ /**
165
+ * Get versioned cache path for a plugin under the primary plugins directory.
166
+ * Format: ~/.claude/plugins/cache/{marketplace}/{plugin}/{version}/
167
+ *
168
+ * @param pluginId - Plugin identifier in format "name@marketplace"
169
+ * @param version - Version string (semver, git SHA, etc.)
170
+ * @returns Absolute path to versioned plugin directory
171
+ */
172
+ export function getVersionedCachePath(
173
+ pluginId: string,
174
+ version: string,
175
+ ): string {
176
+ return getVersionedCachePathIn(getPluginsDirectory(), pluginId, version)
177
+ }
178
+
179
+ /**
180
+ * Get versioned ZIP cache path for a plugin.
181
+ * This is the zip cache variant of getVersionedCachePath.
182
+ */
183
+ export function getVersionedZipCachePath(
184
+ pluginId: string,
185
+ version: string,
186
+ ): string {
187
+ return `${getVersionedCachePath(pluginId, version)}.zip`
188
+ }
189
+
190
+ /**
191
+ * Probe seed directories for a populated cache at this plugin version.
192
+ * Seeds are checked in precedence order; first hit wins. Returns null if no
193
+ * seed is configured or none contains a populated directory at this version.
194
+ */
195
+ async function probeSeedCache(
196
+ pluginId: string,
197
+ version: string,
198
+ ): Promise<string | null> {
199
+ for (const seedDir of getPluginSeedDirs()) {
200
+ const seedPath = getVersionedCachePathIn(seedDir, pluginId, version)
201
+ try {
202
+ const entries = await readdir(seedPath)
203
+ if (entries.length > 0) return seedPath
204
+ } catch {
205
+ // Try next seed
206
+ }
207
+ }
208
+ return null
209
+ }
210
+
211
+ /**
212
+ * When the computed version is 'unknown', probe seed/cache/<m>/<p>/ for an
213
+ * actual version dir. Handles the first-boot chicken-and-egg where the
214
+ * version can only be known after cloning, but seed already has the clone.
215
+ *
216
+ * Per seed, only matches when exactly one version exists (typical BYOC case).
217
+ * Multiple versions within a single seed → ambiguous → try next seed.
218
+ * Seeds are checked in precedence order; first match wins.
219
+ */
220
+ export async function probeSeedCacheAnyVersion(
221
+ pluginId: string,
222
+ ): Promise<string | null> {
223
+ for (const seedDir of getPluginSeedDirs()) {
224
+ // The parent of the version dir — computed the same way as
225
+ // getVersionedCachePathIn, just without the version component.
226
+ const pluginDir = dirname(getVersionedCachePathIn(seedDir, pluginId, '_'))
227
+ try {
228
+ const versions = await readdir(pluginDir)
229
+ if (versions.length !== 1) continue
230
+ const versionDir = join(pluginDir, versions[0]!)
231
+ const entries = await readdir(versionDir)
232
+ if (entries.length > 0) return versionDir
233
+ } catch {
234
+ // Try next seed
235
+ }
236
+ }
237
+ return null
238
+ }
239
+
240
+ /**
241
+ * Get legacy (non-versioned) cache path for a plugin.
242
+ * Format: ~/.claude/plugins/cache/{plugin-name}/
243
+ *
244
+ * Used for backward compatibility with existing installations.
245
+ *
246
+ * @param pluginName - Plugin name (without marketplace suffix)
247
+ * @returns Absolute path to legacy plugin directory
248
+ */
249
+ export function getLegacyCachePath(pluginName: string): string {
250
+ const cachePath = getPluginCachePath()
251
+ return join(cachePath, pluginName.replace(/[^a-zA-Z0-9\-_]/g, '-'))
252
+ }
253
+
254
+ /**
255
+ * Resolve plugin path with fallback to legacy location.
256
+ *
257
+ * Always:
258
+ * 1. Try versioned path first if version is provided
259
+ * 2. Fall back to legacy path for existing installations
260
+ * 3. Return versioned path for new installations
261
+ *
262
+ * @param pluginId - Plugin identifier in format "name@marketplace"
263
+ * @param version - Optional version string
264
+ * @returns Absolute path to plugin directory
265
+ */
266
+ export async function resolvePluginPath(
267
+ pluginId: string,
268
+ version?: string,
269
+ ): Promise<string> {
270
+ // Try versioned path first
271
+ if (version) {
272
+ const versionedPath = getVersionedCachePath(pluginId, version)
273
+ if (await pathExists(versionedPath)) {
274
+ return versionedPath
275
+ }
276
+ }
277
+
278
+ // Fall back to legacy path for existing installations
279
+ const pluginName = parsePluginIdentifier(pluginId).name || pluginId
280
+ const legacyPath = getLegacyCachePath(pluginName)
281
+ if (await pathExists(legacyPath)) {
282
+ return legacyPath
283
+ }
284
+
285
+ // Return versioned path for new installations
286
+ return version ? getVersionedCachePath(pluginId, version) : legacyPath
287
+ }
288
+
289
+ /**
290
+ * Recursively copy a directory.
291
+ * Exported for testing purposes.
292
+ */
293
+ export async function copyDir(src: string, dest: string): Promise<void> {
294
+ await getFsImplementation().mkdir(dest)
295
+
296
+ const entries = await readdir(src, { withFileTypes: true })
297
+
298
+ for (const entry of entries) {
299
+ const srcPath = join(src, entry.name)
300
+ const destPath = join(dest, entry.name)
301
+
302
+ if (entry.isDirectory()) {
303
+ await copyDir(srcPath, destPath)
304
+ } else if (entry.isFile()) {
305
+ await copyFile(srcPath, destPath)
306
+ } else if (entry.isSymbolicLink()) {
307
+ const linkTarget = await readlink(srcPath)
308
+
309
+ // Resolve the symlink to get the actual target path
310
+ // This prevents circular symlinks when src and dest overlap (e.g., via symlink chains)
311
+ let resolvedTarget: string
312
+ try {
313
+ resolvedTarget = await realpath(srcPath)
314
+ } catch {
315
+ // Broken symlink - copy the raw link target as-is
316
+ await symlink(linkTarget, destPath)
317
+ continue
318
+ }
319
+
320
+ // Resolve the source directory to handle symlinked source dirs
321
+ let resolvedSrc: string
322
+ try {
323
+ resolvedSrc = await realpath(src)
324
+ } catch {
325
+ resolvedSrc = src
326
+ }
327
+
328
+ // Check if target is within the source tree (using proper path prefix matching)
329
+ const srcPrefix = resolvedSrc.endsWith(sep)
330
+ ? resolvedSrc
331
+ : resolvedSrc + sep
332
+ if (
333
+ resolvedTarget.startsWith(srcPrefix) ||
334
+ resolvedTarget === resolvedSrc
335
+ ) {
336
+ // Target is within source tree - create relative symlink that preserves
337
+ // the same structure in the destination
338
+ const targetRelativeToSrc = relative(resolvedSrc, resolvedTarget)
339
+ const destTargetPath = join(dest, targetRelativeToSrc)
340
+ const relativeLinkPath = relative(dirname(destPath), destTargetPath)
341
+ await symlink(relativeLinkPath, destPath)
342
+ } else {
343
+ // Target is outside source tree - use absolute resolved path
344
+ await symlink(resolvedTarget, destPath)
345
+ }
346
+ }
347
+ }
348
+ }
349
+
350
+ /**
351
+ * Copy plugin files to versioned cache directory.
352
+ *
353
+ * For local plugins: Uses entry.source from marketplace.json as the single source of truth.
354
+ * For remote plugins: Falls back to copying sourcePath (the downloaded content).
355
+ *
356
+ * @param sourcePath - Path to the plugin source (used as fallback for remote plugins)
357
+ * @param pluginId - Plugin identifier in format "name@marketplace"
358
+ * @param version - Version string for versioned path
359
+ * @param entry - Optional marketplace entry containing the source field
360
+ * @param marketplaceDir - Marketplace directory for resolving entry.source (undefined for remote plugins)
361
+ * @returns Path to the cached plugin directory
362
+ * @throws Error if the source directory is not found
363
+ * @throws Error if the destination directory is empty after copy
364
+ */
365
+ export async function copyPluginToVersionedCache(
366
+ sourcePath: string,
367
+ pluginId: string,
368
+ version: string,
369
+ entry?: PluginMarketplaceEntry,
370
+ marketplaceDir?: string,
371
+ ): Promise<string> {
372
+ // When zip cache is enabled, the canonical format is a ZIP file
373
+ const zipCacheMode = isPluginZipCacheEnabled()
374
+ const cachePath = getVersionedCachePath(pluginId, version)
375
+ const zipPath = getVersionedZipCachePath(pluginId, version)
376
+
377
+ // If cache already exists (directory or ZIP), return it
378
+ if (zipCacheMode) {
379
+ if (await pathExists(zipPath)) {
380
+ logForDebugging(
381
+ `Plugin ${pluginId} version ${version} already cached at ${zipPath}`,
382
+ )
383
+ return zipPath
384
+ }
385
+ } else if (await pathExists(cachePath)) {
386
+ const entries = await readdir(cachePath)
387
+ if (entries.length > 0) {
388
+ logForDebugging(
389
+ `Plugin ${pluginId} version ${version} already cached at ${cachePath}`,
390
+ )
391
+ return cachePath
392
+ }
393
+ // Directory exists but is empty, remove it so we can recreate with content
394
+ logForDebugging(
395
+ `Removing empty cache directory for ${pluginId} at ${cachePath}`,
396
+ )
397
+ await rmdir(cachePath)
398
+ }
399
+
400
+ // Seed cache hit — return seed path in place (read-only, no copy).
401
+ // Callers handle both directory and .zip paths; this returns a directory.
402
+ const seedPath = await probeSeedCache(pluginId, version)
403
+ if (seedPath) {
404
+ logForDebugging(
405
+ `Using seed cache for ${pluginId}@${version} at ${seedPath}`,
406
+ )
407
+ return seedPath
408
+ }
409
+
410
+ // Create parent directories
411
+ await getFsImplementation().mkdir(dirname(cachePath))
412
+
413
+ // For local plugins: copy entry.source directory (the single source of truth)
414
+ // For remote plugins: marketplaceDir is undefined, fall back to copying sourcePath
415
+ if (entry && typeof entry.source === 'string' && marketplaceDir) {
416
+ const sourceDir = validatePathWithinBase(marketplaceDir, entry.source)
417
+
418
+ logForDebugging(
419
+ `Copying source directory ${entry.source} for plugin ${pluginId}`,
420
+ )
421
+ try {
422
+ await copyDir(sourceDir, cachePath)
423
+ } catch (e: unknown) {
424
+ // Only remap ENOENT from the top-level sourceDir itself — nested ENOENTs
425
+ // from recursive copyDir (broken symlinks, raced deletes) should preserve
426
+ // their original path in the error.
427
+ if (isENOENT(e) && getErrnoPath(e) === sourceDir) {
428
+ throw new Error(
429
+ `Plugin source directory not found: ${sourceDir} (from entry.source: ${entry.source})`,
430
+ )
431
+ }
432
+ throw e
433
+ }
434
+ } else {
435
+ // Fallback for remote plugins (already downloaded) or plugins without entry.source
436
+ logForDebugging(
437
+ `Copying plugin ${pluginId} to versioned cache (fallback to full copy)`,
438
+ )
439
+ await copyDir(sourcePath, cachePath)
440
+ }
441
+
442
+ // Remove .git directory from cache if present
443
+ const gitPath = join(cachePath, '.git')
444
+ await rm(gitPath, { recursive: true, force: true })
445
+
446
+ // Validate that cache has content - if empty, throw so fallback can be used
447
+ const cacheEntries = await readdir(cachePath)
448
+ if (cacheEntries.length === 0) {
449
+ throw new Error(
450
+ `Failed to copy plugin ${pluginId} to versioned cache: destination is empty after copy`,
451
+ )
452
+ }
453
+
454
+ // Zip cache mode: convert directory to ZIP and remove the directory
455
+ if (zipCacheMode) {
456
+ await convertDirectoryToZipInPlace(cachePath, zipPath)
457
+ logForDebugging(
458
+ `Successfully cached plugin ${pluginId} as ZIP at ${zipPath}`,
459
+ )
460
+ return zipPath
461
+ }
462
+
463
+ logForDebugging(`Successfully cached plugin ${pluginId} at ${cachePath}`)
464
+ return cachePath
465
+ }
466
+
467
+ /**
468
+ * Validate a git URL using Node.js URL parsing
469
+ */
470
+ function validateGitUrl(url: string): string {
471
+ try {
472
+ const parsed = new URL(url)
473
+ if (!['https:', 'http:', 'file:'].includes(parsed.protocol)) {
474
+ if (!/^git@[a-zA-Z0-9.-]+:/.test(url)) {
475
+ throw new Error(
476
+ `Invalid git URL protocol: ${parsed.protocol}. Only HTTPS, HTTP, file:// and SSH (git@) URLs are supported.`,
477
+ )
478
+ }
479
+ }
480
+ return url
481
+ } catch {
482
+ if (/^git@[a-zA-Z0-9.-]+:/.test(url)) {
483
+ return url
484
+ }
485
+ throw new Error(`Invalid git URL: ${url}`)
486
+ }
487
+ }
488
+
489
+ /**
490
+ * Install a plugin from npm using a global cache (exported for testing)
491
+ */
492
+ export async function installFromNpm(
493
+ packageName: string,
494
+ targetPath: string,
495
+ options: { registry?: string; version?: string } = {},
496
+ ): Promise<void> {
497
+ const npmCachePath = join(getPluginsDirectory(), 'npm-cache')
498
+
499
+ await getFsImplementation().mkdir(npmCachePath)
500
+
501
+ const packageSpec = options.version
502
+ ? `${packageName}@${options.version}`
503
+ : packageName
504
+ const packagePath = join(npmCachePath, 'node_modules', packageName)
505
+ const needsInstall = !(await pathExists(packagePath))
506
+
507
+ if (needsInstall) {
508
+ logForDebugging(`Installing npm package ${packageSpec} to cache`)
509
+ const args = ['install', packageSpec, '--prefix', npmCachePath]
510
+ if (options.registry) {
511
+ args.push('--registry', options.registry)
512
+ }
513
+ const result = await execFileNoThrow('npm', args, { useCwd: false })
514
+
515
+ if (result.code !== 0) {
516
+ throw new Error(`Failed to install npm package: ${result.stderr}`)
517
+ }
518
+ }
519
+
520
+ await copyDir(packagePath, targetPath)
521
+ logForDebugging(
522
+ `Copied npm package ${packageName} from cache to ${targetPath}`,
523
+ )
524
+ }
525
+
526
+ /**
527
+ * Clone a git repository (exported for testing)
528
+ *
529
+ * @param gitUrl - The git URL to clone
530
+ * @param targetPath - Where to clone the repository
531
+ * @param ref - Optional branch or tag to checkout
532
+ * @param sha - Optional specific commit SHA to checkout
533
+ */
534
+ export async function gitClone(
535
+ gitUrl: string,
536
+ targetPath: string,
537
+ ref?: string,
538
+ sha?: string,
539
+ ): Promise<void> {
540
+ // Use --recurse-submodules to initialize submodules
541
+ // Always start with shallow clone for efficiency
542
+ const args = [
543
+ 'clone',
544
+ '--depth',
545
+ '1',
546
+ '--recurse-submodules',
547
+ '--shallow-submodules',
548
+ ]
549
+
550
+ // Add --branch flag for specific ref (works for both branches and tags)
551
+ if (ref) {
552
+ args.push('--branch', ref)
553
+ }
554
+
555
+ // If sha is specified, use --no-checkout since we'll checkout the SHA separately
556
+ if (sha) {
557
+ args.push('--no-checkout')
558
+ }
559
+
560
+ args.push(gitUrl, targetPath)
561
+
562
+ const cloneStarted = performance.now()
563
+ const cloneResult = await execFileNoThrow(gitExe(), args)
564
+
565
+ if (cloneResult.code !== 0) {
566
+ logPluginFetch(
567
+ 'plugin_clone',
568
+ gitUrl,
569
+ 'failure',
570
+ performance.now() - cloneStarted,
571
+ classifyFetchError(cloneResult.stderr),
572
+ )
573
+ throw new Error(`Failed to clone repository: ${cloneResult.stderr}`)
574
+ }
575
+
576
+ // If sha is specified, fetch and checkout that specific commit
577
+ if (sha) {
578
+ // Try shallow fetch of the specific SHA first (most efficient)
579
+ const shallowFetchResult = await execFileNoThrowWithCwd(
580
+ gitExe(),
581
+ ['fetch', '--depth', '1', 'origin', sha],
582
+ { cwd: targetPath },
583
+ )
584
+
585
+ if (shallowFetchResult.code !== 0) {
586
+ // Some servers don't support fetching arbitrary SHAs
587
+ // Fall back to unshallow fetch to get full history
588
+ logForDebugging(
589
+ `Shallow fetch of SHA ${sha} failed, falling back to unshallow fetch`,
590
+ )
591
+ const unshallowResult = await execFileNoThrowWithCwd(
592
+ gitExe(),
593
+ ['fetch', '--unshallow'],
594
+ { cwd: targetPath },
595
+ )
596
+
597
+ if (unshallowResult.code !== 0) {
598
+ logPluginFetch(
599
+ 'plugin_clone',
600
+ gitUrl,
601
+ 'failure',
602
+ performance.now() - cloneStarted,
603
+ classifyFetchError(unshallowResult.stderr),
604
+ )
605
+ throw new Error(
606
+ `Failed to fetch commit ${sha}: ${unshallowResult.stderr}`,
607
+ )
608
+ }
609
+ }
610
+
611
+ // Checkout the specific commit
612
+ const checkoutResult = await execFileNoThrowWithCwd(
613
+ gitExe(),
614
+ ['checkout', sha],
615
+ { cwd: targetPath },
616
+ )
617
+
618
+ if (checkoutResult.code !== 0) {
619
+ logPluginFetch(
620
+ 'plugin_clone',
621
+ gitUrl,
622
+ 'failure',
623
+ performance.now() - cloneStarted,
624
+ classifyFetchError(checkoutResult.stderr),
625
+ )
626
+ throw new Error(
627
+ `Failed to checkout commit ${sha}: ${checkoutResult.stderr}`,
628
+ )
629
+ }
630
+ }
631
+
632
+ // Fire success only after ALL network ops (clone + optional SHA fetch)
633
+ // complete — same telemetry-scope discipline as mcpb and marketplace_url.
634
+ logPluginFetch(
635
+ 'plugin_clone',
636
+ gitUrl,
637
+ 'success',
638
+ performance.now() - cloneStarted,
639
+ )
640
+ }
641
+
642
+ /**
643
+ * Install a plugin from a git URL
644
+ */
645
+ async function installFromGit(
646
+ gitUrl: string,
647
+ targetPath: string,
648
+ ref?: string,
649
+ sha?: string,
650
+ ): Promise<void> {
651
+ const safeUrl = validateGitUrl(gitUrl)
652
+ await gitClone(safeUrl, targetPath, ref, sha)
653
+ const refMessage = ref ? ` (ref: ${ref})` : ''
654
+ logForDebugging(
655
+ `Cloned repository from ${safeUrl}${refMessage} to ${targetPath}`,
656
+ )
657
+ }
658
+
659
+ /**
660
+ * Install a plugin from GitHub
661
+ */
662
+ async function installFromGitHub(
663
+ repo: string,
664
+ targetPath: string,
665
+ ref?: string,
666
+ sha?: string,
667
+ ): Promise<void> {
668
+ if (!/^[a-zA-Z0-9-_.]+\/[a-zA-Z0-9-_.]+$/.test(repo)) {
669
+ throw new Error(
670
+ `Invalid GitHub repository format: ${repo}. Expected format: owner/repo`,
671
+ )
672
+ }
673
+ // Use HTTPS for CCR (no SSH keys), SSH for normal CLI
674
+ const gitUrl = isEnvTruthy(process.env.CLAUDE_CODE_REMOTE)
675
+ ? `https://github.com/${repo}.git`
676
+ : `git@github.com:${repo}.git`
677
+ return installFromGit(gitUrl, targetPath, ref, sha)
678
+ }
679
+
680
+ /**
681
+ * Resolve a git-subdir `url` field to a clonable git URL.
682
+ * Accepts GitHub owner/repo shorthand (converted to ssh or https depending on
683
+ * CLAUDE_CODE_REMOTE) or any URL that passes validateGitUrl (https, http,
684
+ * file, git@ ssh).
685
+ */
686
+ function resolveGitSubdirUrl(url: string): string {
687
+ if (/^[a-zA-Z0-9-_.]+\/[a-zA-Z0-9-_.]+$/.test(url)) {
688
+ return isEnvTruthy(process.env.CLAUDE_CODE_REMOTE)
689
+ ? `https://github.com/${url}.git`
690
+ : `git@github.com:${url}.git`
691
+ }
692
+ return validateGitUrl(url)
693
+ }
694
+
695
+ /**
696
+ * Install a plugin from a subdirectory of a git repository (exported for
697
+ * testing).
698
+ *
699
+ * Uses partial clone (--filter=tree:0) + sparse-checkout so only the tree
700
+ * objects along the path and the blobs under it are downloaded. For large
701
+ * monorepos this is dramatically cheaper than a full clone — the tree objects
702
+ * for a million-file repo can be hundreds of MB, all avoided here.
703
+ *
704
+ * Sequence:
705
+ * 1. clone --depth 1 --filter=tree:0 --no-checkout [--branch ref]
706
+ * 2. sparse-checkout set --cone -- <path>
707
+ * 3. If sha: fetch --depth 1 origin <sha> (fallback: --unshallow), then
708
+ * checkout <sha>. The partial-clone filter is stored in remote config so
709
+ * subsequent fetches respect it; --unshallow gets all commits but trees
710
+ * and blobs remain lazy.
711
+ * If no sha: checkout HEAD (points to ref if --branch was used).
712
+ * 4. Move <cloneDir>/<path> to targetPath and discard the clone.
713
+ *
714
+ * The clone is ephemeral — it goes into a sibling temp directory and is
715
+ * removed after the subdir is extracted. targetPath ends up containing only
716
+ * the plugin files with no .git directory.
717
+ */
718
+ export async function installFromGitSubdir(
719
+ url: string,
720
+ targetPath: string,
721
+ subdirPath: string,
722
+ ref?: string,
723
+ sha?: string,
724
+ ): Promise<string | undefined> {
725
+ if (!(await checkGitAvailable())) {
726
+ throw new Error(
727
+ 'git-subdir plugin source requires git to be installed and on PATH. ' +
728
+ 'Install git (version 2.25 or later for sparse-checkout cone mode) and try again.',
729
+ )
730
+ }
731
+
732
+ const gitUrl = resolveGitSubdirUrl(url)
733
+ // Clone into a sibling temp dir (same filesystem → rename works, no EXDEV).
734
+ const cloneDir = `${targetPath}.clone`
735
+
736
+ const cloneArgs = [
737
+ 'clone',
738
+ '--depth',
739
+ '1',
740
+ '--filter=tree:0',
741
+ '--no-checkout',
742
+ ]
743
+ if (ref) {
744
+ cloneArgs.push('--branch', ref)
745
+ }
746
+ cloneArgs.push(gitUrl, cloneDir)
747
+
748
+ const cloneResult = await execFileNoThrow(gitExe(), cloneArgs)
749
+ if (cloneResult.code !== 0) {
750
+ throw new Error(
751
+ `Failed to clone repository for git-subdir source: ${cloneResult.stderr}`,
752
+ )
753
+ }
754
+
755
+ try {
756
+ const sparseResult = await execFileNoThrowWithCwd(
757
+ gitExe(),
758
+ ['sparse-checkout', 'set', '--cone', '--', subdirPath],
759
+ { cwd: cloneDir },
760
+ )
761
+ if (sparseResult.code !== 0) {
762
+ throw new Error(
763
+ `git sparse-checkout set failed (git >= 2.25 required for cone mode): ${sparseResult.stderr}`,
764
+ )
765
+ }
766
+
767
+ // Capture the resolved commit SHA before discarding the clone. The
768
+ // extracted subdir has no .git, so the caller can't rev-parse it later.
769
+ // If the source specified a full 40-char sha we already know it; otherwise
770
+ // read HEAD (which points to ref's tip after --branch, or the remote
771
+ // default branch if no ref was given).
772
+ let resolvedSha: string | undefined
773
+
774
+ if (sha) {
775
+ const fetchSha = await execFileNoThrowWithCwd(
776
+ gitExe(),
777
+ ['fetch', '--depth', '1', 'origin', sha],
778
+ { cwd: cloneDir },
779
+ )
780
+ if (fetchSha.code !== 0) {
781
+ logForDebugging(
782
+ `Shallow fetch of SHA ${sha} failed for git-subdir, falling back to unshallow fetch`,
783
+ )
784
+ const unshallow = await execFileNoThrowWithCwd(
785
+ gitExe(),
786
+ ['fetch', '--unshallow'],
787
+ { cwd: cloneDir },
788
+ )
789
+ if (unshallow.code !== 0) {
790
+ throw new Error(`Failed to fetch commit ${sha}: ${unshallow.stderr}`)
791
+ }
792
+ }
793
+ const checkout = await execFileNoThrowWithCwd(
794
+ gitExe(),
795
+ ['checkout', sha],
796
+ { cwd: cloneDir },
797
+ )
798
+ if (checkout.code !== 0) {
799
+ throw new Error(`Failed to checkout commit ${sha}: ${checkout.stderr}`)
800
+ }
801
+ resolvedSha = sha
802
+ } else {
803
+ // checkout HEAD materializes the working tree (this is where blobs are
804
+ // lazy-fetched — the slow, network-bound step). It doesn't move HEAD;
805
+ // --branch at clone time already positioned it. rev-parse HEAD is a
806
+ // purely read-only ref lookup (no index lock), so it runs safely in
807
+ // parallel with checkout and we avoid waiting on the network for it.
808
+ const [checkout, revParse] = await Promise.all([
809
+ execFileNoThrowWithCwd(gitExe(), ['checkout', 'HEAD'], {
810
+ cwd: cloneDir,
811
+ }),
812
+ execFileNoThrowWithCwd(gitExe(), ['rev-parse', 'HEAD'], {
813
+ cwd: cloneDir,
814
+ }),
815
+ ])
816
+ if (checkout.code !== 0) {
817
+ throw new Error(
818
+ `git checkout after sparse-checkout failed: ${checkout.stderr}`,
819
+ )
820
+ }
821
+ if (revParse.code === 0) {
822
+ resolvedSha = revParse.stdout.trim()
823
+ }
824
+ }
825
+
826
+ // Path traversal guard: resolve+verify the subdir stays inside cloneDir
827
+ // before moving it out. rename ENOENT is wrapped with a friendlier
828
+ // message that references the source path, not internal temp dirs.
829
+ const resolvedSubdir = validatePathWithinBase(cloneDir, subdirPath)
830
+ try {
831
+ await rename(resolvedSubdir, targetPath)
832
+ } catch (e: unknown) {
833
+ if (isENOENT(e)) {
834
+ throw new Error(
835
+ `Subdirectory '${subdirPath}' not found in repository ${gitUrl}${ref ? ` (ref: ${ref})` : ''}. ` +
836
+ 'Check that the path is correct and exists at the specified ref/sha.',
837
+ )
838
+ }
839
+ throw e
840
+ }
841
+
842
+ const refMsg = ref ? ` ref=${ref}` : ''
843
+ const shaMsg = resolvedSha ? ` sha=${resolvedSha}` : ''
844
+ logForDebugging(
845
+ `Extracted subdir ${subdirPath} from ${gitUrl}${refMsg}${shaMsg} to ${targetPath}`,
846
+ )
847
+ return resolvedSha
848
+ } finally {
849
+ await rm(cloneDir, { recursive: true, force: true })
850
+ }
851
+ }
852
+
853
+ /**
854
+ * Install a plugin from a local path
855
+ */
856
+ async function installFromLocal(
857
+ sourcePath: string,
858
+ targetPath: string,
859
+ ): Promise<void> {
860
+ if (!(await pathExists(sourcePath))) {
861
+ throw new Error(`Source path does not exist: ${sourcePath}`)
862
+ }
863
+
864
+ await copyDir(sourcePath, targetPath)
865
+
866
+ const gitPath = join(targetPath, '.git')
867
+ await rm(gitPath, { recursive: true, force: true })
868
+ }
869
+
870
+ /**
871
+ * Generate a temporary cache name for a plugin
872
+ */
873
+ export function generateTemporaryCacheNameForPlugin(
874
+ source: PluginSource,
875
+ ): string {
876
+ const timestamp = Date.now()
877
+ const random = Math.random().toString(36).substring(2, 8)
878
+
879
+ let prefix: string
880
+
881
+ if (typeof source === 'string') {
882
+ prefix = 'local'
883
+ } else {
884
+ switch (source.source) {
885
+ case 'npm':
886
+ prefix = 'npm'
887
+ break
888
+ case 'pip':
889
+ prefix = 'pip'
890
+ break
891
+ case 'github':
892
+ prefix = 'github'
893
+ break
894
+ case 'url':
895
+ prefix = 'git'
896
+ break
897
+ case 'git-subdir':
898
+ prefix = 'subdir'
899
+ break
900
+ default:
901
+ prefix = 'unknown'
902
+ }
903
+ }
904
+
905
+ return `temp_${prefix}_${timestamp}_${random}`
906
+ }
907
+
908
+ /**
909
+ * Cache a plugin from an external source
910
+ */
911
+ export async function cachePlugin(
912
+ source: PluginSource,
913
+ options?: {
914
+ manifest?: PluginManifest
915
+ },
916
+ ): Promise<{ path: string; manifest: PluginManifest; gitCommitSha?: string }> {
917
+ const cachePath = getPluginCachePath()
918
+
919
+ await getFsImplementation().mkdir(cachePath)
920
+
921
+ const tempName = generateTemporaryCacheNameForPlugin(source)
922
+ const tempPath = join(cachePath, tempName)
923
+
924
+ let shouldCleanup = false
925
+ let gitCommitSha: string | undefined
926
+
927
+ try {
928
+ logForDebugging(
929
+ `Caching plugin from source: ${jsonStringify(source)} to temporary path ${tempPath}`,
930
+ )
931
+
932
+ shouldCleanup = true
933
+
934
+ if (typeof source === 'string') {
935
+ await installFromLocal(source, tempPath)
936
+ } else {
937
+ switch (source.source) {
938
+ case 'npm':
939
+ await installFromNpm(source.package, tempPath, {
940
+ registry: source.registry,
941
+ version: source.version,
942
+ })
943
+ break
944
+ case 'github':
945
+ await installFromGitHub(source.repo, tempPath, source.ref, source.sha)
946
+ break
947
+ case 'url':
948
+ await installFromGit(source.url, tempPath, source.ref, source.sha)
949
+ break
950
+ case 'git-subdir':
951
+ gitCommitSha = await installFromGitSubdir(
952
+ source.url,
953
+ tempPath,
954
+ source.path,
955
+ source.ref,
956
+ source.sha,
957
+ )
958
+ break
959
+ case 'pip':
960
+ throw new Error('Python package plugins are not yet supported')
961
+ default:
962
+ throw new Error(`Unsupported plugin source type`)
963
+ }
964
+ }
965
+ } catch (error) {
966
+ if (shouldCleanup && (await pathExists(tempPath))) {
967
+ logForDebugging(`Cleaning up failed installation at ${tempPath}`)
968
+ try {
969
+ await rm(tempPath, { recursive: true, force: true })
970
+ } catch (cleanupError) {
971
+ logForDebugging(`Failed to clean up installation: ${cleanupError}`, {
972
+ level: 'error',
973
+ })
974
+ }
975
+ }
976
+ throw error
977
+ }
978
+
979
+ const manifestPath = join(tempPath, '.claude-plugin', 'plugin.json')
980
+ const legacyManifestPath = join(tempPath, 'plugin.json')
981
+ let manifest: PluginManifest
982
+
983
+ if (await pathExists(manifestPath)) {
984
+ try {
985
+ const content = await readFile(manifestPath, { encoding: 'utf-8' })
986
+ const parsed = jsonParse(content)
987
+ const result = PluginManifestSchema().safeParse(parsed)
988
+
989
+ if (result.success) {
990
+ manifest = result.data
991
+ } else {
992
+ // Manifest exists but is invalid - throw error
993
+ const errors = result.error.issues
994
+ .map(err => `${err.path.join('.')}: ${err.message}`)
995
+ .join(', ')
996
+
997
+ logForDebugging(`Invalid manifest at ${manifestPath}: ${errors}`, {
998
+ level: 'error',
999
+ })
1000
+
1001
+ throw new Error(
1002
+ `Plugin has an invalid manifest file at ${manifestPath}. Validation errors: ${errors}`,
1003
+ )
1004
+ }
1005
+ } catch (error) {
1006
+ // Check if this is a validation error we just threw
1007
+ if (
1008
+ error instanceof Error &&
1009
+ error.message.includes('invalid manifest file')
1010
+ ) {
1011
+ throw error
1012
+ }
1013
+
1014
+ // JSON parse error
1015
+ const errorMsg = errorMessage(error)
1016
+ logForDebugging(
1017
+ `Failed to parse manifest at ${manifestPath}: ${errorMsg}`,
1018
+ {
1019
+ level: 'error',
1020
+ },
1021
+ )
1022
+
1023
+ throw new Error(
1024
+ `Plugin has a corrupt manifest file at ${manifestPath}. JSON parse error: ${errorMsg}`,
1025
+ )
1026
+ }
1027
+ } else if (await pathExists(legacyManifestPath)) {
1028
+ try {
1029
+ const content = await readFile(legacyManifestPath, {
1030
+ encoding: 'utf-8',
1031
+ })
1032
+ const parsed = jsonParse(content)
1033
+ const result = PluginManifestSchema().safeParse(parsed)
1034
+
1035
+ if (result.success) {
1036
+ manifest = result.data
1037
+ } else {
1038
+ // Manifest exists but is invalid - throw error
1039
+ const errors = result.error.issues
1040
+ .map(err => `${err.path.join('.')}: ${err.message}`)
1041
+ .join(', ')
1042
+
1043
+ logForDebugging(
1044
+ `Invalid legacy manifest at ${legacyManifestPath}: ${errors}`,
1045
+ { level: 'error' },
1046
+ )
1047
+
1048
+ throw new Error(
1049
+ `Plugin has an invalid manifest file at ${legacyManifestPath}. Validation errors: ${errors}`,
1050
+ )
1051
+ }
1052
+ } catch (error) {
1053
+ // Check if this is a validation error we just threw
1054
+ if (
1055
+ error instanceof Error &&
1056
+ error.message.includes('invalid manifest file')
1057
+ ) {
1058
+ throw error
1059
+ }
1060
+
1061
+ // JSON parse error
1062
+ const errorMsg = errorMessage(error)
1063
+ logForDebugging(
1064
+ `Failed to parse legacy manifest at ${legacyManifestPath}: ${errorMsg}`,
1065
+ {
1066
+ level: 'error',
1067
+ },
1068
+ )
1069
+
1070
+ throw new Error(
1071
+ `Plugin has a corrupt manifest file at ${legacyManifestPath}. JSON parse error: ${errorMsg}`,
1072
+ )
1073
+ }
1074
+ } else {
1075
+ manifest = options?.manifest || {
1076
+ name: tempName,
1077
+ description: `Plugin cached from ${typeof source === 'string' ? source : source.source}`,
1078
+ }
1079
+ }
1080
+
1081
+ const finalName = manifest.name.replace(/[^a-zA-Z0-9-_]/g, '-')
1082
+ const finalPath = join(cachePath, finalName)
1083
+
1084
+ if (await pathExists(finalPath)) {
1085
+ logForDebugging(`Removing old cached version at ${finalPath}`)
1086
+ await rm(finalPath, { recursive: true, force: true })
1087
+ }
1088
+
1089
+ await rename(tempPath, finalPath)
1090
+
1091
+ logForDebugging(`Successfully cached plugin ${manifest.name} to ${finalPath}`)
1092
+
1093
+ return {
1094
+ path: finalPath,
1095
+ manifest,
1096
+ ...(gitCommitSha && { gitCommitSha }),
1097
+ }
1098
+ }
1099
+
1100
+ /**
1101
+ * Loads and validates a plugin manifest from a JSON file.
1102
+ *
1103
+ * The manifest provides metadata about the plugin including name, version,
1104
+ * description, author, and other optional fields. If no manifest exists,
1105
+ * a minimal one is created to allow the plugin to function.
1106
+ *
1107
+ * Example plugin.json:
1108
+ * ```json
1109
+ * {
1110
+ * "name": "code-assistant",
1111
+ * "version": "1.2.0",
1112
+ * "description": "AI-powered code assistance tools",
1113
+ * "author": {
1114
+ * "name": "John Doe",
1115
+ * "email": "john@example.com"
1116
+ * },
1117
+ * "keywords": ["coding", "ai", "assistant"],
1118
+ * "homepage": "https://example.com/code-assistant",
1119
+ * "hooks": "./custom-hooks.json",
1120
+ * "commands": ["./extra-commands/*.md"]
1121
+ * }
1122
+ * ```
1123
+ */
1124
+
1125
+ /**
1126
+ * Loads and validates a plugin manifest from a JSON file.
1127
+ *
1128
+ * The manifest provides metadata about the plugin including name, version,
1129
+ * description, author, and other optional fields. If no manifest exists,
1130
+ * a minimal one is created to allow the plugin to function.
1131
+ *
1132
+ * Unknown keys in the manifest are silently stripped (PluginManifestSchema
1133
+ * uses zod's default strip behavior, not .strict()). Type mismatches and
1134
+ * other validation errors still fail.
1135
+ *
1136
+ * Behavior:
1137
+ * - Missing file: Creates default with provided name and source
1138
+ * - Invalid JSON: Throws error with parse details
1139
+ * - Schema validation failure: Throws error with validation details
1140
+ *
1141
+ * @param manifestPath - Full path to the plugin.json file
1142
+ * @param pluginName - Name to use in default manifest (e.g., "my-plugin")
1143
+ * @param source - Source description for default manifest (e.g., "git:repo" or ".claude-plugin/name")
1144
+ * @returns A valid PluginManifest object (either loaded or default)
1145
+ * @throws Error if manifest exists but is invalid (corrupt JSON or schema validation failure)
1146
+ */
1147
+ export async function loadPluginManifest(
1148
+ manifestPath: string,
1149
+ pluginName: string,
1150
+ source: string,
1151
+ ): Promise<PluginManifest> {
1152
+ // Check if manifest file exists
1153
+ // If not, create a minimal manifest to allow plugin to function
1154
+ if (!(await pathExists(manifestPath))) {
1155
+ // Return default manifest with provided name and source
1156
+ return {
1157
+ name: pluginName,
1158
+ description: `Plugin from ${source}`,
1159
+ }
1160
+ }
1161
+
1162
+ try {
1163
+ // Read and parse the manifest JSON file
1164
+ const content = await readFile(manifestPath, { encoding: 'utf-8' })
1165
+ const parsedJson = jsonParse(content)
1166
+
1167
+ // Validate against the PluginManifest schema
1168
+ const result = PluginManifestSchema().safeParse(parsedJson)
1169
+
1170
+ if (result.success) {
1171
+ // Valid manifest - return the validated data
1172
+ return result.data
1173
+ }
1174
+
1175
+ // Schema validation failed but JSON was valid
1176
+ const errors = result.error.issues
1177
+ .map(err =>
1178
+ err.path.length > 0
1179
+ ? `${err.path.join('.')}: ${err.message}`
1180
+ : err.message,
1181
+ )
1182
+ .join(', ')
1183
+
1184
+ logForDebugging(
1185
+ `Plugin ${pluginName} has an invalid manifest file at ${manifestPath}. Validation errors: ${errors}`,
1186
+ { level: 'error' },
1187
+ )
1188
+
1189
+ throw new Error(
1190
+ `Plugin ${pluginName} has an invalid manifest file at ${manifestPath}.\n\nValidation errors: ${errors}`,
1191
+ )
1192
+ } catch (error) {
1193
+ // Check if this is the error we just threw (validation error)
1194
+ if (
1195
+ error instanceof Error &&
1196
+ error.message.includes('invalid manifest file')
1197
+ ) {
1198
+ throw error
1199
+ }
1200
+
1201
+ // JSON parsing failed or file read error
1202
+ const errorMsg = errorMessage(error)
1203
+
1204
+ logForDebugging(
1205
+ `Plugin ${pluginName} has a corrupt manifest file at ${manifestPath}. Parse error: ${errorMsg}`,
1206
+ { level: 'error' },
1207
+ )
1208
+
1209
+ throw new Error(
1210
+ `Plugin ${pluginName} has a corrupt manifest file at ${manifestPath}.\n\nJSON parse error: ${errorMsg}`,
1211
+ )
1212
+ }
1213
+ }
1214
+
1215
+ /**
1216
+ * Loads and validates plugin hooks configuration from a JSON file.
1217
+ * IMPORTANT: Only call this when the hooks file is expected to exist.
1218
+ *
1219
+ * @param hooksConfigPath - Full path to the hooks.json file
1220
+ * @param pluginName - Plugin name for error messages
1221
+ * @returns Validated HooksSettings
1222
+ * @throws Error if file doesn't exist or is invalid
1223
+ */
1224
+ async function loadPluginHooks(
1225
+ hooksConfigPath: string,
1226
+ pluginName: string,
1227
+ ): Promise<HooksSettings> {
1228
+ if (!(await pathExists(hooksConfigPath))) {
1229
+ throw new Error(
1230
+ `Hooks file not found at ${hooksConfigPath} for plugin ${pluginName}. If the manifest declares hooks, the file must exist.`,
1231
+ )
1232
+ }
1233
+
1234
+ const content = await readFile(hooksConfigPath, { encoding: 'utf-8' })
1235
+ const rawHooksConfig = jsonParse(content)
1236
+
1237
+ // The hooks.json file has a wrapper structure with description and hooks
1238
+ // Use PluginHooksSchema to validate and extract the hooks property
1239
+ const validatedPluginHooks = PluginHooksSchema().parse(rawHooksConfig)
1240
+
1241
+ return validatedPluginHooks.hooks as HooksSettings
1242
+ }
1243
+
1244
+ /**
1245
+ * Validate a list of plugin component relative paths by checking existence in parallel.
1246
+ *
1247
+ * This helper parallelizes the pathExists checks (the expensive async part) while
1248
+ * preserving deterministic error/log ordering by iterating results sequentially.
1249
+ *
1250
+ * Introduced to fix a perf regression from the sync→async fs migration: sequential
1251
+ * `for { await pathExists }` loops add ~1-5ms of event-loop overhead per iteration.
1252
+ * With many plugins × several component types, this compounds to hundreds of ms.
1253
+ *
1254
+ * @param relPaths - Relative paths from the manifest/marketplace entry to validate
1255
+ * @param pluginPath - Plugin root directory to resolve relative paths against
1256
+ * @param pluginName - Plugin name for error messages
1257
+ * @param source - Source identifier for PluginError records
1258
+ * @param component - Which component these paths belong to (for error records)
1259
+ * @param componentLabel - Human-readable label for log messages (e.g. "Agent", "Skill")
1260
+ * @param contextLabel - Where the path came from, for log messages
1261
+ * (e.g. "specified in manifest but", "from marketplace entry")
1262
+ * @param errors - Error array to push path-not-found errors into (mutated)
1263
+ * @returns Array of full paths that exist on disk, in original order
1264
+ */
1265
+ async function validatePluginPaths(
1266
+ relPaths: string[],
1267
+ pluginPath: string,
1268
+ pluginName: string,
1269
+ source: string,
1270
+ component: PluginComponent,
1271
+ componentLabel: string,
1272
+ contextLabel: string,
1273
+ errors: PluginError[],
1274
+ ): Promise<string[]> {
1275
+ // Parallelize the async pathExists checks
1276
+ const checks = await Promise.all(
1277
+ relPaths.map(async relPath => {
1278
+ const fullPath = join(pluginPath, relPath)
1279
+ return { relPath, fullPath, exists: await pathExists(fullPath) }
1280
+ }),
1281
+ )
1282
+ // Process results in original order to keep error/log ordering deterministic
1283
+ const validPaths: string[] = []
1284
+ for (const { relPath, fullPath, exists } of checks) {
1285
+ if (exists) {
1286
+ validPaths.push(fullPath)
1287
+ } else {
1288
+ logForDebugging(
1289
+ `${componentLabel} path ${relPath} ${contextLabel} not found at ${fullPath} for ${pluginName}`,
1290
+ { level: 'warn' },
1291
+ )
1292
+ logError(
1293
+ new Error(
1294
+ `Plugin component file not found: ${fullPath} for ${pluginName}`,
1295
+ ),
1296
+ )
1297
+ errors.push({
1298
+ type: 'path-not-found',
1299
+ source,
1300
+ plugin: pluginName,
1301
+ path: fullPath,
1302
+ component,
1303
+ })
1304
+ }
1305
+ }
1306
+ return validPaths
1307
+ }
1308
+
1309
+ /**
1310
+ * Creates a LoadedPlugin object from a plugin directory path.
1311
+ *
1312
+ * This is the central function that assembles a complete plugin representation
1313
+ * by scanning the plugin directory structure and loading all components.
1314
+ * It handles both fully-featured plugins with manifests and minimal plugins
1315
+ * with just commands or agents directories.
1316
+ *
1317
+ * Directory structure it looks for:
1318
+ * ```
1319
+ * plugin-directory/
1320
+ * ├── plugin.json # Optional: Plugin manifest
1321
+ * ├── commands/ # Optional: Custom slash commands
1322
+ * │ ├── build.md # /build command
1323
+ * │ └── test.md # /test command
1324
+ * ├── agents/ # Optional: Custom AI agents
1325
+ * │ ├── reviewer.md # Code review agent
1326
+ * │ └── optimizer.md # Performance optimization agent
1327
+ * └── hooks/ # Optional: Hook configurations
1328
+ * └── hooks.json # Hook definitions
1329
+ * ```
1330
+ *
1331
+ * Component detection:
1332
+ * - Manifest: Loaded from plugin.json if present, otherwise creates default
1333
+ * - Commands: Sets commandsPath if commands/ directory exists
1334
+ * - Agents: Sets agentsPath if agents/ directory exists
1335
+ * - Hooks: Loads from hooks/hooks.json if present
1336
+ *
1337
+ * The function is tolerant of missing components - a plugin can have
1338
+ * any combination of the above directories/files. Missing component files
1339
+ * are reported as errors but don't prevent plugin loading.
1340
+ *
1341
+ * @param pluginPath - Absolute path to the plugin directory
1342
+ * @param source - Source identifier (e.g., "git:repo", ".claude-plugin/my-plugin")
1343
+ * @param enabled - Initial enabled state (may be overridden by settings)
1344
+ * @param fallbackName - Name to use if manifest doesn't specify one
1345
+ * @param strict - When true, adds errors for duplicate hook files (default: true)
1346
+ * @returns Object containing the LoadedPlugin and any errors encountered
1347
+ */
1348
+ export async function createPluginFromPath(
1349
+ pluginPath: string,
1350
+ source: string,
1351
+ enabled: boolean,
1352
+ fallbackName: string,
1353
+ strict = true,
1354
+ ): Promise<{ plugin: LoadedPlugin; errors: PluginError[] }> {
1355
+ const errors: PluginError[] = []
1356
+
1357
+ // Step 1: Load or create the plugin manifest
1358
+ // This provides metadata about the plugin (name, version, etc.)
1359
+ const manifestPath = join(pluginPath, '.claude-plugin', 'plugin.json')
1360
+ const manifest = await loadPluginManifest(manifestPath, fallbackName, source)
1361
+
1362
+ // Step 2: Create the base plugin object
1363
+ // Start with required fields from manifest and parameters
1364
+ const plugin: LoadedPlugin = {
1365
+ name: manifest.name, // Use name from manifest (or fallback)
1366
+ manifest, // Store full manifest for later use
1367
+ path: pluginPath, // Absolute path to plugin directory
1368
+ source, // Source identifier (e.g., "git:repo" or ".claude-plugin/name")
1369
+ repository: source, // For backward compatibility with Plugin Repository
1370
+ enabled, // Current enabled state
1371
+ }
1372
+
1373
+ // Step 3: Auto-detect optional directories in parallel
1374
+ const [
1375
+ commandsDirExists,
1376
+ agentsDirExists,
1377
+ skillsDirExists,
1378
+ outputStylesDirExists,
1379
+ ] = await Promise.all([
1380
+ !manifest.commands ? pathExists(join(pluginPath, 'commands')) : false,
1381
+ !manifest.agents ? pathExists(join(pluginPath, 'agents')) : false,
1382
+ !manifest.skills ? pathExists(join(pluginPath, 'skills')) : false,
1383
+ !manifest.outputStyles
1384
+ ? pathExists(join(pluginPath, 'output-styles'))
1385
+ : false,
1386
+ ])
1387
+
1388
+ const commandsPath = join(pluginPath, 'commands')
1389
+ if (commandsDirExists) {
1390
+ plugin.commandsPath = commandsPath
1391
+ }
1392
+
1393
+ // Step 3a: Process additional command paths from manifest
1394
+ if (manifest.commands) {
1395
+ // Check if it's an object mapping (record of command name → metadata)
1396
+ const firstValue = Object.values(manifest.commands)[0]
1397
+ if (
1398
+ typeof manifest.commands === 'object' &&
1399
+ !Array.isArray(manifest.commands) &&
1400
+ firstValue &&
1401
+ typeof firstValue === 'object' &&
1402
+ ('source' in firstValue || 'content' in firstValue)
1403
+ ) {
1404
+ // Object mapping format: { "about": { "source": "./README.md", ... } }
1405
+ const commandsMetadata: Record<string, CommandMetadata> = {}
1406
+ const validPaths: string[] = []
1407
+
1408
+ // Parallelize pathExists checks; process results in order to keep
1409
+ // error/log ordering deterministic.
1410
+ const entries = Object.entries(manifest.commands)
1411
+ const checks = await Promise.all(
1412
+ entries.map(async ([commandName, metadata]) => {
1413
+ if (!metadata || typeof metadata !== 'object') {
1414
+ return { commandName, metadata, kind: 'skip' as const }
1415
+ }
1416
+ if (metadata.source) {
1417
+ const fullPath = join(pluginPath, metadata.source)
1418
+ return {
1419
+ commandName,
1420
+ metadata,
1421
+ kind: 'source' as const,
1422
+ fullPath,
1423
+ exists: await pathExists(fullPath),
1424
+ }
1425
+ }
1426
+ if (metadata.content) {
1427
+ return { commandName, metadata, kind: 'content' as const }
1428
+ }
1429
+ return { commandName, metadata, kind: 'skip' as const }
1430
+ }),
1431
+ )
1432
+ for (const check of checks) {
1433
+ if (check.kind === 'skip') continue
1434
+ if (check.kind === 'content') {
1435
+ // For inline content commands, add metadata without path
1436
+ commandsMetadata[check.commandName] = check.metadata
1437
+ continue
1438
+ }
1439
+ // kind === 'source'
1440
+ if (check.exists) {
1441
+ validPaths.push(check.fullPath)
1442
+ commandsMetadata[check.commandName] = check.metadata
1443
+ } else {
1444
+ logForDebugging(
1445
+ `Command ${check.commandName} path ${check.metadata.source} specified in manifest but not found at ${check.fullPath} for ${manifest.name}`,
1446
+ { level: 'warn' },
1447
+ )
1448
+ logError(
1449
+ new Error(
1450
+ `Plugin component file not found: ${check.fullPath} for ${manifest.name}`,
1451
+ ),
1452
+ )
1453
+ errors.push({
1454
+ type: 'path-not-found',
1455
+ source,
1456
+ plugin: manifest.name,
1457
+ path: check.fullPath,
1458
+ component: 'commands',
1459
+ })
1460
+ }
1461
+ }
1462
+
1463
+ // Set commandsPaths if there are file-based commands
1464
+ if (validPaths.length > 0) {
1465
+ plugin.commandsPaths = validPaths
1466
+ }
1467
+ // Set commandsMetadata if there are any commands (file-based or inline)
1468
+ if (Object.keys(commandsMetadata).length > 0) {
1469
+ plugin.commandsMetadata = commandsMetadata
1470
+ }
1471
+ } else {
1472
+ // Path or array of paths format
1473
+ const commandPaths = Array.isArray(manifest.commands)
1474
+ ? manifest.commands
1475
+ : [manifest.commands]
1476
+
1477
+ // Parallelize pathExists checks; process results in order.
1478
+ const checks = await Promise.all(
1479
+ commandPaths.map(async cmdPath => {
1480
+ if (typeof cmdPath !== 'string') {
1481
+ return { cmdPath, kind: 'invalid' as const }
1482
+ }
1483
+ const fullPath = join(pluginPath, cmdPath)
1484
+ return {
1485
+ cmdPath,
1486
+ kind: 'path' as const,
1487
+ fullPath,
1488
+ exists: await pathExists(fullPath),
1489
+ }
1490
+ }),
1491
+ )
1492
+ const validPaths: string[] = []
1493
+ for (const check of checks) {
1494
+ if (check.kind === 'invalid') {
1495
+ logForDebugging(
1496
+ `Unexpected command format in manifest for ${manifest.name}`,
1497
+ { level: 'error' },
1498
+ )
1499
+ continue
1500
+ }
1501
+ if (check.exists) {
1502
+ validPaths.push(check.fullPath)
1503
+ } else {
1504
+ logForDebugging(
1505
+ `Command path ${check.cmdPath} specified in manifest but not found at ${check.fullPath} for ${manifest.name}`,
1506
+ { level: 'warn' },
1507
+ )
1508
+ logError(
1509
+ new Error(
1510
+ `Plugin component file not found: ${check.fullPath} for ${manifest.name}`,
1511
+ ),
1512
+ )
1513
+ errors.push({
1514
+ type: 'path-not-found',
1515
+ source,
1516
+ plugin: manifest.name,
1517
+ path: check.fullPath,
1518
+ component: 'commands',
1519
+ })
1520
+ }
1521
+ }
1522
+
1523
+ if (validPaths.length > 0) {
1524
+ plugin.commandsPaths = validPaths
1525
+ }
1526
+ }
1527
+ }
1528
+
1529
+ // Step 4: Register agents directory if detected
1530
+ const agentsPath = join(pluginPath, 'agents')
1531
+ if (agentsDirExists) {
1532
+ plugin.agentsPath = agentsPath
1533
+ }
1534
+
1535
+ // Step 4a: Process additional agent paths from manifest
1536
+ if (manifest.agents) {
1537
+ const agentPaths = Array.isArray(manifest.agents)
1538
+ ? manifest.agents
1539
+ : [manifest.agents]
1540
+
1541
+ const validPaths = await validatePluginPaths(
1542
+ agentPaths,
1543
+ pluginPath,
1544
+ manifest.name,
1545
+ source,
1546
+ 'agents',
1547
+ 'Agent',
1548
+ 'specified in manifest but',
1549
+ errors,
1550
+ )
1551
+
1552
+ if (validPaths.length > 0) {
1553
+ plugin.agentsPaths = validPaths
1554
+ }
1555
+ }
1556
+
1557
+ // Step 4b: Register skills directory if detected
1558
+ const skillsPath = join(pluginPath, 'skills')
1559
+ if (skillsDirExists) {
1560
+ plugin.skillsPath = skillsPath
1561
+ }
1562
+
1563
+ // Step 4c: Process additional skill paths from manifest
1564
+ if (manifest.skills) {
1565
+ const skillPaths = Array.isArray(manifest.skills)
1566
+ ? manifest.skills
1567
+ : [manifest.skills]
1568
+
1569
+ const validPaths = await validatePluginPaths(
1570
+ skillPaths,
1571
+ pluginPath,
1572
+ manifest.name,
1573
+ source,
1574
+ 'skills',
1575
+ 'Skill',
1576
+ 'specified in manifest but',
1577
+ errors,
1578
+ )
1579
+
1580
+ if (validPaths.length > 0) {
1581
+ plugin.skillsPaths = validPaths
1582
+ }
1583
+ }
1584
+
1585
+ // Step 4d: Register output-styles directory if detected
1586
+ const outputStylesPath = join(pluginPath, 'output-styles')
1587
+ if (outputStylesDirExists) {
1588
+ plugin.outputStylesPath = outputStylesPath
1589
+ }
1590
+
1591
+ // Step 4e: Process additional output style paths from manifest
1592
+ if (manifest.outputStyles) {
1593
+ const outputStylePaths = Array.isArray(manifest.outputStyles)
1594
+ ? manifest.outputStyles
1595
+ : [manifest.outputStyles]
1596
+
1597
+ const validPaths = await validatePluginPaths(
1598
+ outputStylePaths,
1599
+ pluginPath,
1600
+ manifest.name,
1601
+ source,
1602
+ 'output-styles',
1603
+ 'Output style',
1604
+ 'specified in manifest but',
1605
+ errors,
1606
+ )
1607
+
1608
+ if (validPaths.length > 0) {
1609
+ plugin.outputStylesPaths = validPaths
1610
+ }
1611
+ }
1612
+
1613
+ // Step 5: Load hooks configuration
1614
+ let mergedHooks: HooksSettings | undefined
1615
+ const loadedHookPaths = new Set<string>() // Track loaded hook files
1616
+
1617
+ // Load from standard hooks/hooks.json if it exists
1618
+ const standardHooksPath = join(pluginPath, 'hooks', 'hooks.json')
1619
+ if (await pathExists(standardHooksPath)) {
1620
+ try {
1621
+ mergedHooks = await loadPluginHooks(standardHooksPath, manifest.name)
1622
+ // Track the normalized path to prevent duplicate loading
1623
+ try {
1624
+ loadedHookPaths.add(await realpath(standardHooksPath))
1625
+ } catch {
1626
+ // If realpathSync fails, use original path
1627
+ loadedHookPaths.add(standardHooksPath)
1628
+ }
1629
+ logForDebugging(
1630
+ `Loaded hooks from standard location for plugin ${manifest.name}: ${standardHooksPath}`,
1631
+ )
1632
+ } catch (error) {
1633
+ const errorMsg = errorMessage(error)
1634
+ logForDebugging(
1635
+ `Failed to load hooks for ${manifest.name}: ${errorMsg}`,
1636
+ {
1637
+ level: 'error',
1638
+ },
1639
+ )
1640
+ logError(toError(error))
1641
+ errors.push({
1642
+ type: 'hook-load-failed',
1643
+ source,
1644
+ plugin: manifest.name,
1645
+ hookPath: standardHooksPath,
1646
+ reason: errorMsg,
1647
+ })
1648
+ }
1649
+ }
1650
+
1651
+ // Load and merge hooks from manifest.hooks if specified
1652
+ if (manifest.hooks) {
1653
+ const manifestHooksArray = Array.isArray(manifest.hooks)
1654
+ ? manifest.hooks
1655
+ : [manifest.hooks]
1656
+
1657
+ for (const hookSpec of manifestHooksArray) {
1658
+ if (typeof hookSpec === 'string') {
1659
+ // Path to additional hooks file
1660
+ const hookFilePath = join(pluginPath, hookSpec)
1661
+ if (!(await pathExists(hookFilePath))) {
1662
+ logForDebugging(
1663
+ `Hooks file ${hookSpec} specified in manifest but not found at ${hookFilePath} for ${manifest.name}`,
1664
+ { level: 'error' },
1665
+ )
1666
+ logError(
1667
+ new Error(
1668
+ `Plugin component file not found: ${hookFilePath} for ${manifest.name}`,
1669
+ ),
1670
+ )
1671
+ errors.push({
1672
+ type: 'path-not-found',
1673
+ source,
1674
+ plugin: manifest.name,
1675
+ path: hookFilePath,
1676
+ component: 'hooks',
1677
+ })
1678
+ continue
1679
+ }
1680
+
1681
+ // Check if this path resolves to an already-loaded hooks file
1682
+ let normalizedPath: string
1683
+ try {
1684
+ normalizedPath = await realpath(hookFilePath)
1685
+ } catch {
1686
+ // If realpathSync fails, use original path
1687
+ normalizedPath = hookFilePath
1688
+ }
1689
+
1690
+ if (loadedHookPaths.has(normalizedPath)) {
1691
+ logForDebugging(
1692
+ `Skipping duplicate hooks file for plugin ${manifest.name}: ${hookSpec} ` +
1693
+ `(resolves to already-loaded file: ${normalizedPath})`,
1694
+ )
1695
+ if (strict) {
1696
+ const errorMsg = `Duplicate hooks file detected: ${hookSpec} resolves to already-loaded file ${normalizedPath}. The standard hooks/hooks.json is loaded automatically, so manifest.hooks should only reference additional hook files.`
1697
+ logError(new Error(errorMsg))
1698
+ errors.push({
1699
+ type: 'hook-load-failed',
1700
+ source,
1701
+ plugin: manifest.name,
1702
+ hookPath: hookFilePath,
1703
+ reason: errorMsg,
1704
+ })
1705
+ }
1706
+ continue
1707
+ }
1708
+
1709
+ try {
1710
+ const additionalHooks = await loadPluginHooks(
1711
+ hookFilePath,
1712
+ manifest.name,
1713
+ )
1714
+ try {
1715
+ mergedHooks = mergeHooksSettings(mergedHooks, additionalHooks)
1716
+ loadedHookPaths.add(normalizedPath)
1717
+ logForDebugging(
1718
+ `Loaded and merged hooks from manifest for plugin ${manifest.name}: ${hookSpec}`,
1719
+ )
1720
+ } catch (mergeError) {
1721
+ const mergeErrorMsg = errorMessage(mergeError)
1722
+ logForDebugging(
1723
+ `Failed to merge hooks from ${hookSpec} for ${manifest.name}: ${mergeErrorMsg}`,
1724
+ { level: 'error' },
1725
+ )
1726
+ logError(toError(mergeError))
1727
+ errors.push({
1728
+ type: 'hook-load-failed',
1729
+ source,
1730
+ plugin: manifest.name,
1731
+ hookPath: hookFilePath,
1732
+ reason: `Failed to merge: ${mergeErrorMsg}`,
1733
+ })
1734
+ }
1735
+ } catch (error) {
1736
+ const errorMsg = errorMessage(error)
1737
+ logForDebugging(
1738
+ `Failed to load hooks from ${hookSpec} for ${manifest.name}: ${errorMsg}`,
1739
+ { level: 'error' },
1740
+ )
1741
+ logError(toError(error))
1742
+ errors.push({
1743
+ type: 'hook-load-failed',
1744
+ source,
1745
+ plugin: manifest.name,
1746
+ hookPath: hookFilePath,
1747
+ reason: errorMsg,
1748
+ })
1749
+ }
1750
+ } else if (typeof hookSpec === 'object') {
1751
+ // Inline hooks
1752
+ mergedHooks = mergeHooksSettings(mergedHooks, hookSpec as HooksSettings)
1753
+ }
1754
+ }
1755
+ }
1756
+
1757
+ if (mergedHooks) {
1758
+ plugin.hooksConfig = mergedHooks
1759
+ }
1760
+
1761
+ // Step 6: Load plugin settings
1762
+ // Settings can come from settings.json in the plugin directory or from manifest.settings
1763
+ // Only allowlisted keys are kept (currently: agent)
1764
+ const pluginSettings = await loadPluginSettings(pluginPath, manifest)
1765
+ if (pluginSettings) {
1766
+ plugin.settings = pluginSettings
1767
+ }
1768
+
1769
+ return { plugin, errors }
1770
+ }
1771
+
1772
+ /**
1773
+ * Schema derived from SettingsSchema that only keeps keys plugins are allowed to set.
1774
+ * Uses .strip() so unknown keys are silently removed during parsing.
1775
+ */
1776
+ const PluginSettingsSchema = lazySchema(() =>
1777
+ SettingsSchema()
1778
+ .pick({
1779
+ agent: true,
1780
+ })
1781
+ .strip(),
1782
+ )
1783
+
1784
+ /**
1785
+ * Parse raw settings through PluginSettingsSchema, returning only allowlisted keys.
1786
+ * Returns undefined if parsing fails or all keys are filtered out.
1787
+ */
1788
+ function parsePluginSettings(
1789
+ raw: Record<string, unknown>,
1790
+ ): Record<string, unknown> | undefined {
1791
+ const result = PluginSettingsSchema().safeParse(raw)
1792
+ if (!result.success) {
1793
+ return undefined
1794
+ }
1795
+ const data = result.data
1796
+ if (Object.keys(data).length === 0) {
1797
+ return undefined
1798
+ }
1799
+ return data
1800
+ }
1801
+
1802
+ /**
1803
+ * Load plugin settings from settings.json file or manifest.settings.
1804
+ * settings.json takes priority over manifest.settings when both exist.
1805
+ * Only allowlisted keys are included in the result.
1806
+ */
1807
+ async function loadPluginSettings(
1808
+ pluginPath: string,
1809
+ manifest: PluginManifest,
1810
+ ): Promise<Record<string, unknown> | undefined> {
1811
+ // Try loading settings.json from the plugin directory
1812
+ const settingsJsonPath = join(pluginPath, 'settings.json')
1813
+ try {
1814
+ const content = await readFile(settingsJsonPath, { encoding: 'utf-8' })
1815
+ const parsed = jsonParse(content)
1816
+ if (isRecord(parsed)) {
1817
+ const filtered = parsePluginSettings(parsed)
1818
+ if (filtered) {
1819
+ logForDebugging(
1820
+ `Loaded settings from settings.json for plugin ${manifest.name}`,
1821
+ )
1822
+ return filtered
1823
+ }
1824
+ }
1825
+ } catch (e: unknown) {
1826
+ // Missing/inaccessible is expected - settings.json is optional
1827
+ if (!isFsInaccessible(e)) {
1828
+ logForDebugging(
1829
+ `Failed to parse settings.json for plugin ${manifest.name}: ${e}`,
1830
+ { level: 'warn' },
1831
+ )
1832
+ }
1833
+ }
1834
+
1835
+ // Fall back to manifest.settings
1836
+ if (manifest.settings) {
1837
+ const filtered = parsePluginSettings(
1838
+ manifest.settings as Record<string, unknown>,
1839
+ )
1840
+ if (filtered) {
1841
+ logForDebugging(
1842
+ `Loaded settings from manifest for plugin ${manifest.name}`,
1843
+ )
1844
+ return filtered
1845
+ }
1846
+ }
1847
+
1848
+ return undefined
1849
+ }
1850
+
1851
+ /**
1852
+ * Merge two HooksSettings objects
1853
+ */
1854
+ function mergeHooksSettings(
1855
+ base: HooksSettings | undefined,
1856
+ additional: HooksSettings,
1857
+ ): HooksSettings {
1858
+ if (!base) {
1859
+ return additional
1860
+ }
1861
+
1862
+ const merged = { ...base }
1863
+
1864
+ for (const [event, matchers] of Object.entries(additional)) {
1865
+ if (!merged[event as keyof HooksSettings]) {
1866
+ merged[event as keyof HooksSettings] = matchers
1867
+ } else {
1868
+ // Merge matchers for this event
1869
+ merged[event as keyof HooksSettings] = [
1870
+ ...(merged[event as keyof HooksSettings] || []),
1871
+ ...matchers,
1872
+ ]
1873
+ }
1874
+ }
1875
+
1876
+ return merged
1877
+ }
1878
+
1879
+ /**
1880
+ * Shared discovery/policy/merge pipeline for both load modes.
1881
+ *
1882
+ * Resolves enabledPlugins → marketplace entries, runs enterprise policy
1883
+ * checks, pre-loads catalogs, then dispatches each entry to the full or
1884
+ * cache-only per-entry loader. The ONLY difference between loadAllPlugins
1885
+ * and loadAllPluginsCacheOnly is which loader runs — discovery and policy
1886
+ * are identical.
1887
+ */
1888
+ async function loadPluginsFromMarketplaces({
1889
+ cacheOnly,
1890
+ }: {
1891
+ cacheOnly: boolean
1892
+ }): Promise<{
1893
+ plugins: LoadedPlugin[]
1894
+ errors: PluginError[]
1895
+ }> {
1896
+ const settings = getSettings_DEPRECATED()
1897
+ // Merge --add-dir plugins at lowest priority; standard settings win on conflict
1898
+ const enabledPlugins = {
1899
+ ...getAddDirEnabledPlugins(),
1900
+ ...(settings.enabledPlugins || {}),
1901
+ }
1902
+ const plugins: LoadedPlugin[] = []
1903
+ const errors: PluginError[] = []
1904
+
1905
+ // Filter to plugin@marketplace format and validate
1906
+ const marketplacePluginEntries = Object.entries(enabledPlugins).filter(
1907
+ ([key, value]) => {
1908
+ // Check if it's in plugin@marketplace format (includes both enabled and disabled)
1909
+ const isValidFormat = PluginIdSchema().safeParse(key).success
1910
+ if (!isValidFormat || value === undefined) return false
1911
+ // Skip built-in plugins — handled separately by getBuiltinPlugins()
1912
+ const { marketplace } = parsePluginIdentifier(key)
1913
+ return marketplace !== BUILTIN_MARKETPLACE_NAME
1914
+ },
1915
+ )
1916
+
1917
+ // Load known marketplaces config to look up sources for policy checking.
1918
+ // Use the Safe variant so a corrupted config file doesn't crash all plugin
1919
+ // loading — this is a read-only path, so returning {} degrades gracefully.
1920
+ const knownMarketplaces = await loadKnownMarketplacesConfigSafe()
1921
+
1922
+ // Fail-closed guard for enterprise policy: if a policy IS configured and we
1923
+ // cannot resolve a marketplace's source (config returned {} due to corruption,
1924
+ // or entry missing), we must NOT silently skip the policy check and load the
1925
+ // plugin anyway. Before Safe, a corrupted config crashed everything (loud,
1926
+ // fail-closed). With Safe + no guard, the policy check short-circuits on
1927
+ // undefined marketplaceConfig and the fallback path (getPluginByIdCacheOnly)
1928
+ // loads the plugin unchecked — a silent fail-open. This guard restores
1929
+ // fail-closed: unknown source + active policy → block.
1930
+ //
1931
+ // Allowlist: any value (including []) is active — empty allowlist = deny all.
1932
+ // Blocklist: empty [] is a semantic no-op — only non-empty counts as active.
1933
+ const strictAllowlist = getStrictKnownMarketplaces()
1934
+ const blocklist = getBlockedMarketplaces()
1935
+ const hasEnterprisePolicy =
1936
+ strictAllowlist !== null || (blocklist !== null && blocklist.length > 0)
1937
+
1938
+ // Pre-load marketplace catalogs once per marketplace rather than re-reading
1939
+ // known_marketplaces.json + marketplace.json for every plugin. This is the
1940
+ // hot path — with N plugins across M marketplaces, the old per-plugin
1941
+ // getPluginByIdCacheOnly() did 2N config reads + N catalog reads; this does M.
1942
+ const uniqueMarketplaces = new Set(
1943
+ marketplacePluginEntries
1944
+ .map(([pluginId]) => parsePluginIdentifier(pluginId).marketplace)
1945
+ .filter((m): m is string => !!m),
1946
+ )
1947
+ const marketplaceCatalogs = new Map<
1948
+ string,
1949
+ Awaited<ReturnType<typeof getMarketplaceCacheOnly>>
1950
+ >()
1951
+ await Promise.all(
1952
+ [...uniqueMarketplaces].map(async name => {
1953
+ marketplaceCatalogs.set(name, await getMarketplaceCacheOnly(name))
1954
+ }),
1955
+ )
1956
+
1957
+ // Look up installed versions once so the first-pass ZIP cache check
1958
+ // can hit even when the marketplace entry omits `version`.
1959
+ const installedPluginsData = getInMemoryInstalledPlugins()
1960
+
1961
+ // Load all marketplace plugins in parallel for faster startup
1962
+ const results = await Promise.allSettled(
1963
+ marketplacePluginEntries.map(async ([pluginId, enabledValue]) => {
1964
+ const { name: pluginName, marketplace: marketplaceName } =
1965
+ parsePluginIdentifier(pluginId)
1966
+
1967
+ // Check if marketplace source is allowed by enterprise policy
1968
+ const marketplaceConfig = knownMarketplaces[marketplaceName!]
1969
+
1970
+ // Fail-closed: if enterprise policy is active and we can't look up the
1971
+ // marketplace source (config corrupted/empty, or entry missing), block
1972
+ // rather than silently skip the policy check. See hasEnterprisePolicy
1973
+ // comment above for the fail-open hazard this guards against.
1974
+ //
1975
+ // This also fires for the "stale enabledPlugins entry with no registered
1976
+ // marketplace" case, which is a UX trade-off: the user gets a policy
1977
+ // error instead of plugin-not-found. Accepted because the fallback path
1978
+ // (getPluginByIdCacheOnly) does a raw cast of known_marketplaces.json
1979
+ // with NO schema validation — if one entry is malformed enough to fail
1980
+ // our validation but readable enough for the raw cast, it would load
1981
+ // unchecked. Unverifiable source + active policy → block, always.
1982
+ if (!marketplaceConfig && hasEnterprisePolicy) {
1983
+ // We can't know whether the unverifiable source would actually be in
1984
+ // the blocklist or not in the allowlist — so pick the error variant
1985
+ // that matches whichever policy IS configured. If an allowlist exists,
1986
+ // "not in allowed list" is the right framing; if only a blocklist
1987
+ // exists, "blocked by blocklist" is less misleading than showing an
1988
+ // empty allowed-sources list.
1989
+ errors.push({
1990
+ type: 'marketplace-blocked-by-policy',
1991
+ source: pluginId,
1992
+ plugin: pluginName,
1993
+ marketplace: marketplaceName!,
1994
+ blockedByBlocklist: strictAllowlist === null,
1995
+ allowedSources: (strictAllowlist ?? []).map(s =>
1996
+ formatSourceForDisplay(s),
1997
+ ),
1998
+ })
1999
+ return null
2000
+ }
2001
+
2002
+ if (
2003
+ marketplaceConfig &&
2004
+ !isSourceAllowedByPolicy(marketplaceConfig.source)
2005
+ ) {
2006
+ // Check if explicitly blocked vs not in allowlist for better error context
2007
+ const isBlocked = isSourceInBlocklist(marketplaceConfig.source)
2008
+ const allowlist = getStrictKnownMarketplaces() || []
2009
+ errors.push({
2010
+ type: 'marketplace-blocked-by-policy',
2011
+ source: pluginId,
2012
+ plugin: pluginName,
2013
+ marketplace: marketplaceName!,
2014
+ blockedByBlocklist: isBlocked,
2015
+ allowedSources: isBlocked
2016
+ ? []
2017
+ : allowlist.map(s => formatSourceForDisplay(s)),
2018
+ })
2019
+ return null
2020
+ }
2021
+
2022
+ // Look up plugin entry from pre-loaded marketplace catalog (no per-plugin I/O).
2023
+ // Fall back to getPluginByIdCacheOnly if the catalog couldn't be pre-loaded.
2024
+ let result: Awaited<ReturnType<typeof getPluginByIdCacheOnly>> = null
2025
+ const marketplace = marketplaceCatalogs.get(marketplaceName!)
2026
+ if (marketplace && marketplaceConfig) {
2027
+ const entry = marketplace.plugins.find(p => p.name === pluginName)
2028
+ if (entry) {
2029
+ result = {
2030
+ entry,
2031
+ marketplaceInstallLocation: marketplaceConfig.installLocation,
2032
+ }
2033
+ }
2034
+ } else {
2035
+ result = await getPluginByIdCacheOnly(pluginId)
2036
+ }
2037
+
2038
+ if (!result) {
2039
+ errors.push({
2040
+ type: 'plugin-not-found',
2041
+ source: pluginId,
2042
+ pluginId: pluginName!,
2043
+ marketplace: marketplaceName!,
2044
+ })
2045
+ return null
2046
+ }
2047
+
2048
+ // installed_plugins.json records what's actually cached on disk
2049
+ // (version for the full loader's first-pass probe, installPath for
2050
+ // the cache-only loader's direct read).
2051
+ const installEntry = installedPluginsData.plugins[pluginId]?.[0]
2052
+ return cacheOnly
2053
+ ? loadPluginFromMarketplaceEntryCacheOnly(
2054
+ result.entry,
2055
+ result.marketplaceInstallLocation,
2056
+ pluginId,
2057
+ enabledValue === true,
2058
+ errors,
2059
+ installEntry?.installPath,
2060
+ )
2061
+ : loadPluginFromMarketplaceEntry(
2062
+ result.entry,
2063
+ result.marketplaceInstallLocation,
2064
+ pluginId,
2065
+ enabledValue === true,
2066
+ errors,
2067
+ installEntry?.version,
2068
+ )
2069
+ }),
2070
+ )
2071
+
2072
+ for (const [i, result] of results.entries()) {
2073
+ if (result.status === 'fulfilled' && result.value) {
2074
+ plugins.push(result.value)
2075
+ } else if (result.status === 'rejected') {
2076
+ const err = toError(result.reason)
2077
+ logError(err)
2078
+ const pluginId = marketplacePluginEntries[i]![0]
2079
+ errors.push({
2080
+ type: 'generic-error',
2081
+ source: pluginId,
2082
+ plugin: pluginId.split('@')[0],
2083
+ error: err.message,
2084
+ })
2085
+ }
2086
+ }
2087
+
2088
+ return { plugins, errors }
2089
+ }
2090
+
2091
+ /**
2092
+ * Cache-only variant of loadPluginFromMarketplaceEntry.
2093
+ *
2094
+ * Skips network (cachePlugin) and disk-copy (copyPluginToVersionedCache).
2095
+ * Reads directly from the recorded installPath; if missing, emits
2096
+ * 'plugin-cache-miss'. Still extracts ZIP-cached plugins (local, fast).
2097
+ */
2098
+ async function loadPluginFromMarketplaceEntryCacheOnly(
2099
+ entry: PluginMarketplaceEntry,
2100
+ marketplaceInstallLocation: string,
2101
+ pluginId: string,
2102
+ enabled: boolean,
2103
+ errorsOut: PluginError[],
2104
+ installPath: string | undefined,
2105
+ ): Promise<LoadedPlugin | null> {
2106
+ let pluginPath: string
2107
+
2108
+ if (typeof entry.source === 'string') {
2109
+ // Local relative path — read from the marketplace source dir directly.
2110
+ // Skip copyPluginToVersionedCache; startup doesn't need a fresh copy.
2111
+ let marketplaceDir: string
2112
+ try {
2113
+ marketplaceDir = (await stat(marketplaceInstallLocation)).isDirectory()
2114
+ ? marketplaceInstallLocation
2115
+ : join(marketplaceInstallLocation, '..')
2116
+ } catch {
2117
+ errorsOut.push({
2118
+ type: 'plugin-cache-miss',
2119
+ source: pluginId,
2120
+ plugin: entry.name,
2121
+ installPath: marketplaceInstallLocation,
2122
+ })
2123
+ return null
2124
+ }
2125
+ pluginPath = join(marketplaceDir, entry.source)
2126
+ // finishLoadingPluginFromPath reads pluginPath — its error handling
2127
+ // surfaces ENOENT as a load failure, no need to pre-check here.
2128
+ } else {
2129
+ // External source (npm/github/url/git-subdir) — use recorded installPath.
2130
+ if (!installPath || !(await pathExists(installPath))) {
2131
+ errorsOut.push({
2132
+ type: 'plugin-cache-miss',
2133
+ source: pluginId,
2134
+ plugin: entry.name,
2135
+ installPath: installPath ?? '(not recorded)',
2136
+ })
2137
+ return null
2138
+ }
2139
+ pluginPath = installPath
2140
+ }
2141
+
2142
+ // Zip cache extraction — must still happen in cacheOnly mode (invariant 4)
2143
+ if (isPluginZipCacheEnabled() && pluginPath.endsWith('.zip')) {
2144
+ const sessionDir = await getSessionPluginCachePath()
2145
+ const extractDir = join(
2146
+ sessionDir,
2147
+ pluginId.replace(/[^a-zA-Z0-9@\-_]/g, '-'),
2148
+ )
2149
+ try {
2150
+ await extractZipToDirectory(pluginPath, extractDir)
2151
+ pluginPath = extractDir
2152
+ } catch (error) {
2153
+ logForDebugging(`Failed to extract plugin ZIP ${pluginPath}: ${error}`, {
2154
+ level: 'error',
2155
+ })
2156
+ errorsOut.push({
2157
+ type: 'plugin-cache-miss',
2158
+ source: pluginId,
2159
+ plugin: entry.name,
2160
+ installPath: pluginPath,
2161
+ })
2162
+ return null
2163
+ }
2164
+ }
2165
+
2166
+ // Delegate to the shared tail — identical to the full loader from here
2167
+ return finishLoadingPluginFromPath(
2168
+ entry,
2169
+ pluginId,
2170
+ enabled,
2171
+ errorsOut,
2172
+ pluginPath,
2173
+ )
2174
+ }
2175
+
2176
+ /**
2177
+ * Load a plugin from a marketplace entry based on its source configuration.
2178
+ *
2179
+ * Handles different source types:
2180
+ * - Relative path: Loads from marketplace repo directory
2181
+ * - npm/github/url: Caches then loads from cache
2182
+ *
2183
+ * @param installedVersion - Version from installed_plugins.json, used as a
2184
+ * first-pass hint for the versioned cache lookup when the marketplace entry
2185
+ * omits `version`. Avoids re-cloning external plugins just to discover the
2186
+ * version we already recorded at install time.
2187
+ *
2188
+ * Returns both the loaded plugin and any errors encountered during loading.
2189
+ * Errors include missing component files and hook load failures.
2190
+ */
2191
+ async function loadPluginFromMarketplaceEntry(
2192
+ entry: PluginMarketplaceEntry,
2193
+ marketplaceInstallLocation: string,
2194
+ pluginId: string,
2195
+ enabled: boolean,
2196
+ errorsOut: PluginError[],
2197
+ installedVersion?: string,
2198
+ ): Promise<LoadedPlugin | null> {
2199
+ logForDebugging(
2200
+ `Loading plugin ${entry.name} from source: ${jsonStringify(entry.source)}`,
2201
+ )
2202
+ let pluginPath: string
2203
+
2204
+ if (typeof entry.source === 'string') {
2205
+ // Relative path - resolve relative to marketplace install location
2206
+ const marketplaceDir = (
2207
+ await stat(marketplaceInstallLocation)
2208
+ ).isDirectory()
2209
+ ? marketplaceInstallLocation
2210
+ : join(marketplaceInstallLocation, '..')
2211
+ const sourcePluginPath = join(marketplaceDir, entry.source)
2212
+
2213
+ if (!(await pathExists(sourcePluginPath))) {
2214
+ const error = new Error(`Plugin path not found: ${sourcePluginPath}`)
2215
+ logForDebugging(`Plugin path not found: ${sourcePluginPath}`, {
2216
+ level: 'error',
2217
+ })
2218
+ logError(error)
2219
+ errorsOut.push({
2220
+ type: 'generic-error',
2221
+ source: pluginId,
2222
+ error: `Plugin directory not found at path: ${sourcePluginPath}. Check that the marketplace entry has the correct path.`,
2223
+ })
2224
+ return null
2225
+ }
2226
+
2227
+ // Always copy local plugins to versioned cache
2228
+ try {
2229
+ // Try to load manifest from plugin directory to check for version field first
2230
+ const manifestPath = join(
2231
+ sourcePluginPath,
2232
+ '.claude-plugin',
2233
+ 'plugin.json',
2234
+ )
2235
+ let pluginManifest: PluginManifest | undefined
2236
+ try {
2237
+ pluginManifest = await loadPluginManifest(
2238
+ manifestPath,
2239
+ entry.name,
2240
+ entry.source,
2241
+ )
2242
+ } catch {
2243
+ // Manifest loading failed - will fall back to provided version or git SHA
2244
+ }
2245
+
2246
+ // Calculate version with fallback order:
2247
+ // 1. Plugin manifest version, 2. Marketplace entry version, 3. Git SHA, 4. 'unknown'
2248
+ const version = await calculatePluginVersion(
2249
+ pluginId,
2250
+ entry.source,
2251
+ pluginManifest,
2252
+ marketplaceDir,
2253
+ entry.version, // Marketplace entry version as fallback
2254
+ )
2255
+
2256
+ // Copy to versioned cache
2257
+ pluginPath = await copyPluginToVersionedCache(
2258
+ sourcePluginPath,
2259
+ pluginId,
2260
+ version,
2261
+ entry,
2262
+ marketplaceDir,
2263
+ )
2264
+
2265
+ logForDebugging(
2266
+ `Resolved local plugin ${entry.name} to versioned cache: ${pluginPath}`,
2267
+ )
2268
+ } catch (error) {
2269
+ // If copy fails, fall back to loading from marketplace directly
2270
+ const errorMsg = errorMessage(error)
2271
+ logForDebugging(
2272
+ `Failed to copy plugin ${entry.name} to versioned cache: ${errorMsg}. Using marketplace path.`,
2273
+ { level: 'warn' },
2274
+ )
2275
+ pluginPath = sourcePluginPath
2276
+ }
2277
+ } else {
2278
+ // External source (npm, github, url, pip) - always use versioned cache
2279
+ try {
2280
+ // Calculate version with fallback order:
2281
+ // 1. No manifest yet, 2. installed_plugins.json version,
2282
+ // 3. Marketplace entry version, 4. source.sha (pinned commits — the
2283
+ // exact value the post-clone call at cached.gitCommitSha would see),
2284
+ // 5. 'unknown' → ref-tracked, falls through to clone by design.
2285
+ const version = await calculatePluginVersion(
2286
+ pluginId,
2287
+ entry.source,
2288
+ undefined,
2289
+ undefined,
2290
+ installedVersion ?? entry.version,
2291
+ 'sha' in entry.source ? entry.source.sha : undefined,
2292
+ )
2293
+
2294
+ const versionedPath = getVersionedCachePath(pluginId, version)
2295
+
2296
+ // Check for cached version — ZIP file (zip cache mode) or directory
2297
+ const zipPath = getVersionedZipCachePath(pluginId, version)
2298
+ if (isPluginZipCacheEnabled() && (await pathExists(zipPath))) {
2299
+ logForDebugging(
2300
+ `Using versioned cached plugin ZIP ${entry.name} from ${zipPath}`,
2301
+ )
2302
+ pluginPath = zipPath
2303
+ } else if (await pathExists(versionedPath)) {
2304
+ logForDebugging(
2305
+ `Using versioned cached plugin ${entry.name} from ${versionedPath}`,
2306
+ )
2307
+ pluginPath = versionedPath
2308
+ } else {
2309
+ // Seed cache probe (CCR pre-baked images, read-only). Seed content is
2310
+ // frozen at image build time — no freshness concern, 'whatever's there'
2311
+ // is what the image builder put there. Primary cache is NOT probed
2312
+ // here; ref-tracked sources fall through to clone (the re-clone IS
2313
+ // the freshness mechanism). If the clone fails, the plugin is simply
2314
+ // disabled for this session — errorsOut.push below surfaces it.
2315
+ const seedPath =
2316
+ (await probeSeedCache(pluginId, version)) ??
2317
+ (version === 'unknown'
2318
+ ? await probeSeedCacheAnyVersion(pluginId)
2319
+ : null)
2320
+ if (seedPath) {
2321
+ pluginPath = seedPath
2322
+ logForDebugging(
2323
+ `Using seed cache for external plugin ${entry.name} at ${seedPath}`,
2324
+ )
2325
+ } else {
2326
+ // Download to temp location, then copy to versioned cache
2327
+ const cached = await cachePlugin(entry.source, {
2328
+ manifest: { name: entry.name },
2329
+ })
2330
+
2331
+ // If the pre-clone version was deterministic (source.sha /
2332
+ // entry.version / installedVersion), REUSE it. The post-clone
2333
+ // recomputation with cached.manifest can return a DIFFERENT value
2334
+ // — manifest.version (step 1) outranks gitCommitSha (step 3) —
2335
+ // which would cache at e.g. "2.0.0/" while every warm start
2336
+ // probes "{sha12}-{hash}/". Mismatched keys = re-clone forever.
2337
+ // Recomputation is only needed when pre-clone was 'unknown'
2338
+ // (ref-tracked, no hints) — the clone is the ONLY way to learn.
2339
+ const actualVersion =
2340
+ version !== 'unknown'
2341
+ ? version
2342
+ : await calculatePluginVersion(
2343
+ pluginId,
2344
+ entry.source,
2345
+ cached.manifest,
2346
+ cached.path,
2347
+ installedVersion ?? entry.version,
2348
+ cached.gitCommitSha,
2349
+ )
2350
+
2351
+ // Copy to versioned cache
2352
+ // For external sources, marketplaceDir is not applicable (already downloaded)
2353
+ pluginPath = await copyPluginToVersionedCache(
2354
+ cached.path,
2355
+ pluginId,
2356
+ actualVersion,
2357
+ entry,
2358
+ undefined,
2359
+ )
2360
+
2361
+ // Clean up temp path
2362
+ if (cached.path !== pluginPath) {
2363
+ await rm(cached.path, { recursive: true, force: true })
2364
+ }
2365
+ }
2366
+ }
2367
+ } catch (error) {
2368
+ const errorMsg = errorMessage(error)
2369
+ logForDebugging(`Failed to cache plugin ${entry.name}: ${errorMsg}`, {
2370
+ level: 'error',
2371
+ })
2372
+ logError(toError(error))
2373
+ errorsOut.push({
2374
+ type: 'generic-error',
2375
+ source: pluginId,
2376
+ error: `Failed to download/cache plugin ${entry.name}: ${errorMsg}`,
2377
+ })
2378
+ return null
2379
+ }
2380
+ }
2381
+
2382
+ // Zip cache mode: extract ZIP to session temp dir before loading
2383
+ if (isPluginZipCacheEnabled() && pluginPath.endsWith('.zip')) {
2384
+ const sessionDir = await getSessionPluginCachePath()
2385
+ const extractDir = join(
2386
+ sessionDir,
2387
+ pluginId.replace(/[^a-zA-Z0-9@\-_]/g, '-'),
2388
+ )
2389
+ try {
2390
+ await extractZipToDirectory(pluginPath, extractDir)
2391
+ logForDebugging(`Extracted plugin ZIP to session dir: ${extractDir}`)
2392
+ pluginPath = extractDir
2393
+ } catch (error) {
2394
+ // Corrupt ZIP: delete it so next install attempt re-creates it
2395
+ logForDebugging(
2396
+ `Failed to extract plugin ZIP ${pluginPath}, deleting corrupt file: ${error}`,
2397
+ )
2398
+ await rm(pluginPath, { force: true }).catch(() => {})
2399
+ throw error
2400
+ }
2401
+ }
2402
+
2403
+ return finishLoadingPluginFromPath(
2404
+ entry,
2405
+ pluginId,
2406
+ enabled,
2407
+ errorsOut,
2408
+ pluginPath,
2409
+ )
2410
+ }
2411
+
2412
+ /**
2413
+ * Shared tail of both loadPluginFromMarketplaceEntry variants.
2414
+ *
2415
+ * Once pluginPath is resolved (via clone, cache, or installPath lookup),
2416
+ * the rest of the load — manifest probe, createPluginFromPath, marketplace
2417
+ * entry supplementation — is identical. Extracted so the cache-only path
2418
+ * doesn't duplicate ~500 lines.
2419
+ */
2420
+ async function finishLoadingPluginFromPath(
2421
+ entry: PluginMarketplaceEntry,
2422
+ pluginId: string,
2423
+ enabled: boolean,
2424
+ errorsOut: PluginError[],
2425
+ pluginPath: string,
2426
+ ): Promise<LoadedPlugin | null> {
2427
+ const errors: PluginError[] = []
2428
+
2429
+ // Check if plugin.json exists to determine if we should use marketplace manifest
2430
+ const manifestPath = join(pluginPath, '.claude-plugin', 'plugin.json')
2431
+ const hasManifest = await pathExists(manifestPath)
2432
+
2433
+ const { plugin, errors: pluginErrors } = await createPluginFromPath(
2434
+ pluginPath,
2435
+ pluginId,
2436
+ enabled,
2437
+ entry.name,
2438
+ entry.strict ?? true, // Respect marketplace entry's strict setting
2439
+ )
2440
+ errors.push(...pluginErrors)
2441
+
2442
+ // Set sha from source if available (for github and url source types)
2443
+ if (
2444
+ typeof entry.source === 'object' &&
2445
+ 'sha' in entry.source &&
2446
+ entry.source.sha
2447
+ ) {
2448
+ plugin.sha = entry.source.sha
2449
+ }
2450
+
2451
+ // If there's no plugin.json, use marketplace entry as manifest (regardless of strict mode)
2452
+ if (!hasManifest) {
2453
+ plugin.manifest = {
2454
+ ...entry,
2455
+ id: undefined,
2456
+ source: undefined,
2457
+ strict: undefined,
2458
+ } as PluginManifest
2459
+ plugin.name = plugin.manifest.name
2460
+
2461
+ // Process commands from marketplace entry
2462
+ if (entry.commands) {
2463
+ // Check if it's an object mapping
2464
+ const firstValue = Object.values(entry.commands)[0]
2465
+ if (
2466
+ typeof entry.commands === 'object' &&
2467
+ !Array.isArray(entry.commands) &&
2468
+ firstValue &&
2469
+ typeof firstValue === 'object' &&
2470
+ ('source' in firstValue || 'content' in firstValue)
2471
+ ) {
2472
+ // Object mapping format
2473
+ const commandsMetadata: Record<string, CommandMetadata> = {}
2474
+ const validPaths: string[] = []
2475
+
2476
+ // Parallelize pathExists checks; process results in order.
2477
+ const entries = Object.entries(entry.commands)
2478
+ const checks = await Promise.all(
2479
+ entries.map(async ([commandName, metadata]) => {
2480
+ if (!metadata || typeof metadata !== 'object' || !metadata.source) {
2481
+ return { commandName, metadata, skip: true as const }
2482
+ }
2483
+ const fullPath = join(pluginPath, metadata.source)
2484
+ return {
2485
+ commandName,
2486
+ metadata,
2487
+ skip: false as const,
2488
+ fullPath,
2489
+ exists: await pathExists(fullPath),
2490
+ }
2491
+ }),
2492
+ )
2493
+ for (const check of checks) {
2494
+ if (check.skip) continue
2495
+ if (check.exists) {
2496
+ validPaths.push(check.fullPath)
2497
+ commandsMetadata[check.commandName] = check.metadata
2498
+ } else {
2499
+ logForDebugging(
2500
+ `Command ${check.commandName} path ${check.metadata.source} from marketplace entry not found at ${check.fullPath} for ${entry.name}`,
2501
+ { level: 'warn' },
2502
+ )
2503
+ logError(
2504
+ new Error(
2505
+ `Plugin component file not found: ${check.fullPath} for ${entry.name}`,
2506
+ ),
2507
+ )
2508
+ errors.push({
2509
+ type: 'path-not-found',
2510
+ source: pluginId,
2511
+ plugin: entry.name,
2512
+ path: check.fullPath,
2513
+ component: 'commands',
2514
+ })
2515
+ }
2516
+ }
2517
+
2518
+ if (validPaths.length > 0) {
2519
+ plugin.commandsPaths = validPaths
2520
+ plugin.commandsMetadata = commandsMetadata
2521
+ }
2522
+ } else {
2523
+ // Path or array of paths format
2524
+ const commandPaths = Array.isArray(entry.commands)
2525
+ ? entry.commands
2526
+ : [entry.commands]
2527
+
2528
+ // Parallelize pathExists checks; process results in order.
2529
+ const checks = await Promise.all(
2530
+ commandPaths.map(async cmdPath => {
2531
+ if (typeof cmdPath !== 'string') {
2532
+ return { cmdPath, kind: 'invalid' as const }
2533
+ }
2534
+ const fullPath = join(pluginPath, cmdPath)
2535
+ return {
2536
+ cmdPath,
2537
+ kind: 'path' as const,
2538
+ fullPath,
2539
+ exists: await pathExists(fullPath),
2540
+ }
2541
+ }),
2542
+ )
2543
+ const validPaths: string[] = []
2544
+ for (const check of checks) {
2545
+ if (check.kind === 'invalid') {
2546
+ logForDebugging(
2547
+ `Unexpected command format in marketplace entry for ${entry.name}`,
2548
+ { level: 'error' },
2549
+ )
2550
+ continue
2551
+ }
2552
+ if (check.exists) {
2553
+ validPaths.push(check.fullPath)
2554
+ } else {
2555
+ logForDebugging(
2556
+ `Command path ${check.cmdPath} from marketplace entry not found at ${check.fullPath} for ${entry.name}`,
2557
+ { level: 'warn' },
2558
+ )
2559
+ logError(
2560
+ new Error(
2561
+ `Plugin component file not found: ${check.fullPath} for ${entry.name}`,
2562
+ ),
2563
+ )
2564
+ errors.push({
2565
+ type: 'path-not-found',
2566
+ source: pluginId,
2567
+ plugin: entry.name,
2568
+ path: check.fullPath,
2569
+ component: 'commands',
2570
+ })
2571
+ }
2572
+ }
2573
+
2574
+ if (validPaths.length > 0) {
2575
+ plugin.commandsPaths = validPaths
2576
+ }
2577
+ }
2578
+ }
2579
+
2580
+ // Process agents from marketplace entry
2581
+ if (entry.agents) {
2582
+ const agentPaths = Array.isArray(entry.agents)
2583
+ ? entry.agents
2584
+ : [entry.agents]
2585
+
2586
+ const validPaths = await validatePluginPaths(
2587
+ agentPaths,
2588
+ pluginPath,
2589
+ entry.name,
2590
+ pluginId,
2591
+ 'agents',
2592
+ 'Agent',
2593
+ 'from marketplace entry',
2594
+ errors,
2595
+ )
2596
+
2597
+ if (validPaths.length > 0) {
2598
+ plugin.agentsPaths = validPaths
2599
+ }
2600
+ }
2601
+
2602
+ // Process skills from marketplace entry
2603
+ if (entry.skills) {
2604
+ logForDebugging(
2605
+ `Processing ${Array.isArray(entry.skills) ? entry.skills.length : 1} skill paths for plugin ${entry.name}`,
2606
+ )
2607
+ const skillPaths = Array.isArray(entry.skills)
2608
+ ? entry.skills
2609
+ : [entry.skills]
2610
+
2611
+ // Parallelize pathExists checks; process results in order.
2612
+ // Note: previously this loop called pathExists() TWICE per iteration
2613
+ // (once in a debug log template, once in the if) — now called once.
2614
+ const checks = await Promise.all(
2615
+ skillPaths.map(async skillPath => {
2616
+ const fullPath = join(pluginPath, skillPath)
2617
+ return { skillPath, fullPath, exists: await pathExists(fullPath) }
2618
+ }),
2619
+ )
2620
+ const validPaths: string[] = []
2621
+ for (const { skillPath, fullPath, exists } of checks) {
2622
+ logForDebugging(
2623
+ `Checking skill path: ${skillPath} -> ${fullPath} (exists: ${exists})`,
2624
+ )
2625
+ if (exists) {
2626
+ validPaths.push(fullPath)
2627
+ } else {
2628
+ logForDebugging(
2629
+ `Skill path ${skillPath} from marketplace entry not found at ${fullPath} for ${entry.name}`,
2630
+ { level: 'warn' },
2631
+ )
2632
+ logError(
2633
+ new Error(
2634
+ `Plugin component file not found: ${fullPath} for ${entry.name}`,
2635
+ ),
2636
+ )
2637
+ errors.push({
2638
+ type: 'path-not-found',
2639
+ source: pluginId,
2640
+ plugin: entry.name,
2641
+ path: fullPath,
2642
+ component: 'skills',
2643
+ })
2644
+ }
2645
+ }
2646
+
2647
+ logForDebugging(
2648
+ `Found ${validPaths.length} valid skill paths for plugin ${entry.name}, setting skillsPaths`,
2649
+ )
2650
+ if (validPaths.length > 0) {
2651
+ plugin.skillsPaths = validPaths
2652
+ }
2653
+ } else {
2654
+ logForDebugging(`Plugin ${entry.name} has no entry.skills defined`)
2655
+ }
2656
+
2657
+ // Process output styles from marketplace entry
2658
+ if (entry.outputStyles) {
2659
+ const outputStylePaths = Array.isArray(entry.outputStyles)
2660
+ ? entry.outputStyles
2661
+ : [entry.outputStyles]
2662
+
2663
+ const validPaths = await validatePluginPaths(
2664
+ outputStylePaths,
2665
+ pluginPath,
2666
+ entry.name,
2667
+ pluginId,
2668
+ 'output-styles',
2669
+ 'Output style',
2670
+ 'from marketplace entry',
2671
+ errors,
2672
+ )
2673
+
2674
+ if (validPaths.length > 0) {
2675
+ plugin.outputStylesPaths = validPaths
2676
+ }
2677
+ }
2678
+
2679
+ // Process inline hooks from marketplace entry
2680
+ if (entry.hooks) {
2681
+ plugin.hooksConfig = entry.hooks as HooksSettings
2682
+ }
2683
+ } else if (
2684
+ !entry.strict &&
2685
+ hasManifest &&
2686
+ (entry.commands ||
2687
+ entry.agents ||
2688
+ entry.skills ||
2689
+ entry.hooks ||
2690
+ entry.outputStyles)
2691
+ ) {
2692
+ // In non-strict mode with plugin.json, marketplace entries for commands/agents/skills/hooks/outputStyles are conflicts
2693
+ const error = new Error(
2694
+ `Plugin ${entry.name} has both plugin.json and marketplace manifest entries for commands/agents/skills/hooks/outputStyles. This is a conflict.`,
2695
+ )
2696
+ logForDebugging(
2697
+ `Plugin ${entry.name} has both plugin.json and marketplace manifest entries for commands/agents/skills/hooks/outputStyles. This is a conflict.`,
2698
+ { level: 'error' },
2699
+ )
2700
+ logError(error)
2701
+ errorsOut.push({
2702
+ type: 'generic-error',
2703
+ source: pluginId,
2704
+ error: `Plugin ${entry.name} has conflicting manifests: both plugin.json and marketplace entry specify components. Set strict: true in marketplace entry or remove component specs from one location.`,
2705
+ })
2706
+ return null
2707
+ } else if (hasManifest) {
2708
+ // Has plugin.json - marketplace can supplement commands/agents/skills/hooks/outputStyles
2709
+
2710
+ // Supplement commands from marketplace entry
2711
+ if (entry.commands) {
2712
+ // Check if it's an object mapping
2713
+ const firstValue = Object.values(entry.commands)[0]
2714
+ if (
2715
+ typeof entry.commands === 'object' &&
2716
+ !Array.isArray(entry.commands) &&
2717
+ firstValue &&
2718
+ typeof firstValue === 'object' &&
2719
+ ('source' in firstValue || 'content' in firstValue)
2720
+ ) {
2721
+ // Object mapping format - merge metadata
2722
+ const commandsMetadata: Record<string, CommandMetadata> = {
2723
+ ...(plugin.commandsMetadata || {}),
2724
+ }
2725
+ const validPaths: string[] = []
2726
+
2727
+ // Parallelize pathExists checks; process results in order.
2728
+ const entries = Object.entries(entry.commands)
2729
+ const checks = await Promise.all(
2730
+ entries.map(async ([commandName, metadata]) => {
2731
+ if (!metadata || typeof metadata !== 'object' || !metadata.source) {
2732
+ return { commandName, metadata, skip: true as const }
2733
+ }
2734
+ const fullPath = join(pluginPath, metadata.source)
2735
+ return {
2736
+ commandName,
2737
+ metadata,
2738
+ skip: false as const,
2739
+ fullPath,
2740
+ exists: await pathExists(fullPath),
2741
+ }
2742
+ }),
2743
+ )
2744
+ for (const check of checks) {
2745
+ if (check.skip) continue
2746
+ if (check.exists) {
2747
+ validPaths.push(check.fullPath)
2748
+ commandsMetadata[check.commandName] = check.metadata
2749
+ } else {
2750
+ logForDebugging(
2751
+ `Command ${check.commandName} path ${check.metadata.source} from marketplace entry not found at ${check.fullPath} for ${entry.name}`,
2752
+ { level: 'warn' },
2753
+ )
2754
+ logError(
2755
+ new Error(
2756
+ `Plugin component file not found: ${check.fullPath} for ${entry.name}`,
2757
+ ),
2758
+ )
2759
+ errors.push({
2760
+ type: 'path-not-found',
2761
+ source: pluginId,
2762
+ plugin: entry.name,
2763
+ path: check.fullPath,
2764
+ component: 'commands',
2765
+ })
2766
+ }
2767
+ }
2768
+
2769
+ if (validPaths.length > 0) {
2770
+ plugin.commandsPaths = [
2771
+ ...(plugin.commandsPaths || []),
2772
+ ...validPaths,
2773
+ ]
2774
+ plugin.commandsMetadata = commandsMetadata
2775
+ }
2776
+ } else {
2777
+ // Path or array of paths format
2778
+ const commandPaths = Array.isArray(entry.commands)
2779
+ ? entry.commands
2780
+ : [entry.commands]
2781
+
2782
+ // Parallelize pathExists checks; process results in order.
2783
+ const checks = await Promise.all(
2784
+ commandPaths.map(async cmdPath => {
2785
+ if (typeof cmdPath !== 'string') {
2786
+ return { cmdPath, kind: 'invalid' as const }
2787
+ }
2788
+ const fullPath = join(pluginPath, cmdPath)
2789
+ return {
2790
+ cmdPath,
2791
+ kind: 'path' as const,
2792
+ fullPath,
2793
+ exists: await pathExists(fullPath),
2794
+ }
2795
+ }),
2796
+ )
2797
+ const validPaths: string[] = []
2798
+ for (const check of checks) {
2799
+ if (check.kind === 'invalid') {
2800
+ logForDebugging(
2801
+ `Unexpected command format in marketplace entry for ${entry.name}`,
2802
+ { level: 'error' },
2803
+ )
2804
+ continue
2805
+ }
2806
+ if (check.exists) {
2807
+ validPaths.push(check.fullPath)
2808
+ } else {
2809
+ logForDebugging(
2810
+ `Command path ${check.cmdPath} from marketplace entry not found at ${check.fullPath} for ${entry.name}`,
2811
+ { level: 'warn' },
2812
+ )
2813
+ logError(
2814
+ new Error(
2815
+ `Plugin component file not found: ${check.fullPath} for ${entry.name}`,
2816
+ ),
2817
+ )
2818
+ errors.push({
2819
+ type: 'path-not-found',
2820
+ source: pluginId,
2821
+ plugin: entry.name,
2822
+ path: check.fullPath,
2823
+ component: 'commands',
2824
+ })
2825
+ }
2826
+ }
2827
+
2828
+ if (validPaths.length > 0) {
2829
+ plugin.commandsPaths = [
2830
+ ...(plugin.commandsPaths || []),
2831
+ ...validPaths,
2832
+ ]
2833
+ }
2834
+ }
2835
+ }
2836
+
2837
+ // Supplement agents from marketplace entry
2838
+ if (entry.agents) {
2839
+ const agentPaths = Array.isArray(entry.agents)
2840
+ ? entry.agents
2841
+ : [entry.agents]
2842
+
2843
+ const validPaths = await validatePluginPaths(
2844
+ agentPaths,
2845
+ pluginPath,
2846
+ entry.name,
2847
+ pluginId,
2848
+ 'agents',
2849
+ 'Agent',
2850
+ 'from marketplace entry',
2851
+ errors,
2852
+ )
2853
+
2854
+ if (validPaths.length > 0) {
2855
+ plugin.agentsPaths = [...(plugin.agentsPaths || []), ...validPaths]
2856
+ }
2857
+ }
2858
+
2859
+ // Supplement skills from marketplace entry
2860
+ if (entry.skills) {
2861
+ const skillPaths = Array.isArray(entry.skills)
2862
+ ? entry.skills
2863
+ : [entry.skills]
2864
+
2865
+ const validPaths = await validatePluginPaths(
2866
+ skillPaths,
2867
+ pluginPath,
2868
+ entry.name,
2869
+ pluginId,
2870
+ 'skills',
2871
+ 'Skill',
2872
+ 'from marketplace entry',
2873
+ errors,
2874
+ )
2875
+
2876
+ if (validPaths.length > 0) {
2877
+ plugin.skillsPaths = [...(plugin.skillsPaths || []), ...validPaths]
2878
+ }
2879
+ }
2880
+
2881
+ // Supplement output styles from marketplace entry
2882
+ if (entry.outputStyles) {
2883
+ const outputStylePaths = Array.isArray(entry.outputStyles)
2884
+ ? entry.outputStyles
2885
+ : [entry.outputStyles]
2886
+
2887
+ const validPaths = await validatePluginPaths(
2888
+ outputStylePaths,
2889
+ pluginPath,
2890
+ entry.name,
2891
+ pluginId,
2892
+ 'output-styles',
2893
+ 'Output style',
2894
+ 'from marketplace entry',
2895
+ errors,
2896
+ )
2897
+
2898
+ if (validPaths.length > 0) {
2899
+ plugin.outputStylesPaths = [
2900
+ ...(plugin.outputStylesPaths || []),
2901
+ ...validPaths,
2902
+ ]
2903
+ }
2904
+ }
2905
+
2906
+ // Supplement hooks from marketplace entry
2907
+ if (entry.hooks) {
2908
+ plugin.hooksConfig = {
2909
+ ...(plugin.hooksConfig || {}),
2910
+ ...(entry.hooks as HooksSettings),
2911
+ }
2912
+ }
2913
+ }
2914
+
2915
+ errorsOut.push(...errors)
2916
+ return plugin
2917
+ }
2918
+
2919
+ /**
2920
+ * Load session-only plugins from --plugin-dir CLI flag.
2921
+ *
2922
+ * These plugins are loaded directly without going through the marketplace system.
2923
+ * They appear with source='plugin-name@inline' and are always enabled for the current session.
2924
+ *
2925
+ * @param sessionPluginPaths - Array of plugin directory paths from CLI
2926
+ * @returns LoadedPlugin objects and any errors encountered
2927
+ */
2928
+ async function loadSessionOnlyPlugins(
2929
+ sessionPluginPaths: Array<string>,
2930
+ ): Promise<{ plugins: LoadedPlugin[]; errors: PluginError[] }> {
2931
+ if (sessionPluginPaths.length === 0) {
2932
+ return { plugins: [], errors: [] }
2933
+ }
2934
+
2935
+ const plugins: LoadedPlugin[] = []
2936
+ const errors: PluginError[] = []
2937
+
2938
+ for (const [index, pluginPath] of sessionPluginPaths.entries()) {
2939
+ try {
2940
+ const resolvedPath = resolve(pluginPath)
2941
+
2942
+ if (!(await pathExists(resolvedPath))) {
2943
+ logForDebugging(
2944
+ `Plugin path does not exist: ${resolvedPath}, skipping`,
2945
+ { level: 'warn' },
2946
+ )
2947
+ errors.push({
2948
+ type: 'path-not-found',
2949
+ source: `inline[${index}]`,
2950
+ path: resolvedPath,
2951
+ component: 'commands',
2952
+ })
2953
+ continue
2954
+ }
2955
+
2956
+ const dirName = basename(resolvedPath)
2957
+ const { plugin, errors: pluginErrors } = await createPluginFromPath(
2958
+ resolvedPath,
2959
+ `${dirName}@inline`, // temporary, will be updated after we know the real name
2960
+ true, // always enabled
2961
+ dirName,
2962
+ )
2963
+
2964
+ // Update source to use the actual plugin name from manifest
2965
+ plugin.source = `${plugin.name}@inline`
2966
+ plugin.repository = `${plugin.name}@inline`
2967
+
2968
+ plugins.push(plugin)
2969
+ errors.push(...pluginErrors)
2970
+
2971
+ logForDebugging(`Loaded inline plugin from path: ${plugin.name}`)
2972
+ } catch (error) {
2973
+ const errorMsg = errorMessage(error)
2974
+ logForDebugging(
2975
+ `Failed to load session plugin from ${pluginPath}: ${errorMsg}`,
2976
+ { level: 'warn' },
2977
+ )
2978
+ errors.push({
2979
+ type: 'generic-error',
2980
+ source: `inline[${index}]`,
2981
+ error: `Failed to load plugin: ${errorMsg}`,
2982
+ })
2983
+ }
2984
+ }
2985
+
2986
+ if (plugins.length > 0) {
2987
+ logForDebugging(
2988
+ `Loaded ${plugins.length} session-only plugins from --plugin-dir`,
2989
+ )
2990
+ }
2991
+
2992
+ return { plugins, errors }
2993
+ }
2994
+
2995
+ /**
2996
+ * Merge plugins from session (--plugin-dir), marketplace (installed), and
2997
+ * builtin sources. Session plugins override marketplace plugins with the
2998
+ * same name — the user explicitly pointed at a directory for this session.
2999
+ *
3000
+ * Exception: marketplace plugins locked by managed settings (policySettings)
3001
+ * cannot be overridden. Enterprise admin intent beats local dev convenience.
3002
+ * When a session plugin collides with a managed one, the session copy is
3003
+ * dropped and an error is returned for surfacing.
3004
+ *
3005
+ * Without this dedup, both versions sat in the array and marketplace won
3006
+ * on first-match, making --plugin-dir useless for iterating on an
3007
+ * installed plugin.
3008
+ */
3009
+ export function mergePluginSources(sources: {
3010
+ session: LoadedPlugin[]
3011
+ marketplace: LoadedPlugin[]
3012
+ builtin: LoadedPlugin[]
3013
+ managedNames?: Set<string> | null
3014
+ }): { plugins: LoadedPlugin[]; errors: PluginError[] } {
3015
+ const errors: PluginError[] = []
3016
+ const managed = sources.managedNames
3017
+
3018
+ // Managed settings win over --plugin-dir. Drop session plugins whose
3019
+ // name appears in policySettings.enabledPlugins (whether force-enabled
3020
+ // OR force-disabled — both are admin intent that --plugin-dir must not
3021
+ // bypass). Surface an error so the user knows why their dev copy was
3022
+ // ignored.
3023
+ //
3024
+ // NOTE: managedNames contains the pluginId prefix (entry.name), which is
3025
+ // expected to equal manifest.name by convention (schema description at
3026
+ // schemas.ts PluginMarketplaceEntry.name). If a marketplace publishes a
3027
+ // plugin where entry.name ≠ manifest.name, this guard will silently miss —
3028
+ // but that's a marketplace misconfiguration that breaks other things too
3029
+ // (e.g., ManagePlugins constructs pluginIds from manifest.name).
3030
+ const sessionPlugins = sources.session.filter(p => {
3031
+ if (managed?.has(p.name)) {
3032
+ logForDebugging(
3033
+ `Plugin "${p.name}" from --plugin-dir is blocked by managed settings`,
3034
+ { level: 'warn' },
3035
+ )
3036
+ errors.push({
3037
+ type: 'generic-error',
3038
+ source: p.source,
3039
+ plugin: p.name,
3040
+ error: `--plugin-dir copy of "${p.name}" ignored: plugin is locked by managed settings`,
3041
+ })
3042
+ return false
3043
+ }
3044
+ return true
3045
+ })
3046
+
3047
+ const sessionNames = new Set(sessionPlugins.map(p => p.name))
3048
+ const marketplacePlugins = sources.marketplace.filter(p => {
3049
+ if (sessionNames.has(p.name)) {
3050
+ logForDebugging(
3051
+ `Plugin "${p.name}" from --plugin-dir overrides installed version`,
3052
+ )
3053
+ return false
3054
+ }
3055
+ return true
3056
+ })
3057
+ // Session first, then non-overridden marketplace, then builtin.
3058
+ // Downstream first-match consumers see session plugins before
3059
+ // installed ones for any that slipped past the name filter.
3060
+ return {
3061
+ plugins: [...sessionPlugins, ...marketplacePlugins, ...sources.builtin],
3062
+ errors,
3063
+ }
3064
+ }
3065
+
3066
+ /**
3067
+ * Main plugin loading function that discovers and loads all plugins.
3068
+ *
3069
+ * This function is memoized to avoid repeated filesystem scanning and is
3070
+ * the primary entry point for the plugin system. It discovers plugins from
3071
+ * multiple sources and returns categorized results.
3072
+ *
3073
+ * Loading order and precedence (see mergePluginSources):
3074
+ * 1. Session-only plugins (from --plugin-dir CLI flag) — override
3075
+ * installed plugins with the same name, UNLESS that plugin is
3076
+ * locked by managed settings (policySettings, either force-enabled
3077
+ * or force-disabled)
3078
+ * 2. Marketplace-based plugins (plugin@marketplace format from settings)
3079
+ * 3. Built-in plugins shipped with the CLI
3080
+ *
3081
+ * Name collision: session plugin wins over installed. The user explicitly
3082
+ * pointed at a directory for this session — that intent beats whatever
3083
+ * is installed. Exception: managed settings (enterprise policy) win over
3084
+ * --plugin-dir. Admin intent beats local dev convenience.
3085
+ *
3086
+ * Error collection:
3087
+ * - Non-fatal errors are collected and returned
3088
+ * - System continues loading other plugins on errors
3089
+ * - Errors include source information for debugging
3090
+ *
3091
+ * @returns Promise resolving to categorized plugin results:
3092
+ * - enabled: Array of enabled LoadedPlugin objects
3093
+ * - disabled: Array of disabled LoadedPlugin objects
3094
+ * - errors: Array of loading errors with source information
3095
+ */
3096
+ export const loadAllPlugins = memoize(async (): Promise<PluginLoadResult> => {
3097
+ const result = await assemblePluginLoadResult(() =>
3098
+ loadPluginsFromMarketplaces({ cacheOnly: false }),
3099
+ )
3100
+ // A fresh full-load result is strictly valid for cache-only callers
3101
+ // (both variants share assemblePluginLoadResult). Warm the separate
3102
+ // memoize so refreshActivePlugins()'s downstream getPluginCommands() /
3103
+ // getAgentDefinitionsWithOverrides() — which now call
3104
+ // loadAllPluginsCacheOnly — see just-cloned plugins instead of reading
3105
+ // an installed_plugins.json that nothing writes mid-session.
3106
+ loadAllPluginsCacheOnly.cache?.set(undefined, Promise.resolve(result))
3107
+ return result
3108
+ })
3109
+
3110
+ /**
3111
+ * Cache-only variant of loadAllPlugins.
3112
+ *
3113
+ * Same merge/dependency/settings logic, but the marketplace loader never
3114
+ * hits the network (no cachePlugin, no copyPluginToVersionedCache). Reads
3115
+ * from installed_plugins.json's installPath. Plugins not on disk emit
3116
+ * 'plugin-cache-miss' and are skipped.
3117
+ *
3118
+ * Use this in startup consumers (getCommands, loadPluginAgents, MCP/LSP
3119
+ * config) so interactive startup never blocks on git clones for ref-tracked
3120
+ * plugins. Use loadAllPlugins() in explicit refresh paths (/plugins,
3121
+ * refresh.ts, headlessPluginInstall) where fresh source is the intent.
3122
+ *
3123
+ * CLAUDE_CODE_SYNC_PLUGIN_INSTALL=1 delegates to the full loader — that
3124
+ * mode explicitly opts into blocking install before first query, and
3125
+ * main.tsx's getClaudeCodeMcpConfigs()/getInitialSettings().agent run
3126
+ * BEFORE runHeadless() can warm this cache. First-run CCR/headless has
3127
+ * no installed_plugins.json, so cache-only would miss plugin MCP servers
3128
+ * and plugin settings (the agent key). The interactive startup win is
3129
+ * preserved since interactive mode doesn't set SYNC_PLUGIN_INSTALL.
3130
+ *
3131
+ * Separate memoize cache from loadAllPlugins — a cache-only result must
3132
+ * never satisfy a caller that wants fresh source. The reverse IS valid:
3133
+ * loadAllPlugins warms this cache on completion so refresh paths that run
3134
+ * the full loader don't get plugin-cache-miss from their downstream
3135
+ * cache-only consumers.
3136
+ */
3137
+ export const loadAllPluginsCacheOnly = memoize(
3138
+ async (): Promise<PluginLoadResult> => {
3139
+ if (isEnvTruthy(process.env.CLAUDE_CODE_SYNC_PLUGIN_INSTALL)) {
3140
+ return loadAllPlugins()
3141
+ }
3142
+ return assemblePluginLoadResult(() =>
3143
+ loadPluginsFromMarketplaces({ cacheOnly: true }),
3144
+ )
3145
+ },
3146
+ )
3147
+
3148
+ /**
3149
+ * Shared body of loadAllPlugins and loadAllPluginsCacheOnly.
3150
+ *
3151
+ * The only difference between the two is which marketplace loader runs —
3152
+ * session plugins, builtins, merge, verifyAndDemote, and cachePluginSettings
3153
+ * are identical (invariants 1-3).
3154
+ */
3155
+ async function assemblePluginLoadResult(
3156
+ marketplaceLoader: () => Promise<{
3157
+ plugins: LoadedPlugin[]
3158
+ errors: PluginError[]
3159
+ }>,
3160
+ ): Promise<PluginLoadResult> {
3161
+ // Load marketplace plugins and session-only plugins in parallel.
3162
+ // getInlinePlugins() is a synchronous state read with no dependency on
3163
+ // marketplace loading, so these two sources can be fetched concurrently.
3164
+ const inlinePlugins = getInlinePlugins()
3165
+ const [marketplaceResult, sessionResult] = await Promise.all([
3166
+ marketplaceLoader(),
3167
+ inlinePlugins.length > 0
3168
+ ? loadSessionOnlyPlugins(inlinePlugins)
3169
+ : Promise.resolve({ plugins: [], errors: [] }),
3170
+ ])
3171
+ // 3. Load built-in plugins that ship with the CLI
3172
+ const builtinResult = getBuiltinPlugins()
3173
+
3174
+ // Session plugins (--plugin-dir) override installed ones by name,
3175
+ // UNLESS the installed plugin is locked by managed settings
3176
+ // (policySettings). See mergePluginSources() for details.
3177
+ const { plugins: allPlugins, errors: mergeErrors } = mergePluginSources({
3178
+ session: sessionResult.plugins,
3179
+ marketplace: marketplaceResult.plugins,
3180
+ builtin: [...builtinResult.enabled, ...builtinResult.disabled],
3181
+ managedNames: getManagedPluginNames(),
3182
+ })
3183
+ const allErrors = [
3184
+ ...marketplaceResult.errors,
3185
+ ...sessionResult.errors,
3186
+ ...mergeErrors,
3187
+ ]
3188
+
3189
+ // Verify dependencies. Runs AFTER the parallel load — deps are presence
3190
+ // checks, not load-order, so no topological sort needed. Demotion is
3191
+ // session-local: does NOT write settings (user fixes intent via /doctor).
3192
+ const { demoted, errors: depErrors } = verifyAndDemote(allPlugins)
3193
+ for (const p of allPlugins) {
3194
+ if (demoted.has(p.source)) p.enabled = false
3195
+ }
3196
+ allErrors.push(...depErrors)
3197
+
3198
+ const enabledPlugins = allPlugins.filter(p => p.enabled)
3199
+ logForDebugging(
3200
+ `Found ${allPlugins.length} plugins (${enabledPlugins.length} enabled, ${allPlugins.length - enabledPlugins.length} disabled)`,
3201
+ )
3202
+
3203
+ // 3. Cache plugin settings for synchronous access by the settings cascade
3204
+ cachePluginSettings(enabledPlugins)
3205
+
3206
+ return {
3207
+ enabled: enabledPlugins,
3208
+ disabled: allPlugins.filter(p => !p.enabled),
3209
+ errors: allErrors,
3210
+ }
3211
+ }
3212
+
3213
+ /**
3214
+ * Clears the memoized plugin cache.
3215
+ *
3216
+ * Call this when plugins are installed, removed, or settings change
3217
+ * to force a fresh scan on the next loadAllPlugins call.
3218
+ *
3219
+ * Use cases:
3220
+ * - After installing/uninstalling plugins
3221
+ * - After modifying .claude-plugin/ directory (for export)
3222
+ * - After changing enabledPlugins settings
3223
+ * - When debugging plugin loading issues
3224
+ */
3225
+ export function clearPluginCache(reason?: string): void {
3226
+ if (reason) {
3227
+ logForDebugging(
3228
+ `clearPluginCache: invalidating loadAllPlugins cache (${reason})`,
3229
+ )
3230
+ }
3231
+ loadAllPlugins.cache?.clear?.()
3232
+ loadAllPluginsCacheOnly.cache?.clear?.()
3233
+ // If a plugin previously contributed settings, the session settings cache
3234
+ // holds a merged result that includes them. cachePluginSettings() on reload
3235
+ // won't bust the cache when the new base is empty (the startup perf win),
3236
+ // so bust it here to drop stale plugin overrides. When the base is already
3237
+ // undefined (startup, or no prior plugin settings) this is a no-op.
3238
+ if (getPluginSettingsBase() !== undefined) {
3239
+ resetSettingsCache()
3240
+ }
3241
+ clearPluginSettingsBase()
3242
+ // TODO: Clear installed plugins cache when installedPluginsManager is implemented
3243
+ }
3244
+
3245
+ /**
3246
+ * Merge settings from all enabled plugins into a single record.
3247
+ * Later plugins override earlier ones for the same key.
3248
+ * Only allowlisted keys are included (filtering happens at load time).
3249
+ */
3250
+ function mergePluginSettings(
3251
+ plugins: LoadedPlugin[],
3252
+ ): Record<string, unknown> | undefined {
3253
+ let merged: Record<string, unknown> | undefined
3254
+
3255
+ for (const plugin of plugins) {
3256
+ if (!plugin.settings) {
3257
+ continue
3258
+ }
3259
+
3260
+ if (!merged) {
3261
+ merged = {}
3262
+ }
3263
+
3264
+ for (const [key, value] of Object.entries(plugin.settings)) {
3265
+ if (key in merged) {
3266
+ logForDebugging(
3267
+ `Plugin "${plugin.name}" overrides setting "${key}" (previously set by another plugin)`,
3268
+ )
3269
+ }
3270
+ merged[key] = value
3271
+ }
3272
+ }
3273
+
3274
+ return merged
3275
+ }
3276
+
3277
+ /**
3278
+ * Store merged plugin settings in the synchronous cache.
3279
+ * Called after loadAllPlugins resolves.
3280
+ */
3281
+ export function cachePluginSettings(plugins: LoadedPlugin[]): void {
3282
+ const settings = mergePluginSettings(plugins)
3283
+ setPluginSettingsBase(settings)
3284
+ // Only bust the session settings cache if there are actually plugin settings
3285
+ // to merge. In the common case (no plugins, or plugins without settings) the
3286
+ // base layer is empty and loadSettingsFromDisk would produce the same result
3287
+ // anyway — resetting here would waste ~17ms on startup re-reading and
3288
+ // re-validating every settings file on the next getSettingsWithErrors() call.
3289
+ if (settings && Object.keys(settings).length > 0) {
3290
+ resetSettingsCache()
3291
+ logForDebugging(
3292
+ `Cached plugin settings with keys: ${Object.keys(settings).join(', ')}`,
3293
+ )
3294
+ }
3295
+ }
3296
+
3297
+ /**
3298
+ * Type predicate: check if a value is a non-null, non-array object (i.e., a record).
3299
+ */
3300
+ function isRecord(value: unknown): value is Record<string, unknown> {
3301
+ return typeof value === 'object' && value !== null && !Array.isArray(value)
3302
+ }