@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,3997 @@
1
+ // biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered
2
+ import {
3
+ logEvent,
4
+ type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
5
+ } from 'src/services/analytics/index.js'
6
+ import {
7
+ toolMatchesName,
8
+ type Tools,
9
+ type ToolUseContext,
10
+ type ToolPermissionContext,
11
+ } from '../Tool.js'
12
+ import {
13
+ FileReadTool,
14
+ MaxFileReadTokenExceededError,
15
+ type Output as FileReadToolOutput,
16
+ readImageWithTokenBudget,
17
+ } from '../tools/FileReadTool/FileReadTool.js'
18
+ import { FileTooLargeError, readFileInRange } from './readFileInRange.js'
19
+ import { expandPath } from './path.js'
20
+ import { countCharInString } from './stringUtils.js'
21
+ import { count, uniq } from './array.js'
22
+ import { getFsImplementation } from './fsOperations.js'
23
+ import { readdir, stat } from 'fs/promises'
24
+ import type { IDESelection } from '../hooks/useIdeSelection.js'
25
+ import { TODO_WRITE_TOOL_NAME } from '../tools/TodoWriteTool/constants.js'
26
+ import { TASK_CREATE_TOOL_NAME } from '../tools/TaskCreateTool/constants.js'
27
+ import { TASK_UPDATE_TOOL_NAME } from '../tools/TaskUpdateTool/constants.js'
28
+ import { BASH_TOOL_NAME } from '../tools/BashTool/toolName.js'
29
+ import { SKILL_TOOL_NAME } from '../tools/SkillTool/constants.js'
30
+ import type { TodoList } from './todo/types.js'
31
+ import {
32
+ type Task,
33
+ listTasks,
34
+ getTaskListId,
35
+ isTodoV2Enabled,
36
+ } from './tasks.js'
37
+ import { getPlanFilePath, getPlan } from './plans.js'
38
+ import { getConnectedIdeName } from './ide.js'
39
+ import {
40
+ filterInjectedMemoryFiles,
41
+ getManagedAndUserConditionalRules,
42
+ getMemoryFiles,
43
+ getMemoryFilesForNestedDirectory,
44
+ getConditionalRulesForCwdLevelDirectory,
45
+ type MemoryFileInfo,
46
+ } from './claudemd.js'
47
+ import { dirname, parse, relative, resolve } from 'path'
48
+ import { getCwd } from 'src/utils/cwd.js'
49
+ import { getViewedTeammateTask } from '../state/selectors.js'
50
+ import { logError } from './log.js'
51
+ import { logAntError } from './debug.js'
52
+ import { isENOENT, toError } from './errors.js'
53
+ import type { DiagnosticFile } from '../services/diagnosticTracking.js'
54
+ import { diagnosticTracker } from '../services/diagnosticTracking.js'
55
+ import type {
56
+ AttachmentMessage,
57
+ Message,
58
+ MessageOrigin,
59
+ } from 'src/types/message.js'
60
+ import {
61
+ type QueuedCommand,
62
+ getImagePasteIds,
63
+ isValidImagePaste,
64
+ } from 'src/types/textInputTypes.js'
65
+ import { randomUUID, type UUID } from 'crypto'
66
+ import { getSettings_DEPRECATED } from './settings/settings.js'
67
+ import { getSnippetForTwoFileDiff } from 'src/tools/FileEditTool/utils.js'
68
+ import type {
69
+ ContentBlockParam,
70
+ ImageBlockParam,
71
+ Base64ImageSource,
72
+ } from '@anthropic-ai/sdk/resources/messages.mjs'
73
+ import { maybeResizeAndDownsampleImageBlock } from './imageResizer.js'
74
+ import type { PastedContent } from './config.js'
75
+ import { getGlobalConfig } from './config.js'
76
+ import {
77
+ getDefaultSonnetModel,
78
+ getDefaultHaikuModel,
79
+ getDefaultOpusModel,
80
+ } from './model/model.js'
81
+ import type { ReadResourceResult } from '@modelcontextprotocol/sdk/types.js'
82
+ import { getSkillToolCommands, getMcpSkillCommands } from '../commands.js'
83
+ import type { Command } from '../types/command.js'
84
+ import uniqBy from 'lodash-es/uniqBy.js'
85
+ import { getProjectRoot } from '../bootstrap/state.js'
86
+ import { formatCommandsWithinBudget } from '../tools/SkillTool/prompt.js'
87
+ import { getContextWindowForModel } from './context.js'
88
+ import type { DiscoverySignal } from '../services/skillSearch/signals.js'
89
+ // Conditional require for DCE. All skill-search string literals that would
90
+ // otherwise leak into external builds live inside these modules. The only
91
+ // surfaces in THIS file are: the maybe() call (gated via spread below) and
92
+ // the skill_listing suppression check (uses the same skillSearchModules null
93
+ // check). The type-only DiscoverySignal import above is erased at compile time.
94
+ /* eslint-disable @typescript-eslint/no-require-imports */
95
+ const skillSearchModules = feature('EXPERIMENTAL_SKILL_SEARCH')
96
+ ? {
97
+ featureCheck:
98
+ require('../services/skillSearch/featureCheck.js') as typeof import('../services/skillSearch/featureCheck.js'),
99
+ prefetch:
100
+ require('../services/skillSearch/prefetch.js') as typeof import('../services/skillSearch/prefetch.js'),
101
+ }
102
+ : null
103
+ const autoModeStateModule = feature('TRANSCRIPT_CLASSIFIER')
104
+ ? (require('./permissions/autoModeState.js') as typeof import('./permissions/autoModeState.js'))
105
+ : null
106
+ /* eslint-enable @typescript-eslint/no-require-imports */
107
+ import {
108
+ MAX_LINES_TO_READ,
109
+ FILE_READ_TOOL_NAME,
110
+ } from 'src/tools/FileReadTool/prompt.js'
111
+ import { getDefaultFileReadingLimits } from 'src/tools/FileReadTool/limits.js'
112
+ import { cacheKeys, type FileStateCache } from './fileStateCache.js'
113
+ import {
114
+ createAbortController,
115
+ createChildAbortController,
116
+ } from './abortController.js'
117
+ import { isAbortError } from './errors.js'
118
+ import {
119
+ getFileModificationTimeAsync,
120
+ isFileWithinReadSizeLimit,
121
+ } from './file.js'
122
+ import type { AgentDefinition } from '../tools/AgentTool/loadAgentsDir.js'
123
+ import { filterAgentsByMcpRequirements } from '../tools/AgentTool/loadAgentsDir.js'
124
+ import { AGENT_TOOL_NAME } from '../tools/AgentTool/constants.js'
125
+ import {
126
+ formatAgentLine,
127
+ shouldInjectAgentListInMessages,
128
+ } from '../tools/AgentTool/prompt.js'
129
+ import { filterDeniedAgents } from './permissions/permissions.js'
130
+ import { getSubscriptionType } from './auth.js'
131
+ import { mcpInfoFromString } from '../services/mcp/mcpStringUtils.js'
132
+ import {
133
+ matchingRuleForInput,
134
+ pathInAllowedWorkingPath,
135
+ } from './permissions/filesystem.js'
136
+ import {
137
+ generateTaskAttachments,
138
+ applyTaskOffsetsAndEvictions,
139
+ } from './task/framework.js'
140
+ import { getTaskOutputPath } from './task/diskOutput.js'
141
+ import { drainPendingMessages } from '../tasks/LocalAgentTask/LocalAgentTask.js'
142
+ import type { TaskType, TaskStatus } from '../Task.js'
143
+ import {
144
+ getOriginalCwd,
145
+ getSessionId,
146
+ getSdkBetas,
147
+ getTotalCostUSD,
148
+ getTotalOutputTokens,
149
+ getCurrentTurnTokenBudget,
150
+ getTurnOutputTokens,
151
+ hasExitedPlanModeInSession,
152
+ setHasExitedPlanMode,
153
+ needsPlanModeExitAttachment,
154
+ setNeedsPlanModeExitAttachment,
155
+ needsAutoModeExitAttachment,
156
+ setNeedsAutoModeExitAttachment,
157
+ getLastEmittedDate,
158
+ setLastEmittedDate,
159
+ getKairosActive,
160
+ } from '../bootstrap/state.js'
161
+ import type { QuerySource } from '../constants/querySource.js'
162
+ import {
163
+ getDeferredToolsDelta,
164
+ isDeferredToolsDeltaEnabled,
165
+ isToolSearchEnabledOptimistic,
166
+ isToolSearchToolAvailable,
167
+ modelSupportsToolReference,
168
+ type DeferredToolsDeltaScanContext,
169
+ } from './toolSearch.js'
170
+ import {
171
+ getMcpInstructionsDelta,
172
+ isMcpInstructionsDeltaEnabled,
173
+ type ClientSideInstruction,
174
+ } from './mcpInstructionsDelta.js'
175
+ import { CLAUDE_IN_CHROME_MCP_SERVER_NAME } from './claudeInChrome/common.js'
176
+ import { CHROME_TOOL_SEARCH_INSTRUCTIONS } from './claudeInChrome/prompt.js'
177
+ import type { MCPServerConnection } from '../services/mcp/types.js'
178
+ import type {
179
+ HookEvent,
180
+ SyncHookJSONOutput,
181
+ } from 'src/entrypoints/agentSdkTypes.js'
182
+ import {
183
+ checkForAsyncHookResponses,
184
+ removeDeliveredAsyncHooks,
185
+ } from './hooks/AsyncHookRegistry.js'
186
+ import {
187
+ checkForLSPDiagnostics,
188
+ clearAllLSPDiagnostics,
189
+ } from '../services/lsp/LSPDiagnosticRegistry.js'
190
+ import { logForDebugging } from './debug.js'
191
+ import {
192
+ extractTextContent,
193
+ getUserMessageText,
194
+ isThinkingMessage,
195
+ } from './messages.js'
196
+ import { isHumanTurn } from './messagePredicates.js'
197
+ import { isEnvTruthy, getClaudeConfigHomeDir } from './envUtils.js'
198
+ import { feature } from 'bun:bundle'
199
+ /* eslint-disable @typescript-eslint/no-require-imports */
200
+ const BRIEF_TOOL_NAME: string | null =
201
+ feature('KAIROS') || feature('KAIROS_BRIEF')
202
+ ? (
203
+ require('../tools/BriefTool/prompt.js') as typeof import('../tools/BriefTool/prompt.js')
204
+ ).BRIEF_TOOL_NAME
205
+ : null
206
+ const sessionTranscriptModule = feature('KAIROS')
207
+ ? (require('../services/sessionTranscript/sessionTranscript.js') as typeof import('../services/sessionTranscript/sessionTranscript.js'))
208
+ : null
209
+ /* eslint-enable @typescript-eslint/no-require-imports */
210
+ import { hasUltrathinkKeyword, isUltrathinkEnabled } from './thinking.js'
211
+ import {
212
+ tokenCountFromLastAPIResponse,
213
+ tokenCountWithEstimation,
214
+ } from './tokens.js'
215
+ import {
216
+ getEffectiveContextWindowSize,
217
+ isAutoCompactEnabled,
218
+ } from '../services/compact/autoCompact.js'
219
+ import { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js'
220
+ import {
221
+ hasInstructionsLoadedHook,
222
+ executeInstructionsLoadedHooks,
223
+ type HookBlockingError,
224
+ type InstructionsMemoryType,
225
+ } from './hooks.js'
226
+ import { jsonStringify } from './slowOperations.js'
227
+ import { isPDFExtension } from './pdfUtils.js'
228
+ import { getLocalISODate } from '../constants/common.js'
229
+ import { getPDFPageCount } from './pdf.js'
230
+ import { PDF_AT_MENTION_INLINE_THRESHOLD } from '../constants/apiLimits.js'
231
+ import { isAgentSwarmsEnabled } from './agentSwarmsEnabled.js'
232
+ import { findRelevantMemories } from '../memdir/findRelevantMemories.js'
233
+ import { memoryAge, memoryFreshnessText } from '../memdir/memoryAge.js'
234
+ import { getAutoMemPath, isAutoMemoryEnabled } from '../memdir/paths.js'
235
+ import { getAgentMemoryDir } from '../tools/AgentTool/agentMemory.js'
236
+ import {
237
+ readUnreadMessages,
238
+ markMessagesAsReadByPredicate,
239
+ isShutdownApproved,
240
+ isStructuredProtocolMessage,
241
+ isIdleNotification,
242
+ } from './teammateMailbox.js'
243
+ import {
244
+ getAgentName,
245
+ getAgentId,
246
+ getTeamName,
247
+ isTeamLead,
248
+ } from './teammate.js'
249
+ import { isInProcessTeammate } from './teammateContext.js'
250
+ import { removeTeammateFromTeamFile } from './swarm/teamHelpers.js'
251
+ import { unassignTeammateTasks } from './tasks.js'
252
+ import { getCompanionIntroAttachment } from '../buddy/prompt.js'
253
+
254
+ export const TODO_REMINDER_CONFIG = {
255
+ TURNS_SINCE_WRITE: 10,
256
+ TURNS_BETWEEN_REMINDERS: 10,
257
+ } as const
258
+
259
+ export const PLAN_MODE_ATTACHMENT_CONFIG = {
260
+ TURNS_BETWEEN_ATTACHMENTS: 5,
261
+ FULL_REMINDER_EVERY_N_ATTACHMENTS: 5,
262
+ } as const
263
+
264
+ export const AUTO_MODE_ATTACHMENT_CONFIG = {
265
+ TURNS_BETWEEN_ATTACHMENTS: 5,
266
+ FULL_REMINDER_EVERY_N_ATTACHMENTS: 5,
267
+ } as const
268
+
269
+ const MAX_MEMORY_LINES = 200
270
+ // Line cap alone doesn't bound size (200 × 500-char lines = 100KB). The
271
+ // surfacer injects up to 5 files per turn via <system-reminder>, bypassing
272
+ // the per-message tool-result budget, so a tight per-file byte cap keeps
273
+ // aggregate injection bounded (5 × 4KB = 20KB/turn). Enforced via
274
+ // readFileInRange's truncateOnByteLimit option. Truncation means the
275
+ // most-relevant memory still surfaces: the frontmatter + opening context
276
+ // is usually what matters.
277
+ const MAX_MEMORY_BYTES = 4096
278
+
279
+ export const RELEVANT_MEMORIES_CONFIG = {
280
+ // Per-turn cap (5 × 4KB = 20KB) bounds a single injection, but over a
281
+ // long session the selector keeps surfacing distinct files — ~26K tokens/
282
+ // session observed in prod. Cap the cumulative bytes: once hit, stop
283
+ // prefetching entirely. Budget is ~3 full injections; after that the
284
+ // most-relevant memories are already in context. Scanning messages
285
+ // (rather than tracking in toolUseContext) means compact naturally
286
+ // resets the counter — old attachments are gone from context, so
287
+ // re-surfacing is valid.
288
+ MAX_SESSION_BYTES: 60 * 1024,
289
+ } as const
290
+
291
+ export const VERIFY_PLAN_REMINDER_CONFIG = {
292
+ TURNS_BETWEEN_REMINDERS: 10,
293
+ } as const
294
+
295
+ export type FileAttachment = {
296
+ type: 'file'
297
+ filename: string
298
+ content: FileReadToolOutput
299
+ /**
300
+ * Whether the file was truncated due to size limits
301
+ */
302
+ truncated?: boolean
303
+ /** Path relative to CWD at creation time, for stable display */
304
+ displayPath: string
305
+ }
306
+
307
+ export type CompactFileReferenceAttachment = {
308
+ type: 'compact_file_reference'
309
+ filename: string
310
+ /** Path relative to CWD at creation time, for stable display */
311
+ displayPath: string
312
+ }
313
+
314
+ export type PDFReferenceAttachment = {
315
+ type: 'pdf_reference'
316
+ filename: string
317
+ pageCount: number
318
+ fileSize: number
319
+ /** Path relative to CWD at creation time, for stable display */
320
+ displayPath: string
321
+ }
322
+
323
+ export type AlreadyReadFileAttachment = {
324
+ type: 'already_read_file'
325
+ filename: string
326
+ content: FileReadToolOutput
327
+ /**
328
+ * Whether the file was truncated due to size limits
329
+ */
330
+ truncated?: boolean
331
+ /** Path relative to CWD at creation time, for stable display */
332
+ displayPath: string
333
+ }
334
+
335
+ export type AgentMentionAttachment = {
336
+ type: 'agent_mention'
337
+ agentType: string
338
+ }
339
+
340
+ export type AsyncHookResponseAttachment = {
341
+ type: 'async_hook_response'
342
+ processId: string
343
+ hookName: string
344
+ hookEvent: HookEvent | 'StatusLine' | 'FileSuggestion'
345
+ toolName?: string
346
+ response: SyncHookJSONOutput
347
+ stdout: string
348
+ stderr: string
349
+ exitCode?: number
350
+ }
351
+
352
+ export type HookAttachment =
353
+ | HookCancelledAttachment
354
+ | {
355
+ type: 'hook_blocking_error'
356
+ blockingError: HookBlockingError
357
+ hookName: string
358
+ toolUseID: string
359
+ hookEvent: HookEvent
360
+ }
361
+ | HookNonBlockingErrorAttachment
362
+ | HookErrorDuringExecutionAttachment
363
+ | {
364
+ type: 'hook_stopped_continuation'
365
+ message: string
366
+ hookName: string
367
+ toolUseID: string
368
+ hookEvent: HookEvent
369
+ }
370
+ | HookSuccessAttachment
371
+ | {
372
+ type: 'hook_additional_context'
373
+ content: string[]
374
+ hookName: string
375
+ toolUseID: string
376
+ hookEvent: HookEvent
377
+ }
378
+ | HookSystemMessageAttachment
379
+ | HookPermissionDecisionAttachment
380
+
381
+ export type HookPermissionDecisionAttachment = {
382
+ type: 'hook_permission_decision'
383
+ decision: 'allow' | 'deny'
384
+ toolUseID: string
385
+ hookEvent: HookEvent
386
+ }
387
+
388
+ export type HookSystemMessageAttachment = {
389
+ type: 'hook_system_message'
390
+ content: string
391
+ hookName: string
392
+ toolUseID: string
393
+ hookEvent: HookEvent
394
+ }
395
+
396
+ export type HookCancelledAttachment = {
397
+ type: 'hook_cancelled'
398
+ hookName: string
399
+ toolUseID: string
400
+ hookEvent: HookEvent
401
+ command?: string
402
+ durationMs?: number
403
+ }
404
+
405
+ export type HookErrorDuringExecutionAttachment = {
406
+ type: 'hook_error_during_execution'
407
+ content: string
408
+ hookName: string
409
+ toolUseID: string
410
+ hookEvent: HookEvent
411
+ command?: string
412
+ durationMs?: number
413
+ }
414
+
415
+ export type HookSuccessAttachment = {
416
+ type: 'hook_success'
417
+ content: string
418
+ hookName: string
419
+ toolUseID: string
420
+ hookEvent: HookEvent
421
+ stdout?: string
422
+ stderr?: string
423
+ exitCode?: number
424
+ command?: string
425
+ durationMs?: number
426
+ }
427
+
428
+ export type HookNonBlockingErrorAttachment = {
429
+ type: 'hook_non_blocking_error'
430
+ hookName: string
431
+ stderr: string
432
+ stdout: string
433
+ exitCode: number
434
+ toolUseID: string
435
+ hookEvent: HookEvent
436
+ command?: string
437
+ durationMs?: number
438
+ }
439
+
440
+ export type Attachment =
441
+ /**
442
+ * User at-mentioned the file
443
+ */
444
+ | FileAttachment
445
+ | CompactFileReferenceAttachment
446
+ | PDFReferenceAttachment
447
+ | AlreadyReadFileAttachment
448
+ /**
449
+ * An at-mentioned file was edited
450
+ */
451
+ | {
452
+ type: 'edited_text_file'
453
+ filename: string
454
+ snippet: string
455
+ }
456
+ | {
457
+ type: 'edited_image_file'
458
+ filename: string
459
+ content: FileReadToolOutput
460
+ }
461
+ | {
462
+ type: 'directory'
463
+ path: string
464
+ content: string
465
+ /** Path relative to CWD at creation time, for stable display */
466
+ displayPath: string
467
+ }
468
+ | {
469
+ type: 'selected_lines_in_ide'
470
+ ideName: string
471
+ lineStart: number
472
+ lineEnd: number
473
+ filename: string
474
+ content: string
475
+ /** Path relative to CWD at creation time, for stable display */
476
+ displayPath: string
477
+ }
478
+ | {
479
+ type: 'opened_file_in_ide'
480
+ filename: string
481
+ }
482
+ | {
483
+ type: 'todo_reminder'
484
+ content: TodoList
485
+ itemCount: number
486
+ }
487
+ | {
488
+ type: 'task_reminder'
489
+ content: Task[]
490
+ itemCount: number
491
+ }
492
+ | {
493
+ type: 'nested_memory'
494
+ path: string
495
+ content: MemoryFileInfo
496
+ /** Path relative to CWD at creation time, for stable display */
497
+ displayPath: string
498
+ }
499
+ | {
500
+ type: 'relevant_memories'
501
+ memories: {
502
+ path: string
503
+ content: string
504
+ mtimeMs: number
505
+ /**
506
+ * Pre-computed header string (age + path prefix). Computed once
507
+ * at attachment-creation time so the rendered bytes are stable
508
+ * across turns — recomputing memoryAge(mtimeMs) at render time
509
+ * calls Date.now(), so "saved 3 days ago" becomes "saved 4 days
510
+ * ago" across turns → different bytes → prompt cache bust.
511
+ * Optional for backward compat with resumed sessions; render
512
+ * path falls back to recomputing if missing.
513
+ */
514
+ header?: string
515
+ /**
516
+ * lineCount when the file was truncated by readMemoriesForSurfacing,
517
+ * else undefined. Threaded to the readFileState write so
518
+ * getChangedFiles skips truncated memories (partial content would
519
+ * yield a misleading diff).
520
+ */
521
+ limit?: number
522
+ }[]
523
+ }
524
+ | {
525
+ type: 'dynamic_skill'
526
+ skillDir: string
527
+ skillNames: string[]
528
+ /** Path relative to CWD at creation time, for stable display */
529
+ displayPath: string
530
+ }
531
+ | {
532
+ type: 'skill_listing'
533
+ content: string
534
+ skillCount: number
535
+ isInitial: boolean
536
+ }
537
+ | {
538
+ type: 'skill_discovery'
539
+ skills: { name: string; description: string; shortId?: string }[]
540
+ signal: DiscoverySignal
541
+ source: 'native' | 'aki' | 'both'
542
+ }
543
+ | {
544
+ type: 'queued_command'
545
+ prompt: string | Array<ContentBlockParam>
546
+ source_uuid?: UUID
547
+ imagePasteIds?: number[]
548
+ /** Original queue mode — 'prompt' for user messages, 'task-notification' for system events */
549
+ commandMode?: string
550
+ /** Provenance carried from QueuedCommand so mid-turn drains preserve it */
551
+ origin?: MessageOrigin
552
+ /** Carried from QueuedCommand.isMeta — distinguishes human-typed from system-injected */
553
+ isMeta?: boolean
554
+ }
555
+ | {
556
+ type: 'output_style'
557
+ style: string
558
+ }
559
+ | {
560
+ type: 'diagnostics'
561
+ files: DiagnosticFile[]
562
+ isNew: boolean
563
+ }
564
+ | {
565
+ type: 'plan_mode'
566
+ reminderType: 'full' | 'sparse'
567
+ isSubAgent?: boolean
568
+ planFilePath: string
569
+ planExists: boolean
570
+ }
571
+ | {
572
+ type: 'plan_mode_reentry'
573
+ planFilePath: string
574
+ }
575
+ | {
576
+ type: 'plan_mode_exit'
577
+ planFilePath: string
578
+ planExists: boolean
579
+ }
580
+ | {
581
+ type: 'auto_mode'
582
+ reminderType: 'full' | 'sparse'
583
+ }
584
+ | {
585
+ type: 'auto_mode_exit'
586
+ }
587
+ | {
588
+ type: 'critical_system_reminder'
589
+ content: string
590
+ }
591
+ | {
592
+ type: 'plan_file_reference'
593
+ planFilePath: string
594
+ planContent: string
595
+ }
596
+ | {
597
+ type: 'mcp_resource'
598
+ server: string
599
+ uri: string
600
+ name: string
601
+ description?: string
602
+ content: ReadResourceResult
603
+ }
604
+ | {
605
+ type: 'command_permissions'
606
+ allowedTools: string[]
607
+ model?: string
608
+ }
609
+ | AgentMentionAttachment
610
+ | {
611
+ type: 'task_status'
612
+ taskId: string
613
+ taskType: TaskType
614
+ status: TaskStatus
615
+ description: string
616
+ deltaSummary: string | null
617
+ outputFilePath?: string
618
+ }
619
+ | AsyncHookResponseAttachment
620
+ | {
621
+ type: 'token_usage'
622
+ used: number
623
+ total: number
624
+ remaining: number
625
+ }
626
+ | {
627
+ type: 'budget_usd'
628
+ used: number
629
+ total: number
630
+ remaining: number
631
+ }
632
+ | {
633
+ type: 'output_token_usage'
634
+ turn: number
635
+ session: number
636
+ budget: number | null
637
+ }
638
+ | {
639
+ type: 'structured_output'
640
+ data: unknown
641
+ }
642
+ | TeammateMailboxAttachment
643
+ | TeamContextAttachment
644
+ | HookAttachment
645
+ | {
646
+ type: 'invoked_skills'
647
+ skills: Array<{
648
+ name: string
649
+ path: string
650
+ content: string
651
+ }>
652
+ }
653
+ | {
654
+ type: 'verify_plan_reminder'
655
+ }
656
+ | {
657
+ type: 'max_turns_reached'
658
+ maxTurns: number
659
+ turnCount: number
660
+ }
661
+ | {
662
+ type: 'current_session_memory'
663
+ content: string
664
+ path: string
665
+ tokenCount: number
666
+ }
667
+ | {
668
+ type: 'teammate_shutdown_batch'
669
+ count: number
670
+ }
671
+ | {
672
+ type: 'compaction_reminder'
673
+ }
674
+ | {
675
+ type: 'context_efficiency'
676
+ }
677
+ | {
678
+ type: 'date_change'
679
+ newDate: string
680
+ }
681
+ | {
682
+ type: 'ultrathink_effort'
683
+ level: 'high'
684
+ }
685
+ | {
686
+ type: 'deferred_tools_delta'
687
+ addedNames: string[]
688
+ addedLines: string[]
689
+ removedNames: string[]
690
+ }
691
+ | {
692
+ type: 'agent_listing_delta'
693
+ addedTypes: string[]
694
+ addedLines: string[]
695
+ removedTypes: string[]
696
+ /** True when this is the first announcement in the conversation */
697
+ isInitial: boolean
698
+ /** Whether to include the "launch multiple agents concurrently" note (non-pro subscriptions) */
699
+ showConcurrencyNote: boolean
700
+ }
701
+ | {
702
+ type: 'mcp_instructions_delta'
703
+ addedNames: string[]
704
+ addedBlocks: string[]
705
+ removedNames: string[]
706
+ }
707
+ | {
708
+ type: 'companion_intro'
709
+ name: string
710
+ species: string
711
+ }
712
+ | {
713
+ type: 'bagel_console'
714
+ errorCount: number
715
+ warningCount: number
716
+ sample: string
717
+ }
718
+
719
+ export type TeammateMailboxAttachment = {
720
+ type: 'teammate_mailbox'
721
+ messages: Array<{
722
+ from: string
723
+ text: string
724
+ timestamp: string
725
+ color?: string
726
+ summary?: string
727
+ }>
728
+ }
729
+
730
+ export type TeamContextAttachment = {
731
+ type: 'team_context'
732
+ agentId: string
733
+ agentName: string
734
+ teamName: string
735
+ teamConfigPath: string
736
+ taskListPath: string
737
+ }
738
+
739
+ /**
740
+ * This is janky
741
+ * TODO: Generate attachments when we create messages
742
+ */
743
+ export async function getAttachments(
744
+ input: string | null,
745
+ toolUseContext: ToolUseContext,
746
+ ideSelection: IDESelection | null,
747
+ queuedCommands: QueuedCommand[],
748
+ messages?: Message[],
749
+ querySource?: QuerySource,
750
+ options?: { skipSkillDiscovery?: boolean },
751
+ ): Promise<Attachment[]> {
752
+ if (
753
+ isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_ATTACHMENTS) ||
754
+ isEnvTruthy(process.env.CLAUDE_CODE_SIMPLE)
755
+ ) {
756
+ // query.ts:removeFromQueue dequeues these unconditionally after
757
+ // getAttachmentMessages runs — returning [] here silently drops them.
758
+ // Coworker runs with --bare and depends on task-notification for
759
+ // mid-tool-call notifications from Local*Task/Remote*Task.
760
+ return getQueuedCommandAttachments(queuedCommands)
761
+ }
762
+
763
+ // This will slow down submissions
764
+ // TODO: Compute attachments as the user types, not here (though we use this
765
+ // function for slash command prompts too)
766
+ const abortController = createAbortController()
767
+ const timeoutId = setTimeout(ac => ac.abort(), 1000, abortController)
768
+ const context = { ...toolUseContext, abortController }
769
+
770
+ const isMainThread = !toolUseContext.agentId
771
+
772
+ // Attachments which are added in response to on user input
773
+ const userInputAttachments = input
774
+ ? [
775
+ maybe('at_mentioned_files', () =>
776
+ processAtMentionedFiles(input, context),
777
+ ),
778
+ maybe('mcp_resources', () =>
779
+ processMcpResourceAttachments(input, context),
780
+ ),
781
+ maybe('agent_mentions', () =>
782
+ Promise.resolve(
783
+ processAgentMentions(
784
+ input,
785
+ toolUseContext.options.agentDefinitions.activeAgents,
786
+ ),
787
+ ),
788
+ ),
789
+ // Skill discovery on turn 0 (user input as signal). Inter-turn
790
+ // discovery runs via startSkillDiscoveryPrefetch in query.ts,
791
+ // gated on write-pivot detection — see skillSearch/prefetch.ts.
792
+ // feature() here lets DCE drop the 'skill_discovery' string (and the
793
+ // function it calls) from external builds.
794
+ //
795
+ // skipSkillDiscovery gates out the SKILL.md-expansion path
796
+ // (getMessagesForPromptSlashCommand). When a skill is invoked, its
797
+ // SKILL.md content is passed as `input` here to extract @-mentions —
798
+ // but that content is NOT user intent and must not trigger discovery.
799
+ // Without this gate, a 110KB SKILL.md fires ~3.3s of chunked AKI
800
+ // queries on every skill invocation (session 13a9afae).
801
+ ...(feature('EXPERIMENTAL_SKILL_SEARCH') &&
802
+ skillSearchModules &&
803
+ !options?.skipSkillDiscovery
804
+ ? [
805
+ maybe('skill_discovery', () =>
806
+ skillSearchModules.prefetch.getTurnZeroSkillDiscovery(
807
+ input,
808
+ messages ?? [],
809
+ context,
810
+ ),
811
+ ),
812
+ ]
813
+ : []),
814
+ ]
815
+ : []
816
+
817
+ // Process user input attachments first (includes @mentioned files)
818
+ // This ensures files are added to nestedMemoryAttachmentTriggers before nested_memory processes them
819
+ const userAttachmentResults = await Promise.all(userInputAttachments)
820
+
821
+ // Thread-safe attachments available in sub-agents
822
+ // NOTE: These must be created AFTER userInputAttachments completes to ensure
823
+ // nestedMemoryAttachmentTriggers is populated before getNestedMemoryAttachments runs
824
+ const allThreadAttachments = [
825
+ // queuedCommands is already agent-scoped by the drain gate in query.ts —
826
+ // main thread gets agentId===undefined, subagents get their own agentId.
827
+ // Must run for all threads or subagent notifications drain into the void
828
+ // (removed from queue by removeFromQueue but never attached).
829
+ maybe('queued_commands', () => getQueuedCommandAttachments(queuedCommands)),
830
+ maybe('date_change', () =>
831
+ Promise.resolve(getDateChangeAttachments(messages)),
832
+ ),
833
+ maybe('ultrathink_effort', () =>
834
+ Promise.resolve(getUltrathinkEffortAttachment(input)),
835
+ ),
836
+ maybe('deferred_tools_delta', () =>
837
+ Promise.resolve(
838
+ getDeferredToolsDeltaAttachment(
839
+ toolUseContext.options.tools,
840
+ toolUseContext.options.mainLoopModel,
841
+ messages,
842
+ {
843
+ callSite: isMainThread
844
+ ? 'attachments_main'
845
+ : 'attachments_subagent',
846
+ querySource,
847
+ },
848
+ ),
849
+ ),
850
+ ),
851
+ maybe('agent_listing_delta', () =>
852
+ Promise.resolve(getAgentListingDeltaAttachment(toolUseContext, messages)),
853
+ ),
854
+ maybe('mcp_instructions_delta', () =>
855
+ Promise.resolve(
856
+ getMcpInstructionsDeltaAttachment(
857
+ toolUseContext.options.mcpClients,
858
+ toolUseContext.options.tools,
859
+ toolUseContext.options.mainLoopModel,
860
+ messages,
861
+ ),
862
+ ),
863
+ ),
864
+ ...(feature('BUDDY')
865
+ ? [
866
+ maybe('companion_intro', () =>
867
+ Promise.resolve(getCompanionIntroAttachment(messages)),
868
+ ),
869
+ ]
870
+ : []),
871
+ maybe('changed_files', () => getChangedFiles(context)),
872
+ maybe('nested_memory', () => getNestedMemoryAttachments(context)),
873
+ // relevant_memories moved to async prefetch (startRelevantMemoryPrefetch)
874
+ maybe('dynamic_skill', () => getDynamicSkillAttachments(context)),
875
+ maybe('skill_listing', () => getSkillListingAttachments(context)),
876
+ // Inter-turn skill discovery now runs via startSkillDiscoveryPrefetch
877
+ // (query.ts, concurrent with the main turn). The blocking call that
878
+ // previously lived here was the assistant_turn signal — 97% of those
879
+ // Haiku calls found nothing in prod. Prefetch + await-at-collection
880
+ // replaces it; see src/services/skillSearch/prefetch.ts.
881
+ maybe('plan_mode', () => getPlanModeAttachments(messages, toolUseContext)),
882
+ maybe('plan_mode_exit', () => getPlanModeExitAttachment(toolUseContext)),
883
+ ...(feature('TRANSCRIPT_CLASSIFIER')
884
+ ? [
885
+ maybe('auto_mode', () =>
886
+ getAutoModeAttachments(messages, toolUseContext),
887
+ ),
888
+ maybe('auto_mode_exit', () =>
889
+ getAutoModeExitAttachment(toolUseContext),
890
+ ),
891
+ ]
892
+ : []),
893
+ maybe('todo_reminders', () =>
894
+ isTodoV2Enabled()
895
+ ? getTaskReminderAttachments(messages, toolUseContext)
896
+ : getTodoReminderAttachments(messages, toolUseContext),
897
+ ),
898
+ ...(isAgentSwarmsEnabled()
899
+ ? [
900
+ // Skip teammate mailbox for the session_memory forked agent.
901
+ // It shares AppState.teamContext with the leader, so isTeamLead resolves
902
+ // true and it reads+marks-as-read the leader's DMs as ephemeral attachments,
903
+ // silently stealing messages that should be delivered as permanent turns.
904
+ ...(querySource === 'session_memory'
905
+ ? []
906
+ : [
907
+ maybe('teammate_mailbox', async () =>
908
+ getTeammateMailboxAttachments(toolUseContext),
909
+ ),
910
+ ]),
911
+ maybe('team_context', async () =>
912
+ getTeamContextAttachment(messages ?? []),
913
+ ),
914
+ ]
915
+ : []),
916
+ maybe('agent_pending_messages', async () =>
917
+ getAgentPendingMessageAttachments(toolUseContext),
918
+ ),
919
+ maybe('critical_system_reminder', () =>
920
+ Promise.resolve(getCriticalSystemReminderAttachment(toolUseContext)),
921
+ ),
922
+ ...(feature('COMPACTION_REMINDERS')
923
+ ? [
924
+ maybe('compaction_reminder', () =>
925
+ Promise.resolve(
926
+ getCompactionReminderAttachment(
927
+ messages ?? [],
928
+ toolUseContext.options.mainLoopModel,
929
+ ),
930
+ ),
931
+ ),
932
+ ]
933
+ : []),
934
+ ...(feature('HISTORY_SNIP')
935
+ ? [
936
+ maybe('context_efficiency', () =>
937
+ Promise.resolve(getContextEfficiencyAttachment(messages ?? [])),
938
+ ),
939
+ ]
940
+ : []),
941
+ ]
942
+
943
+ // Attachments which are semantically only for the main conversation or don't have concurrency-safe implementations
944
+ const mainThreadAttachments = isMainThread
945
+ ? [
946
+ maybe('ide_selection', async () =>
947
+ getSelectedLinesFromIDE(ideSelection, toolUseContext),
948
+ ),
949
+ maybe('ide_opened_file', async () =>
950
+ getOpenedFileFromIDE(ideSelection, toolUseContext),
951
+ ),
952
+ maybe('output_style', async () =>
953
+ Promise.resolve(getOutputStyleAttachment()),
954
+ ),
955
+ maybe('diagnostics', async () =>
956
+ getDiagnosticAttachments(toolUseContext),
957
+ ),
958
+ maybe('lsp_diagnostics', async () =>
959
+ getLSPDiagnosticAttachments(toolUseContext),
960
+ ),
961
+ maybe('unified_tasks', async () =>
962
+ getUnifiedTaskAttachments(toolUseContext),
963
+ ),
964
+ maybe('async_hook_responses', async () =>
965
+ getAsyncHookResponseAttachments(),
966
+ ),
967
+ maybe('token_usage', async () =>
968
+ Promise.resolve(
969
+ getTokenUsageAttachment(
970
+ messages ?? [],
971
+ toolUseContext.options.mainLoopModel,
972
+ ),
973
+ ),
974
+ ),
975
+ maybe('budget_usd', async () =>
976
+ Promise.resolve(
977
+ getMaxBudgetUsdAttachment(toolUseContext.options.maxBudgetUsd),
978
+ ),
979
+ ),
980
+ maybe('output_token_usage', async () =>
981
+ Promise.resolve(getOutputTokenUsageAttachment()),
982
+ ),
983
+ maybe('verify_plan_reminder', async () =>
984
+ getVerifyPlanReminderAttachment(messages, toolUseContext),
985
+ ),
986
+ ]
987
+ : []
988
+
989
+ // Process thread and main thread attachments in parallel (no dependencies between them)
990
+ const [threadAttachmentResults, mainThreadAttachmentResults] =
991
+ await Promise.all([
992
+ Promise.all(allThreadAttachments),
993
+ Promise.all(mainThreadAttachments),
994
+ ])
995
+
996
+ clearTimeout(timeoutId)
997
+ // Defensive: a getter leaking [undefined] crashes .map(a => a.type) below.
998
+ return [
999
+ ...userAttachmentResults.flat(),
1000
+ ...threadAttachmentResults.flat(),
1001
+ ...mainThreadAttachmentResults.flat(),
1002
+ ].filter(a => a !== undefined && a !== null)
1003
+ }
1004
+
1005
+ async function maybe<A>(label: string, f: () => Promise<A[]>): Promise<A[]> {
1006
+ const startTime = Date.now()
1007
+ try {
1008
+ const result = await f()
1009
+ const duration = Date.now() - startTime
1010
+ // Log only 5% of events to reduce volume
1011
+ if (Math.random() < 0.05) {
1012
+ // jsonStringify(undefined) returns undefined, so .length would throw
1013
+ const attachmentSizeBytes = result
1014
+ .filter(a => a !== undefined && a !== null)
1015
+ .reduce((total, attachment) => {
1016
+ return total + jsonStringify(attachment).length
1017
+ }, 0)
1018
+ logEvent('tengu_attachment_compute_duration', {
1019
+ label,
1020
+ duration_ms: duration,
1021
+ attachment_size_bytes: attachmentSizeBytes,
1022
+ attachment_count: result.length,
1023
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
1024
+ }
1025
+ return result
1026
+ } catch (e) {
1027
+ const duration = Date.now() - startTime
1028
+ // Log only 5% of events to reduce volume
1029
+ if (Math.random() < 0.05) {
1030
+ logEvent('tengu_attachment_compute_duration', {
1031
+ label,
1032
+ duration_ms: duration,
1033
+ error: true,
1034
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
1035
+ }
1036
+ logError(e)
1037
+ // For Ant users, log the full error to help with debugging
1038
+ logAntError(`Attachment error in ${label}`, e)
1039
+
1040
+ return []
1041
+ }
1042
+ }
1043
+
1044
+ const INLINE_NOTIFICATION_MODES = new Set(['prompt', 'task-notification'])
1045
+
1046
+ export async function getQueuedCommandAttachments(
1047
+ queuedCommands: QueuedCommand[],
1048
+ ): Promise<Attachment[]> {
1049
+ if (!queuedCommands) {
1050
+ return []
1051
+ }
1052
+ // Include both 'prompt' and 'task-notification' commands as attachments.
1053
+ // During proactive agentic loops, task-notification commands would otherwise
1054
+ // stay in the queue permanently (useQueueProcessor can't run while a query
1055
+ // is active), causing hasPendingNotifications() to return true and Sleep to
1056
+ // wake immediately with 0ms duration in an infinite loop.
1057
+ const filtered = queuedCommands.filter(_ =>
1058
+ INLINE_NOTIFICATION_MODES.has(_.mode),
1059
+ )
1060
+ return Promise.all(
1061
+ filtered.map(async _ => {
1062
+ const imageBlocks = await buildImageContentBlocks(_.pastedContents)
1063
+ let prompt: string | Array<ContentBlockParam> = _.value
1064
+ if (imageBlocks.length > 0) {
1065
+ // Build content block array with text + images so the model sees them
1066
+ const textValue =
1067
+ typeof _.value === 'string'
1068
+ ? _.value
1069
+ : extractTextContent(_.value, '\n')
1070
+ prompt = [{ type: 'text' as const, text: textValue }, ...imageBlocks]
1071
+ }
1072
+ return {
1073
+ type: 'queued_command' as const,
1074
+ prompt,
1075
+ source_uuid: _.uuid,
1076
+ imagePasteIds: getImagePasteIds(_.pastedContents),
1077
+ commandMode: _.mode,
1078
+ origin: _.origin,
1079
+ isMeta: _.isMeta,
1080
+ }
1081
+ }),
1082
+ )
1083
+ }
1084
+
1085
+ export function getAgentPendingMessageAttachments(
1086
+ toolUseContext: ToolUseContext,
1087
+ ): Attachment[] {
1088
+ const agentId = toolUseContext.agentId
1089
+ if (!agentId) return []
1090
+ const drained = drainPendingMessages(
1091
+ agentId,
1092
+ toolUseContext.getAppState,
1093
+ toolUseContext.setAppStateForTasks ?? toolUseContext.setAppState,
1094
+ )
1095
+ return drained.map(msg => ({
1096
+ type: 'queued_command' as const,
1097
+ prompt: msg,
1098
+ origin: { kind: 'coordinator' as const },
1099
+ isMeta: true,
1100
+ }))
1101
+ }
1102
+
1103
+ async function buildImageContentBlocks(
1104
+ pastedContents: Record<number, PastedContent> | undefined,
1105
+ ): Promise<ImageBlockParam[]> {
1106
+ if (!pastedContents) {
1107
+ return []
1108
+ }
1109
+ const imageContents = Object.values(pastedContents).filter(isValidImagePaste)
1110
+ if (imageContents.length === 0) {
1111
+ return []
1112
+ }
1113
+ const results = await Promise.all(
1114
+ imageContents.map(async img => {
1115
+ const imageBlock: ImageBlockParam = {
1116
+ type: 'image',
1117
+ source: {
1118
+ type: 'base64',
1119
+ media_type: (img.mediaType ||
1120
+ 'image/png') as Base64ImageSource['media_type'],
1121
+ data: img.content,
1122
+ },
1123
+ }
1124
+ const resized = await maybeResizeAndDownsampleImageBlock(imageBlock)
1125
+ return resized.block
1126
+ }),
1127
+ )
1128
+ return results
1129
+ }
1130
+
1131
+ function getPlanModeAttachmentTurnCount(messages: Message[]): {
1132
+ turnCount: number
1133
+ foundPlanModeAttachment: boolean
1134
+ } {
1135
+ let turnsSinceLastAttachment = 0
1136
+ let foundPlanModeAttachment = false
1137
+
1138
+ // Iterate backwards to find most recent plan_mode attachment.
1139
+ // Count HUMAN turns (non-meta, non-tool-result user messages), not assistant
1140
+ // messages — the tool loop in query.ts calls getAttachmentMessages on every
1141
+ // tool round, so counting assistant messages would fire the reminder every
1142
+ // 5 tool calls instead of every 5 human turns.
1143
+ for (let i = messages.length - 1; i >= 0; i--) {
1144
+ const message = messages[i]
1145
+
1146
+ if (
1147
+ message?.type === 'user' &&
1148
+ !message.isMeta &&
1149
+ !hasToolResultContent(message.message.content)
1150
+ ) {
1151
+ turnsSinceLastAttachment++
1152
+ } else if (
1153
+ message?.type === 'attachment' &&
1154
+ (message.attachment.type === 'plan_mode' ||
1155
+ message.attachment.type === 'plan_mode_reentry')
1156
+ ) {
1157
+ foundPlanModeAttachment = true
1158
+ break
1159
+ }
1160
+ }
1161
+
1162
+ return { turnCount: turnsSinceLastAttachment, foundPlanModeAttachment }
1163
+ }
1164
+
1165
+ /**
1166
+ * Count plan_mode attachments since the last plan_mode_exit (or from start if no exit).
1167
+ * This ensures the full/sparse cycle resets when re-entering plan mode.
1168
+ */
1169
+ function countPlanModeAttachmentsSinceLastExit(messages: Message[]): number {
1170
+ let count = 0
1171
+ // Iterate backwards - if we hit a plan_mode_exit, stop counting
1172
+ for (let i = messages.length - 1; i >= 0; i--) {
1173
+ const message = messages[i]
1174
+ if (message?.type === 'attachment') {
1175
+ if (message.attachment.type === 'plan_mode_exit') {
1176
+ break // Stop counting at the last exit
1177
+ }
1178
+ if (message.attachment.type === 'plan_mode') {
1179
+ count++
1180
+ }
1181
+ }
1182
+ }
1183
+ return count
1184
+ }
1185
+
1186
+ async function getPlanModeAttachments(
1187
+ messages: Message[] | undefined,
1188
+ toolUseContext: ToolUseContext,
1189
+ ): Promise<Attachment[]> {
1190
+ const appState = toolUseContext.getAppState()
1191
+ const permissionContext = appState.toolPermissionContext
1192
+ if (permissionContext.mode !== 'plan') {
1193
+ return []
1194
+ }
1195
+
1196
+ // Check if we should attach based on turn count (except for first turn)
1197
+ if (messages && messages.length > 0) {
1198
+ const { turnCount, foundPlanModeAttachment } =
1199
+ getPlanModeAttachmentTurnCount(messages)
1200
+ // Only throttle if we've already sent a plan_mode attachment before
1201
+ // On first turn in plan mode, always attach
1202
+ if (
1203
+ foundPlanModeAttachment &&
1204
+ turnCount < PLAN_MODE_ATTACHMENT_CONFIG.TURNS_BETWEEN_ATTACHMENTS
1205
+ ) {
1206
+ return []
1207
+ }
1208
+ }
1209
+
1210
+ const planFilePath = getPlanFilePath(toolUseContext.agentId)
1211
+ const existingPlan = getPlan(toolUseContext.agentId)
1212
+
1213
+ const attachments: Attachment[] = []
1214
+
1215
+ // Check for re-entry: flag is set AND plan file exists
1216
+ if (hasExitedPlanModeInSession() && existingPlan !== null) {
1217
+ attachments.push({ type: 'plan_mode_reentry', planFilePath })
1218
+ setHasExitedPlanMode(false) // Clear flag - one-time guidance
1219
+ }
1220
+
1221
+ // Determine if this should be a full or sparse reminder
1222
+ // Full reminder on 1st, 6th, 11th... (every Nth attachment)
1223
+ const attachmentCount =
1224
+ countPlanModeAttachmentsSinceLastExit(messages ?? []) + 1
1225
+ const reminderType: 'full' | 'sparse' =
1226
+ attachmentCount %
1227
+ PLAN_MODE_ATTACHMENT_CONFIG.FULL_REMINDER_EVERY_N_ATTACHMENTS ===
1228
+ 1
1229
+ ? 'full'
1230
+ : 'sparse'
1231
+
1232
+ // Always add the main plan_mode attachment
1233
+ attachments.push({
1234
+ type: 'plan_mode',
1235
+ reminderType,
1236
+ isSubAgent: !!toolUseContext.agentId,
1237
+ planFilePath,
1238
+ planExists: existingPlan !== null,
1239
+ })
1240
+
1241
+ return attachments
1242
+ }
1243
+
1244
+ /**
1245
+ * Returns a plan_mode_exit attachment if we just exited plan mode.
1246
+ * This is a one-time notification to tell the model it's no longer in plan mode.
1247
+ */
1248
+ async function getPlanModeExitAttachment(
1249
+ toolUseContext: ToolUseContext,
1250
+ ): Promise<Attachment[]> {
1251
+ // Only trigger if the flag is set (we just exited plan mode)
1252
+ if (!needsPlanModeExitAttachment()) {
1253
+ return []
1254
+ }
1255
+
1256
+ const appState = toolUseContext.getAppState()
1257
+ if (appState.toolPermissionContext.mode === 'plan') {
1258
+ setNeedsPlanModeExitAttachment(false)
1259
+ return []
1260
+ }
1261
+
1262
+ // Clear the flag - this is a one-time notification
1263
+ setNeedsPlanModeExitAttachment(false)
1264
+
1265
+ const planFilePath = getPlanFilePath(toolUseContext.agentId)
1266
+ const planExists = getPlan(toolUseContext.agentId) !== null
1267
+
1268
+ // Note: skill discovery does NOT fire on plan exit. By the time the plan is
1269
+ // written, it's too late — the model should have had relevant skills WHILE
1270
+ // planning. The user_message signal already fires on the request that
1271
+ // triggers planning ("plan how to deploy this"), which is the right moment.
1272
+ return [{ type: 'plan_mode_exit', planFilePath, planExists }]
1273
+ }
1274
+
1275
+ function getAutoModeAttachmentTurnCount(messages: Message[]): {
1276
+ turnCount: number
1277
+ foundAutoModeAttachment: boolean
1278
+ } {
1279
+ let turnsSinceLastAttachment = 0
1280
+ let foundAutoModeAttachment = false
1281
+
1282
+ // Iterate backwards to find most recent auto_mode attachment.
1283
+ // Count HUMAN turns (non-meta, non-tool-result user messages), not assistant
1284
+ // messages — the tool loop in query.ts calls getAttachmentMessages on every
1285
+ // tool round, so a single human turn with 100 tool calls would fire ~20
1286
+ // reminders if we counted assistant messages. Auto mode's target use case is
1287
+ // long agentic sessions, where this accumulated 60-105× per session.
1288
+ for (let i = messages.length - 1; i >= 0; i--) {
1289
+ const message = messages[i]
1290
+
1291
+ if (
1292
+ message?.type === 'user' &&
1293
+ !message.isMeta &&
1294
+ !hasToolResultContent(message.message.content)
1295
+ ) {
1296
+ turnsSinceLastAttachment++
1297
+ } else if (
1298
+ message?.type === 'attachment' &&
1299
+ message.attachment.type === 'auto_mode'
1300
+ ) {
1301
+ foundAutoModeAttachment = true
1302
+ break
1303
+ } else if (
1304
+ message?.type === 'attachment' &&
1305
+ message.attachment.type === 'auto_mode_exit'
1306
+ ) {
1307
+ // Exit resets the throttle — treat as if no prior attachment exists
1308
+ break
1309
+ }
1310
+ }
1311
+
1312
+ return { turnCount: turnsSinceLastAttachment, foundAutoModeAttachment }
1313
+ }
1314
+
1315
+ /**
1316
+ * Count auto_mode attachments since the last auto_mode_exit (or from start if no exit).
1317
+ * This ensures the full/sparse cycle resets when re-entering auto mode.
1318
+ */
1319
+ function countAutoModeAttachmentsSinceLastExit(messages: Message[]): number {
1320
+ let count = 0
1321
+ for (let i = messages.length - 1; i >= 0; i--) {
1322
+ const message = messages[i]
1323
+ if (message?.type === 'attachment') {
1324
+ if (message.attachment.type === 'auto_mode_exit') {
1325
+ break
1326
+ }
1327
+ if (message.attachment.type === 'auto_mode') {
1328
+ count++
1329
+ }
1330
+ }
1331
+ }
1332
+ return count
1333
+ }
1334
+
1335
+ async function getAutoModeAttachments(
1336
+ messages: Message[] | undefined,
1337
+ toolUseContext: ToolUseContext,
1338
+ ): Promise<Attachment[]> {
1339
+ const appState = toolUseContext.getAppState()
1340
+ const permissionContext = appState.toolPermissionContext
1341
+ const inAuto = permissionContext.mode === 'auto'
1342
+ const inPlanWithAuto =
1343
+ permissionContext.mode === 'plan' &&
1344
+ (autoModeStateModule?.isAutoModeActive() ?? false)
1345
+ if (!inAuto && !inPlanWithAuto) {
1346
+ return []
1347
+ }
1348
+
1349
+ // Check if we should attach based on turn count (except for first turn)
1350
+ if (messages && messages.length > 0) {
1351
+ const { turnCount, foundAutoModeAttachment } =
1352
+ getAutoModeAttachmentTurnCount(messages)
1353
+ // Only throttle if we've already sent an auto_mode attachment before
1354
+ // On first turn in auto mode, always attach
1355
+ if (
1356
+ foundAutoModeAttachment &&
1357
+ turnCount < AUTO_MODE_ATTACHMENT_CONFIG.TURNS_BETWEEN_ATTACHMENTS
1358
+ ) {
1359
+ return []
1360
+ }
1361
+ }
1362
+
1363
+ // Determine if this should be a full or sparse reminder
1364
+ const attachmentCount =
1365
+ countAutoModeAttachmentsSinceLastExit(messages ?? []) + 1
1366
+ const reminderType: 'full' | 'sparse' =
1367
+ attachmentCount %
1368
+ AUTO_MODE_ATTACHMENT_CONFIG.FULL_REMINDER_EVERY_N_ATTACHMENTS ===
1369
+ 1
1370
+ ? 'full'
1371
+ : 'sparse'
1372
+
1373
+ return [{ type: 'auto_mode', reminderType }]
1374
+ }
1375
+
1376
+ /**
1377
+ * Returns an auto_mode_exit attachment if we just exited auto mode.
1378
+ * This is a one-time notification to tell the model it's no longer in auto mode.
1379
+ */
1380
+ async function getAutoModeExitAttachment(
1381
+ toolUseContext: ToolUseContext,
1382
+ ): Promise<Attachment[]> {
1383
+ if (!needsAutoModeExitAttachment()) {
1384
+ return []
1385
+ }
1386
+
1387
+ const appState = toolUseContext.getAppState()
1388
+ // Suppress when auto is still active — covers both mode==='auto' and
1389
+ // plan-with-auto-active (where mode==='plan' but classifier runs).
1390
+ if (
1391
+ appState.toolPermissionContext.mode === 'auto' ||
1392
+ (autoModeStateModule?.isAutoModeActive() ?? false)
1393
+ ) {
1394
+ setNeedsAutoModeExitAttachment(false)
1395
+ return []
1396
+ }
1397
+
1398
+ setNeedsAutoModeExitAttachment(false)
1399
+ return [{ type: 'auto_mode_exit' }]
1400
+ }
1401
+
1402
+ /**
1403
+ * Detects when the local date has changed since the last turn (user coding
1404
+ * past midnight) and emits an attachment to notify the model.
1405
+ *
1406
+ * The date_change attachment is appended at the tail of the conversation,
1407
+ * so the model learns the new date without mutating the cached prefix.
1408
+ * messages[0] (from getUserContext → prependUserContext) intentionally
1409
+ * keeps the stale date — clearing that cache would regenerate the prefix
1410
+ * and turn the entire conversation into cache_creation on the next turn
1411
+ * (~920K effective tokens per midnight crossing per overnight session).
1412
+ *
1413
+ * Exported for testing — regression guard for the cache-clear removal.
1414
+ */
1415
+ export function getDateChangeAttachments(
1416
+ messages: Message[] | undefined,
1417
+ ): Attachment[] {
1418
+ const currentDate = getLocalISODate()
1419
+ const lastDate = getLastEmittedDate()
1420
+
1421
+ if (lastDate === null) {
1422
+ // First turn — just record, no attachment needed
1423
+ setLastEmittedDate(currentDate)
1424
+ return []
1425
+ }
1426
+
1427
+ if (currentDate === lastDate) {
1428
+ return []
1429
+ }
1430
+
1431
+ setLastEmittedDate(currentDate)
1432
+
1433
+ // Assistant mode: flush yesterday's transcript to the per-day file so
1434
+ // the /dream skill (1–5am local) finds it even if no compaction fires
1435
+ // today. Fire-and-forget; writeSessionTranscriptSegment buckets by
1436
+ // message timestamp so a multi-day gap flushes each day correctly.
1437
+ if (feature('KAIROS')) {
1438
+ if (getKairosActive() && messages !== undefined) {
1439
+ sessionTranscriptModule?.flushOnDateChange(messages, currentDate)
1440
+ }
1441
+ }
1442
+
1443
+ return [{ type: 'date_change', newDate: currentDate }]
1444
+ }
1445
+
1446
+ function getUltrathinkEffortAttachment(input: string | null): Attachment[] {
1447
+ if (!isUltrathinkEnabled() || !input || !hasUltrathinkKeyword(input)) {
1448
+ return []
1449
+ }
1450
+ logEvent('tengu_ultrathink', {})
1451
+ return [{ type: 'ultrathink_effort', level: 'high' }]
1452
+ }
1453
+
1454
+ // Exported for compact.ts — the gate must be identical at both call sites.
1455
+ export function getDeferredToolsDeltaAttachment(
1456
+ tools: Tools,
1457
+ model: string,
1458
+ messages: Message[] | undefined,
1459
+ scanContext?: DeferredToolsDeltaScanContext,
1460
+ ): Attachment[] {
1461
+ if (!isDeferredToolsDeltaEnabled()) return []
1462
+ // These three checks mirror the sync parts of isToolSearchEnabled —
1463
+ // the attachment text says "available via ToolSearch", so ToolSearch
1464
+ // has to actually be in the request. The async auto-threshold check
1465
+ // is not replicated (would double-fire tengu_tool_search_mode_decision);
1466
+ // in tst-auto below-threshold the attachment can fire while ToolSearch
1467
+ // is filtered out, but that's a narrow case and the tools announced
1468
+ // are directly callable anyway.
1469
+ if (!isToolSearchEnabledOptimistic()) return []
1470
+ if (!modelSupportsToolReference(model)) return []
1471
+ if (!isToolSearchToolAvailable(tools)) return []
1472
+ const delta = getDeferredToolsDelta(tools, messages ?? [], scanContext)
1473
+ if (!delta) return []
1474
+ return [{ type: 'deferred_tools_delta', ...delta }]
1475
+ }
1476
+
1477
+ /**
1478
+ * Diff the current filtered agent pool against what's already been announced
1479
+ * in this conversation (reconstructed from prior agent_listing_delta
1480
+ * attachments). Returns [] if nothing changed or the gate is off.
1481
+ *
1482
+ * The agent list was embedded in AgentTool's description, causing ~10.2% of
1483
+ * fleet cache_creation: MCP async connect, /reload-plugins, or
1484
+ * permission-mode change → description changes → full tool-schema cache bust.
1485
+ * Moving the list here keeps the tool description static.
1486
+ *
1487
+ * Exported for compact.ts — re-announces the full set after compaction eats
1488
+ * prior deltas.
1489
+ */
1490
+ export function getAgentListingDeltaAttachment(
1491
+ toolUseContext: ToolUseContext,
1492
+ messages: Message[] | undefined,
1493
+ ): Attachment[] {
1494
+ if (!shouldInjectAgentListInMessages()) return []
1495
+
1496
+ // Skip if AgentTool isn't in the pool — the listing would be unactionable.
1497
+ if (
1498
+ !toolUseContext.options.tools.some(t => toolMatchesName(t, AGENT_TOOL_NAME))
1499
+ ) {
1500
+ return []
1501
+ }
1502
+
1503
+ const { activeAgents, allowedAgentTypes } =
1504
+ toolUseContext.options.agentDefinitions
1505
+
1506
+ // Mirror AgentTool.prompt()'s filtering: MCP requirements → deny rules →
1507
+ // allowedAgentTypes restriction. Keep this in sync with AgentTool.tsx.
1508
+ const mcpServers = new Set<string>()
1509
+ for (const tool of toolUseContext.options.tools) {
1510
+ const info = mcpInfoFromString(tool.name)
1511
+ if (info) mcpServers.add(info.serverName)
1512
+ }
1513
+ const permissionContext = toolUseContext.getAppState().toolPermissionContext
1514
+ let filtered = filterDeniedAgents(
1515
+ filterAgentsByMcpRequirements(activeAgents, [...mcpServers]),
1516
+ permissionContext,
1517
+ AGENT_TOOL_NAME,
1518
+ )
1519
+ if (allowedAgentTypes) {
1520
+ filtered = filtered.filter(a => allowedAgentTypes.includes(a.agentType))
1521
+ }
1522
+
1523
+ // Reconstruct announced set from prior deltas in the transcript.
1524
+ const announced = new Set<string>()
1525
+ for (const msg of messages ?? []) {
1526
+ if (msg.type !== 'attachment') continue
1527
+ if (msg.attachment.type !== 'agent_listing_delta') continue
1528
+ for (const t of msg.attachment.addedTypes) announced.add(t)
1529
+ for (const t of msg.attachment.removedTypes) announced.delete(t)
1530
+ }
1531
+
1532
+ const currentTypes = new Set(filtered.map(a => a.agentType))
1533
+ const added = filtered.filter(a => !announced.has(a.agentType))
1534
+ const removed: string[] = []
1535
+ for (const t of announced) {
1536
+ if (!currentTypes.has(t)) removed.push(t)
1537
+ }
1538
+
1539
+ if (added.length === 0 && removed.length === 0) return []
1540
+
1541
+ // Sort for deterministic output — agent load order is nondeterministic
1542
+ // (plugin load races, MCP async connect).
1543
+ added.sort((a, b) => a.agentType.localeCompare(b.agentType))
1544
+ removed.sort()
1545
+
1546
+ return [
1547
+ {
1548
+ type: 'agent_listing_delta',
1549
+ addedTypes: added.map(a => a.agentType),
1550
+ addedLines: added.map(formatAgentLine),
1551
+ removedTypes: removed,
1552
+ isInitial: announced.size === 0,
1553
+ showConcurrencyNote: getSubscriptionType() !== 'pro',
1554
+ },
1555
+ ]
1556
+ }
1557
+
1558
+ // Exported for compact.ts / reactiveCompact.ts — single source of truth for the gate.
1559
+ export function getMcpInstructionsDeltaAttachment(
1560
+ mcpClients: MCPServerConnection[],
1561
+ tools: Tools,
1562
+ model: string,
1563
+ messages: Message[] | undefined,
1564
+ ): Attachment[] {
1565
+ if (!isMcpInstructionsDeltaEnabled()) return []
1566
+
1567
+ // The chrome ToolSearch hint is client-authored and ToolSearch-conditional;
1568
+ // actual server `instructions` are unconditional. Decide the chrome part
1569
+ // here, pass it into the pure diff as a synthesized entry.
1570
+ const clientSide: ClientSideInstruction[] = []
1571
+ if (
1572
+ isToolSearchEnabledOptimistic() &&
1573
+ modelSupportsToolReference(model) &&
1574
+ isToolSearchToolAvailable(tools)
1575
+ ) {
1576
+ clientSide.push({
1577
+ serverName: CLAUDE_IN_CHROME_MCP_SERVER_NAME,
1578
+ block: CHROME_TOOL_SEARCH_INSTRUCTIONS,
1579
+ })
1580
+ }
1581
+
1582
+ const delta = getMcpInstructionsDelta(mcpClients, messages ?? [], clientSide)
1583
+ if (!delta) return []
1584
+ return [{ type: 'mcp_instructions_delta', ...delta }]
1585
+ }
1586
+
1587
+ function getCriticalSystemReminderAttachment(
1588
+ toolUseContext: ToolUseContext,
1589
+ ): Attachment[] {
1590
+ const reminder = toolUseContext.criticalSystemReminder_EXPERIMENTAL
1591
+ if (!reminder) {
1592
+ return []
1593
+ }
1594
+ return [{ type: 'critical_system_reminder', content: reminder }]
1595
+ }
1596
+
1597
+ function getOutputStyleAttachment(): Attachment[] {
1598
+ const settings = getSettings_DEPRECATED()
1599
+ const outputStyle = settings?.outputStyle || 'default'
1600
+
1601
+ // Only show for non-default styles
1602
+ if (outputStyle === 'default') {
1603
+ return []
1604
+ }
1605
+
1606
+ return [
1607
+ {
1608
+ type: 'output_style',
1609
+ style: outputStyle,
1610
+ },
1611
+ ]
1612
+ }
1613
+
1614
+ async function getSelectedLinesFromIDE(
1615
+ ideSelection: IDESelection | null,
1616
+ toolUseContext: ToolUseContext,
1617
+ ): Promise<Attachment[]> {
1618
+ const ideName = getConnectedIdeName(toolUseContext.options.mcpClients)
1619
+ if (
1620
+ !ideName ||
1621
+ ideSelection?.lineStart === undefined ||
1622
+ !ideSelection.text ||
1623
+ !ideSelection.filePath
1624
+ ) {
1625
+ return []
1626
+ }
1627
+
1628
+ const appState = toolUseContext.getAppState()
1629
+ if (isFileReadDenied(ideSelection.filePath, appState.toolPermissionContext)) {
1630
+ return []
1631
+ }
1632
+
1633
+ return [
1634
+ {
1635
+ type: 'selected_lines_in_ide',
1636
+ ideName,
1637
+ lineStart: ideSelection.lineStart,
1638
+ lineEnd: ideSelection.lineStart + ideSelection.lineCount - 1,
1639
+ filename: ideSelection.filePath,
1640
+ content: ideSelection.text,
1641
+ displayPath: relative(getCwd(), ideSelection.filePath),
1642
+ },
1643
+ ]
1644
+ }
1645
+
1646
+ /**
1647
+ * Computes the directories to process for nested memory file loading.
1648
+ * Returns two lists:
1649
+ * - nestedDirs: Directories between CWD and targetPath (processed for CLAUDE.md + all rules)
1650
+ * - cwdLevelDirs: Directories from root to CWD (processed for conditional rules only)
1651
+ *
1652
+ * @param targetPath The target file path
1653
+ * @param originalCwd The original current working directory
1654
+ * @returns Object with nestedDirs and cwdLevelDirs arrays, both ordered from parent to child
1655
+ */
1656
+ export function getDirectoriesToProcess(
1657
+ targetPath: string,
1658
+ originalCwd: string,
1659
+ ): { nestedDirs: string[]; cwdLevelDirs: string[] } {
1660
+ // Build list of directories from original CWD to targetPath's directory
1661
+ const targetDir = dirname(resolve(targetPath))
1662
+ const nestedDirs: string[] = []
1663
+ let currentDir = targetDir
1664
+
1665
+ // Walk up from target directory to original CWD
1666
+ while (currentDir !== originalCwd && currentDir !== parse(currentDir).root) {
1667
+ if (currentDir.startsWith(originalCwd)) {
1668
+ nestedDirs.push(currentDir)
1669
+ }
1670
+ currentDir = dirname(currentDir)
1671
+ }
1672
+
1673
+ // Reverse to get order from CWD down to target
1674
+ nestedDirs.reverse()
1675
+
1676
+ // Build list of directories from root to CWD (for conditional rules only)
1677
+ const cwdLevelDirs: string[] = []
1678
+ currentDir = originalCwd
1679
+
1680
+ while (currentDir !== parse(currentDir).root) {
1681
+ cwdLevelDirs.push(currentDir)
1682
+ currentDir = dirname(currentDir)
1683
+ }
1684
+
1685
+ // Reverse to get order from root to CWD
1686
+ cwdLevelDirs.reverse()
1687
+
1688
+ return { nestedDirs, cwdLevelDirs }
1689
+ }
1690
+
1691
+ /**
1692
+ * Converts memory files to attachments, filtering out already-loaded files.
1693
+ *
1694
+ * @param memoryFiles The memory files to convert
1695
+ * @param toolUseContext The tool use context (for tracking loaded files)
1696
+ * @returns Array of nested memory attachments
1697
+ */
1698
+ function isInstructionsMemoryType(
1699
+ type: MemoryFileInfo['type'],
1700
+ ): type is InstructionsMemoryType {
1701
+ return (
1702
+ type === 'User' ||
1703
+ type === 'Project' ||
1704
+ type === 'Local' ||
1705
+ type === 'Managed'
1706
+ )
1707
+ }
1708
+
1709
+ /** Exported for testing — regression guard for LRU-eviction re-injection. */
1710
+ export function memoryFilesToAttachments(
1711
+ memoryFiles: MemoryFileInfo[],
1712
+ toolUseContext: ToolUseContext,
1713
+ triggerFilePath?: string,
1714
+ ): Attachment[] {
1715
+ const attachments: Attachment[] = []
1716
+ const shouldFireHook = hasInstructionsLoadedHook()
1717
+
1718
+ for (const memoryFile of memoryFiles) {
1719
+ // Dedup: loadedNestedMemoryPaths is a non-evicting Set; readFileState
1720
+ // is a 100-entry LRU that drops entries in busy sessions, so relying
1721
+ // on it alone re-injects the same CLAUDE.md on every eviction cycle.
1722
+ if (toolUseContext.loadedNestedMemoryPaths?.has(memoryFile.path)) {
1723
+ continue
1724
+ }
1725
+ if (!toolUseContext.readFileState.has(memoryFile.path)) {
1726
+ attachments.push({
1727
+ type: 'nested_memory',
1728
+ path: memoryFile.path,
1729
+ content: memoryFile,
1730
+ displayPath: relative(getCwd(), memoryFile.path),
1731
+ })
1732
+ toolUseContext.loadedNestedMemoryPaths?.add(memoryFile.path)
1733
+
1734
+ // Mark as loaded in readFileState — this provides cross-function and
1735
+ // cross-turn dedup via the .has() check above.
1736
+ //
1737
+ // When the injected content doesn't match disk (stripped HTML comments,
1738
+ // stripped frontmatter, truncated MEMORY.md), cache the RAW disk bytes
1739
+ // with `isPartialView: true`. Edit/Write see the flag and require a real
1740
+ // Read first; getChangedFiles sees real content + undefined offset/limit
1741
+ // so mid-session change detection still works.
1742
+ toolUseContext.readFileState.set(memoryFile.path, {
1743
+ content: memoryFile.contentDiffersFromDisk
1744
+ ? (memoryFile.rawContent ?? memoryFile.content)
1745
+ : memoryFile.content,
1746
+ timestamp: Date.now(),
1747
+ offset: undefined,
1748
+ limit: undefined,
1749
+ isPartialView: memoryFile.contentDiffersFromDisk,
1750
+ })
1751
+
1752
+
1753
+ // Fire InstructionsLoaded hook for audit/observability (fire-and-forget)
1754
+ if (shouldFireHook && isInstructionsMemoryType(memoryFile.type)) {
1755
+ const loadReason = memoryFile.globs
1756
+ ? 'path_glob_match'
1757
+ : memoryFile.parent
1758
+ ? 'include'
1759
+ : 'nested_traversal'
1760
+ void executeInstructionsLoadedHooks(
1761
+ memoryFile.path,
1762
+ memoryFile.type,
1763
+ loadReason,
1764
+ {
1765
+ globs: memoryFile.globs,
1766
+ triggerFilePath,
1767
+ parentFilePath: memoryFile.parent,
1768
+ },
1769
+ )
1770
+ }
1771
+ }
1772
+ }
1773
+
1774
+ return attachments
1775
+ }
1776
+
1777
+ /**
1778
+ * Loads nested memory files for a given file path and returns them as attachments.
1779
+ * This function performs directory traversal to find CLAUDE.md files and conditional rules
1780
+ * that apply to the target file path.
1781
+ *
1782
+ * Processing order (must be preserved):
1783
+ * 1. Managed/User conditional rules matching targetPath
1784
+ * 2. Nested directories (CWD → target): CLAUDE.md + unconditional + conditional rules
1785
+ * 3. CWD-level directories (root → CWD): conditional rules only
1786
+ *
1787
+ * @param filePath The file path to get nested memory files for
1788
+ * @param toolUseContext The tool use context
1789
+ * @param appState The app state containing tool permission context
1790
+ * @returns Array of nested memory attachments
1791
+ */
1792
+ async function getNestedMemoryAttachmentsForFile(
1793
+ filePath: string,
1794
+ toolUseContext: ToolUseContext,
1795
+ appState: { toolPermissionContext: ToolPermissionContext },
1796
+ ): Promise<Attachment[]> {
1797
+ const attachments: Attachment[] = []
1798
+
1799
+ try {
1800
+ // Early return if path is not in allowed working path
1801
+ if (!pathInAllowedWorkingPath(filePath, appState.toolPermissionContext)) {
1802
+ return attachments
1803
+ }
1804
+
1805
+ const processedPaths = new Set<string>()
1806
+ const originalCwd = getOriginalCwd()
1807
+
1808
+ // Phase 1: Process Managed and User conditional rules
1809
+ const managedUserRules = await getManagedAndUserConditionalRules(
1810
+ filePath,
1811
+ processedPaths,
1812
+ )
1813
+ attachments.push(
1814
+ ...memoryFilesToAttachments(managedUserRules, toolUseContext, filePath),
1815
+ )
1816
+
1817
+ // Phase 2: Get directories to process
1818
+ const { nestedDirs, cwdLevelDirs } = getDirectoriesToProcess(
1819
+ filePath,
1820
+ originalCwd,
1821
+ )
1822
+
1823
+ const skipProjectLevel = getFeatureValue_CACHED_MAY_BE_STALE(
1824
+ 'tengu_paper_halyard',
1825
+ false,
1826
+ )
1827
+
1828
+ // Phase 3: Process nested directories (CWD → target)
1829
+ // Each directory gets: CLAUDE.md + unconditional rules + conditional rules
1830
+ for (const dir of nestedDirs) {
1831
+ const memoryFiles = (
1832
+ await getMemoryFilesForNestedDirectory(dir, filePath, processedPaths)
1833
+ ).filter(
1834
+ f => !skipProjectLevel || (f.type !== 'Project' && f.type !== 'Local'),
1835
+ )
1836
+ attachments.push(
1837
+ ...memoryFilesToAttachments(memoryFiles, toolUseContext, filePath),
1838
+ )
1839
+ }
1840
+
1841
+ // Phase 4: Process CWD-level directories (root → CWD)
1842
+ // Only conditional rules (unconditional rules are already loaded eagerly)
1843
+ for (const dir of cwdLevelDirs) {
1844
+ const conditionalRules = (
1845
+ await getConditionalRulesForCwdLevelDirectory(
1846
+ dir,
1847
+ filePath,
1848
+ processedPaths,
1849
+ )
1850
+ ).filter(
1851
+ f => !skipProjectLevel || (f.type !== 'Project' && f.type !== 'Local'),
1852
+ )
1853
+ attachments.push(
1854
+ ...memoryFilesToAttachments(conditionalRules, toolUseContext, filePath),
1855
+ )
1856
+ }
1857
+ } catch (error) {
1858
+ logError(error)
1859
+ }
1860
+
1861
+ return attachments
1862
+ }
1863
+
1864
+ async function getOpenedFileFromIDE(
1865
+ ideSelection: IDESelection | null,
1866
+ toolUseContext: ToolUseContext,
1867
+ ): Promise<Attachment[]> {
1868
+ if (!ideSelection?.filePath || ideSelection.text) {
1869
+ return []
1870
+ }
1871
+
1872
+ const appState = toolUseContext.getAppState()
1873
+ if (isFileReadDenied(ideSelection.filePath, appState.toolPermissionContext)) {
1874
+ return []
1875
+ }
1876
+
1877
+ // Get nested memory files
1878
+ const nestedMemoryAttachments = await getNestedMemoryAttachmentsForFile(
1879
+ ideSelection.filePath,
1880
+ toolUseContext,
1881
+ appState,
1882
+ )
1883
+
1884
+ // Return nested memory attachments followed by the opened file attachment
1885
+ return [
1886
+ ...nestedMemoryAttachments,
1887
+ {
1888
+ type: 'opened_file_in_ide',
1889
+ filename: ideSelection.filePath,
1890
+ },
1891
+ ]
1892
+ }
1893
+
1894
+ async function processAtMentionedFiles(
1895
+ input: string,
1896
+ toolUseContext: ToolUseContext,
1897
+ ): Promise<Attachment[]> {
1898
+ const files = extractAtMentionedFiles(input)
1899
+ if (files.length === 0) return []
1900
+
1901
+ const appState = toolUseContext.getAppState()
1902
+ const results = await Promise.all(
1903
+ files.map(async file => {
1904
+ try {
1905
+ const { filename, lineStart, lineEnd } = parseAtMentionedFileLines(file)
1906
+ const absoluteFilename = expandPath(filename)
1907
+
1908
+ if (
1909
+ isFileReadDenied(absoluteFilename, appState.toolPermissionContext)
1910
+ ) {
1911
+ return null
1912
+ }
1913
+
1914
+ // Check if it's a directory
1915
+ try {
1916
+ const stats = await stat(absoluteFilename)
1917
+ if (stats.isDirectory()) {
1918
+ try {
1919
+ const entries = await readdir(absoluteFilename, {
1920
+ withFileTypes: true,
1921
+ })
1922
+ const MAX_DIR_ENTRIES = 1000
1923
+ const truncated = entries.length > MAX_DIR_ENTRIES
1924
+ const names = entries.slice(0, MAX_DIR_ENTRIES).map(e => e.name)
1925
+ if (truncated) {
1926
+ names.push(
1927
+ `\u2026 and ${entries.length - MAX_DIR_ENTRIES} more entries`,
1928
+ )
1929
+ }
1930
+ const stdout = names.join('\n')
1931
+ logEvent('tengu_at_mention_extracting_directory_success', {})
1932
+
1933
+ return {
1934
+ type: 'directory' as const,
1935
+ path: absoluteFilename,
1936
+ content: stdout,
1937
+ displayPath: relative(getCwd(), absoluteFilename),
1938
+ }
1939
+ } catch {
1940
+ return null
1941
+ }
1942
+ }
1943
+ } catch {
1944
+ // If stat fails, continue with file logic
1945
+ }
1946
+
1947
+ return await generateFileAttachment(
1948
+ absoluteFilename,
1949
+ toolUseContext,
1950
+ 'tengu_at_mention_extracting_filename_success',
1951
+ 'tengu_at_mention_extracting_filename_error',
1952
+ 'at-mention',
1953
+ {
1954
+ offset: lineStart,
1955
+ limit: lineEnd && lineStart ? lineEnd - lineStart + 1 : undefined,
1956
+ },
1957
+ )
1958
+ } catch {
1959
+ logEvent('tengu_at_mention_extracting_filename_error', {})
1960
+ }
1961
+ }),
1962
+ )
1963
+ return results.filter(Boolean) as Attachment[]
1964
+ }
1965
+
1966
+ function processAgentMentions(
1967
+ input: string,
1968
+ agents: AgentDefinition[],
1969
+ ): Attachment[] {
1970
+ const agentMentions = extractAgentMentions(input)
1971
+ if (agentMentions.length === 0) return []
1972
+
1973
+ const results = agentMentions.map(mention => {
1974
+ const agentType = mention.replace('agent-', '')
1975
+ const agentDef = agents.find(def => def.agentType === agentType)
1976
+
1977
+ if (!agentDef) {
1978
+ logEvent('tengu_at_mention_agent_not_found', {})
1979
+ return null
1980
+ }
1981
+
1982
+ logEvent('tengu_at_mention_agent_success', {})
1983
+
1984
+ return {
1985
+ type: 'agent_mention' as const,
1986
+ agentType: agentDef.agentType,
1987
+ }
1988
+ })
1989
+
1990
+ return results.filter(
1991
+ (result): result is NonNullable<typeof result> => result !== null,
1992
+ )
1993
+ }
1994
+
1995
+ async function processMcpResourceAttachments(
1996
+ input: string,
1997
+ toolUseContext: ToolUseContext,
1998
+ ): Promise<Attachment[]> {
1999
+ const resourceMentions = extractMcpResourceMentions(input)
2000
+ if (resourceMentions.length === 0) return []
2001
+
2002
+ const mcpClients = toolUseContext.options.mcpClients || []
2003
+
2004
+ const results = await Promise.all(
2005
+ resourceMentions.map(async mention => {
2006
+ try {
2007
+ const [serverName, ...uriParts] = mention.split(':')
2008
+ const uri = uriParts.join(':') // Rejoin in case URI contains colons
2009
+
2010
+ if (!serverName || !uri) {
2011
+ logEvent('tengu_at_mention_mcp_resource_error', {})
2012
+ return null
2013
+ }
2014
+
2015
+ // Find the MCP client
2016
+ const client = mcpClients.find(c => c.name === serverName)
2017
+ if (!client || client.type !== 'connected') {
2018
+ logEvent('tengu_at_mention_mcp_resource_error', {})
2019
+ return null
2020
+ }
2021
+
2022
+ // Find the resource in available resources to get its metadata
2023
+ const serverResources =
2024
+ toolUseContext.options.mcpResources?.[serverName] || []
2025
+ const resourceInfo = serverResources.find(r => r.uri === uri)
2026
+ if (!resourceInfo) {
2027
+ logEvent('tengu_at_mention_mcp_resource_error', {})
2028
+ return null
2029
+ }
2030
+
2031
+ try {
2032
+ const result = await client.client.readResource({
2033
+ uri,
2034
+ })
2035
+
2036
+ logEvent('tengu_at_mention_mcp_resource_success', {})
2037
+
2038
+ return {
2039
+ type: 'mcp_resource' as const,
2040
+ server: serverName,
2041
+ uri,
2042
+ name: resourceInfo.name || uri,
2043
+ description: resourceInfo.description,
2044
+ content: result,
2045
+ }
2046
+ } catch (error) {
2047
+ logEvent('tengu_at_mention_mcp_resource_error', {})
2048
+ logError(error)
2049
+ return null
2050
+ }
2051
+ } catch {
2052
+ logEvent('tengu_at_mention_mcp_resource_error', {})
2053
+ return null
2054
+ }
2055
+ }),
2056
+ )
2057
+
2058
+ return results.filter(
2059
+ (result): result is NonNullable<typeof result> => result !== null,
2060
+ ) as Attachment[]
2061
+ }
2062
+
2063
+ export async function getChangedFiles(
2064
+ toolUseContext: ToolUseContext,
2065
+ ): Promise<Attachment[]> {
2066
+ const filePaths = cacheKeys(toolUseContext.readFileState)
2067
+ if (filePaths.length === 0) return []
2068
+
2069
+ const appState = toolUseContext.getAppState()
2070
+ const results = await Promise.all(
2071
+ filePaths.map(async filePath => {
2072
+ const fileState = toolUseContext.readFileState.get(filePath)
2073
+ if (!fileState) return null
2074
+
2075
+ // TODO: Implement offset/limit support for changed files
2076
+ if (fileState.offset !== undefined || fileState.limit !== undefined) {
2077
+ return null
2078
+ }
2079
+
2080
+ const normalizedPath = expandPath(filePath)
2081
+
2082
+ // Check if file has a deny rule configured
2083
+ if (isFileReadDenied(normalizedPath, appState.toolPermissionContext)) {
2084
+ return null
2085
+ }
2086
+
2087
+ try {
2088
+ const mtime = await getFileModificationTimeAsync(normalizedPath)
2089
+ if (mtime <= fileState.timestamp) {
2090
+ return null
2091
+ }
2092
+
2093
+ const fileInput = { file_path: normalizedPath }
2094
+
2095
+ // Validate file path is valid
2096
+ const isValid = await FileReadTool.validateInput(
2097
+ fileInput,
2098
+ toolUseContext,
2099
+ )
2100
+ if (!isValid.result) {
2101
+ return null
2102
+ }
2103
+
2104
+ const result = await FileReadTool.call(fileInput, toolUseContext)
2105
+ // Extract only the changed section
2106
+ if (result.data.type === 'text') {
2107
+ const snippet = getSnippetForTwoFileDiff(
2108
+ fileState.content,
2109
+ result.data.file.content,
2110
+ )
2111
+
2112
+ // File was touched but not modified
2113
+ if (snippet === '') {
2114
+ return null
2115
+ }
2116
+
2117
+ return {
2118
+ type: 'edited_text_file' as const,
2119
+ filename: normalizedPath,
2120
+ snippet,
2121
+ }
2122
+ }
2123
+
2124
+ // For non-text files (images), apply the same token limit logic as FileReadTool
2125
+ if (result.data.type === 'image') {
2126
+ try {
2127
+ const data = await readImageWithTokenBudget(normalizedPath)
2128
+ return {
2129
+ type: 'edited_image_file' as const,
2130
+ filename: normalizedPath,
2131
+ content: data,
2132
+ }
2133
+ } catch (compressionError) {
2134
+ logError(compressionError)
2135
+ logEvent('tengu_watched_file_compression_failed', {
2136
+ file: normalizedPath,
2137
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2138
+ return null
2139
+ }
2140
+ }
2141
+
2142
+ // notebook / pdf / parts — no diff representation; explicitly
2143
+ // null so the map callback has no implicit-undefined path.
2144
+ return null
2145
+ } catch (err) {
2146
+ // Evict ONLY on ENOENT (file truly deleted). Transient stat
2147
+ // failures — atomic-save races (editor writes tmp→rename and
2148
+ // stat hits the gap), EACCES churn, network-FS hiccups — must
2149
+ // NOT evict, or the next Edit fails code-6 even though the
2150
+ // file still exists and the model just read it. VS Code
2151
+ // auto-save/format-on-save hits this race especially often.
2152
+ // See regression analysis on PR #18525.
2153
+ if (isENOENT(err)) {
2154
+ toolUseContext.readFileState.delete(filePath)
2155
+ }
2156
+ return null
2157
+ }
2158
+ }),
2159
+ )
2160
+ return results.filter(result => result != null) as Attachment[]
2161
+ }
2162
+
2163
+ /**
2164
+ * Processes paths that need nested memory attachments and checks for nested CLAUDE.md files
2165
+ * Uses nestedMemoryAttachmentTriggers field from ToolUseContext
2166
+ */
2167
+ async function getNestedMemoryAttachments(
2168
+ toolUseContext: ToolUseContext,
2169
+ ): Promise<Attachment[]> {
2170
+ // Check triggers first — getAppState() waits for a React render cycle,
2171
+ // and the common case is an empty trigger set.
2172
+ if (
2173
+ !toolUseContext.nestedMemoryAttachmentTriggers ||
2174
+ toolUseContext.nestedMemoryAttachmentTriggers.size === 0
2175
+ ) {
2176
+ return []
2177
+ }
2178
+
2179
+ const appState = toolUseContext.getAppState()
2180
+ const attachments: Attachment[] = []
2181
+
2182
+ for (const filePath of toolUseContext.nestedMemoryAttachmentTriggers) {
2183
+ const nestedAttachments = await getNestedMemoryAttachmentsForFile(
2184
+ filePath,
2185
+ toolUseContext,
2186
+ appState,
2187
+ )
2188
+ attachments.push(...nestedAttachments)
2189
+ }
2190
+
2191
+ toolUseContext.nestedMemoryAttachmentTriggers.clear()
2192
+
2193
+ return attachments
2194
+ }
2195
+
2196
+ async function getRelevantMemoryAttachments(
2197
+ input: string,
2198
+ agents: AgentDefinition[],
2199
+ readFileState: FileStateCache,
2200
+ recentTools: readonly string[],
2201
+ signal: AbortSignal,
2202
+ alreadySurfaced: ReadonlySet<string>,
2203
+ ): Promise<Attachment[]> {
2204
+ // If an agent is @-mentioned, search only its memory dir (isolation).
2205
+ // Otherwise search the auto-memory dir.
2206
+ const memoryDirs = extractAgentMentions(input).flatMap(mention => {
2207
+ const agentType = mention.replace('agent-', '')
2208
+ const agentDef = agents.find(def => def.agentType === agentType)
2209
+ return agentDef?.memory
2210
+ ? [getAgentMemoryDir(agentType, agentDef.memory)]
2211
+ : []
2212
+ })
2213
+ const dirs = memoryDirs.length > 0 ? memoryDirs : [getAutoMemPath()]
2214
+
2215
+ const allResults = await Promise.all(
2216
+ dirs.map(dir =>
2217
+ findRelevantMemories(
2218
+ input,
2219
+ dir,
2220
+ signal,
2221
+ recentTools,
2222
+ alreadySurfaced,
2223
+ ).catch(() => []),
2224
+ ),
2225
+ )
2226
+ // alreadySurfaced is filtered inside the selector so Sonnet spends its
2227
+ // 5-slot budget on fresh candidates; readFileState catches files the
2228
+ // model read via FileReadTool. The redundant alreadySurfaced check here
2229
+ // is a belt-and-suspenders guard (multi-dir results may re-introduce a
2230
+ // path the selector filtered in a different dir).
2231
+ const selected = allResults
2232
+ .flat()
2233
+ .filter(m => !readFileState.has(m.path) && !alreadySurfaced.has(m.path))
2234
+ .slice(0, 5)
2235
+
2236
+ const memories = await readMemoriesForSurfacing(selected, signal)
2237
+
2238
+ if (memories.length === 0) {
2239
+ return []
2240
+ }
2241
+ return [{ type: 'relevant_memories' as const, memories }]
2242
+ }
2243
+
2244
+ /**
2245
+ * Scan messages for past relevant_memories attachments. Returns both the
2246
+ * set of surfaced paths (for selector de-dup) and cumulative byte count
2247
+ * (for session-total throttle). Scanning messages rather than tracking
2248
+ * in toolUseContext means compact naturally resets both — old attachments
2249
+ * are gone from the compacted transcript, so re-surfacing is valid again.
2250
+ */
2251
+ export function collectSurfacedMemories(messages: ReadonlyArray<Message>): {
2252
+ paths: Set<string>
2253
+ totalBytes: number
2254
+ } {
2255
+ const paths = new Set<string>()
2256
+ let totalBytes = 0
2257
+ for (const m of messages) {
2258
+ if (m.type === 'attachment' && m.attachment.type === 'relevant_memories') {
2259
+ for (const mem of m.attachment.memories) {
2260
+ paths.add(mem.path)
2261
+ totalBytes += mem.content.length
2262
+ }
2263
+ }
2264
+ }
2265
+ return { paths, totalBytes }
2266
+ }
2267
+
2268
+ /**
2269
+ * Reads a set of relevance-ranked memory files for injection as
2270
+ * <system-reminder> attachments. Enforces both MAX_MEMORY_LINES and
2271
+ * MAX_MEMORY_BYTES via readFileInRange's truncateOnByteLimit option.
2272
+ * Truncation surfaces partial
2273
+ * content with a note rather than dropping the file — findRelevantMemories
2274
+ * already picked this as most-relevant, so the frontmatter + opening context
2275
+ * is worth surfacing even if later lines are cut.
2276
+ *
2277
+ * Exported for direct testing without mocking the ranker + GB gates.
2278
+ */
2279
+ export async function readMemoriesForSurfacing(
2280
+ selected: ReadonlyArray<{ path: string; mtimeMs: number }>,
2281
+ signal?: AbortSignal,
2282
+ ): Promise<
2283
+ Array<{
2284
+ path: string
2285
+ content: string
2286
+ mtimeMs: number
2287
+ header: string
2288
+ limit?: number
2289
+ }>
2290
+ > {
2291
+ const results = await Promise.all(
2292
+ selected.map(async ({ path: filePath, mtimeMs }) => {
2293
+ try {
2294
+ const result = await readFileInRange(
2295
+ filePath,
2296
+ 0,
2297
+ MAX_MEMORY_LINES,
2298
+ MAX_MEMORY_BYTES,
2299
+ signal,
2300
+ { truncateOnByteLimit: true },
2301
+ )
2302
+ const truncated =
2303
+ result.totalLines > MAX_MEMORY_LINES || result.truncatedByBytes
2304
+ const content = truncated
2305
+ ? result.content +
2306
+ `\n\n> This memory file was truncated (${result.truncatedByBytes ? `${MAX_MEMORY_BYTES} byte limit` : `first ${MAX_MEMORY_LINES} lines`}). Use the ${FILE_READ_TOOL_NAME} tool to view the complete file at: ${filePath}`
2307
+ : result.content
2308
+ return {
2309
+ path: filePath,
2310
+ content,
2311
+ mtimeMs,
2312
+ header: memoryHeader(filePath, mtimeMs),
2313
+ limit: truncated ? result.lineCount : undefined,
2314
+ }
2315
+ } catch {
2316
+ return null
2317
+ }
2318
+ }),
2319
+ )
2320
+ return results.filter(r => r !== null)
2321
+ }
2322
+
2323
+ /**
2324
+ * Header string for a relevant-memory block. Exported so messages.ts
2325
+ * can fall back for resumed sessions where the stored header is missing.
2326
+ */
2327
+ export function memoryHeader(path: string, mtimeMs: number): string {
2328
+ const staleness = memoryFreshnessText(mtimeMs)
2329
+ return staleness
2330
+ ? `${staleness}\n\nMemory: ${path}:`
2331
+ : `Memory (saved ${memoryAge(mtimeMs)}): ${path}:`
2332
+ }
2333
+
2334
+ /**
2335
+ * A memory relevance-selector prefetch handle. The promise is started once
2336
+ * per user turn and runs while the main model streams and tools execute.
2337
+ * At the collect point (post-tools), the caller reads settledAt to
2338
+ * consume-if-ready or skip-and-retry-next-iteration — the prefetch never
2339
+ * blocks the turn.
2340
+ *
2341
+ * Disposable: query.ts binds with `using`, so [Symbol.dispose] fires on all
2342
+ * generator exit paths (return, throw, .return() closure) — aborting the
2343
+ * in-flight request and emitting terminal telemetry without instrumenting
2344
+ * each of the ~13 return sites inside the while loop.
2345
+ */
2346
+ export type MemoryPrefetch = {
2347
+ promise: Promise<Attachment[]>
2348
+ /** Set by promise.finally(). null until the promise settles. */
2349
+ settledAt: number | null
2350
+ /** Set by the collect point in query.ts. -1 until consumed. */
2351
+ consumedOnIteration: number
2352
+ [Symbol.dispose](): void
2353
+ }
2354
+
2355
+ /**
2356
+ * Starts the relevant memory search as an async prefetch.
2357
+ * Extracts the last real user prompt from messages (skipping isMeta system
2358
+ * injections) and kicks off a non-blocking search. Returns a Disposable
2359
+ * handle with settlement tracking. Bound with `using` in query.ts.
2360
+ */
2361
+ export function startRelevantMemoryPrefetch(
2362
+ messages: ReadonlyArray<Message>,
2363
+ toolUseContext: ToolUseContext,
2364
+ ): MemoryPrefetch | undefined {
2365
+ if (
2366
+ !isAutoMemoryEnabled() ||
2367
+ !getFeatureValue_CACHED_MAY_BE_STALE('tengu_moth_copse', false)
2368
+ ) {
2369
+ return undefined
2370
+ }
2371
+
2372
+ const lastUserMessage = messages.findLast(m => m.type === 'user' && !m.isMeta)
2373
+ if (!lastUserMessage) {
2374
+ return undefined
2375
+ }
2376
+
2377
+ const input = getUserMessageText(lastUserMessage)
2378
+ // Single-word prompts lack enough context for meaningful term extraction
2379
+ if (!input || !/\s/.test(input.trim())) {
2380
+ return undefined
2381
+ }
2382
+
2383
+ const surfaced = collectSurfacedMemories(messages)
2384
+ if (surfaced.totalBytes >= RELEVANT_MEMORIES_CONFIG.MAX_SESSION_BYTES) {
2385
+ return undefined
2386
+ }
2387
+
2388
+ // Chained to the turn-level abort so user Escape cancels the sideQuery
2389
+ // immediately, not just on [Symbol.dispose] when queryLoop exits.
2390
+ const controller = createChildAbortController(toolUseContext.abortController)
2391
+ const firedAt = Date.now()
2392
+ const promise = getRelevantMemoryAttachments(
2393
+ input,
2394
+ toolUseContext.options.agentDefinitions.activeAgents,
2395
+ toolUseContext.readFileState,
2396
+ collectRecentSuccessfulTools(messages, lastUserMessage),
2397
+ controller.signal,
2398
+ surfaced.paths,
2399
+ ).catch(e => {
2400
+ if (!isAbortError(e)) {
2401
+ logError(e)
2402
+ }
2403
+ return []
2404
+ })
2405
+
2406
+ const handle: MemoryPrefetch = {
2407
+ promise,
2408
+ settledAt: null,
2409
+ consumedOnIteration: -1,
2410
+ [Symbol.dispose]() {
2411
+ controller.abort()
2412
+ logEvent('tengu_memdir_prefetch_collected', {
2413
+ hidden_by_first_iteration:
2414
+ handle.settledAt !== null && handle.consumedOnIteration === 0,
2415
+ consumed_on_iteration: handle.consumedOnIteration,
2416
+ latency_ms: (handle.settledAt ?? Date.now()) - firedAt,
2417
+ })
2418
+ },
2419
+ }
2420
+ void promise.finally(() => {
2421
+ handle.settledAt = Date.now()
2422
+ })
2423
+ return handle
2424
+ }
2425
+
2426
+ type ToolResultBlock = {
2427
+ type: 'tool_result'
2428
+ tool_use_id: string
2429
+ is_error?: boolean
2430
+ }
2431
+
2432
+ function isToolResultBlock(b: unknown): b is ToolResultBlock {
2433
+ return (
2434
+ typeof b === 'object' &&
2435
+ b !== null &&
2436
+ (b as ToolResultBlock).type === 'tool_result' &&
2437
+ typeof (b as ToolResultBlock).tool_use_id === 'string'
2438
+ )
2439
+ }
2440
+
2441
+ /**
2442
+ * Check whether a user message's content contains tool_result blocks.
2443
+ * This is more reliable than checking `toolUseResult === undefined` because
2444
+ * sub-agent tool result messages explicitly set `toolUseResult` to `undefined`
2445
+ * when `preserveToolUseResults` is false (the default for Explore agents).
2446
+ */
2447
+ function hasToolResultContent(content: unknown): boolean {
2448
+ return Array.isArray(content) && content.some(isToolResultBlock)
2449
+ }
2450
+
2451
+ /**
2452
+ * Tools that succeeded (and never errored) since the previous real turn
2453
+ * boundary. The memory selector uses this to suppress docs about tools
2454
+ * that are working — surfacing reference material for a tool the model
2455
+ * is already calling successfully is noise.
2456
+ *
2457
+ * Any error → tool excluded (model is struggling, docs stay available).
2458
+ * No result yet → also excluded (outcome unknown).
2459
+ *
2460
+ * tool_use lives in assistant content; tool_result in user content
2461
+ * (toolUseResult set, isMeta undefined). Both are within the scan window.
2462
+ * Backward scan sees results before uses so we collect both by id and
2463
+ * resolve after.
2464
+ */
2465
+ export function collectRecentSuccessfulTools(
2466
+ messages: ReadonlyArray<Message>,
2467
+ lastUserMessage: Message,
2468
+ ): readonly string[] {
2469
+ const useIdToName = new Map<string, string>()
2470
+ const resultByUseId = new Map<string, boolean>()
2471
+ for (let i = messages.length - 1; i >= 0; i--) {
2472
+ const m = messages[i]
2473
+ if (!m) continue
2474
+ if (isHumanTurn(m) && m !== lastUserMessage) break
2475
+ if (m.type === 'assistant' && typeof m.message.content !== 'string') {
2476
+ for (const block of m.message.content) {
2477
+ if (block.type === 'tool_use') useIdToName.set(block.id, block.name)
2478
+ }
2479
+ } else if (
2480
+ m.type === 'user' &&
2481
+ 'message' in m &&
2482
+ Array.isArray(m.message.content)
2483
+ ) {
2484
+ for (const block of m.message.content) {
2485
+ if (isToolResultBlock(block)) {
2486
+ resultByUseId.set(block.tool_use_id, block.is_error === true)
2487
+ }
2488
+ }
2489
+ }
2490
+ }
2491
+ const failed = new Set<string>()
2492
+ const succeeded = new Set<string>()
2493
+ for (const [id, name] of useIdToName) {
2494
+ const errored = resultByUseId.get(id)
2495
+ if (errored === undefined) continue
2496
+ if (errored) {
2497
+ failed.add(name)
2498
+ } else {
2499
+ succeeded.add(name)
2500
+ }
2501
+ }
2502
+ return [...succeeded].filter(t => !failed.has(t))
2503
+ }
2504
+
2505
+
2506
+ /**
2507
+ * Filters prefetched memory attachments to exclude memories the model already
2508
+ * has in context via FileRead/Write/Edit tool calls (any iteration this turn)
2509
+ * or a previous turn's memory surfacing — both tracked in the cumulative
2510
+ * readFileState. Survivors are then marked in readFileState so subsequent
2511
+ * turns won't re-surface them.
2512
+ *
2513
+ * The mark-after-filter ordering is load-bearing: readMemoriesForSurfacing
2514
+ * used to write to readFileState during the prefetch, which meant the filter
2515
+ * saw every prefetch-selected path as "already in context" and dropped them
2516
+ * all (self-referential filter). Deferring the write to here, after the
2517
+ * filter runs, breaks that cycle while still deduping against tool calls
2518
+ * from any iteration.
2519
+ */
2520
+ export function filterDuplicateMemoryAttachments(
2521
+ attachments: Attachment[],
2522
+ readFileState: FileStateCache,
2523
+ ): Attachment[] {
2524
+ return attachments
2525
+ .map(attachment => {
2526
+ if (attachment.type !== 'relevant_memories') return attachment
2527
+ const filtered = attachment.memories.filter(
2528
+ m => !readFileState.has(m.path),
2529
+ )
2530
+ for (const m of filtered) {
2531
+ readFileState.set(m.path, {
2532
+ content: m.content,
2533
+ timestamp: m.mtimeMs,
2534
+ offset: undefined,
2535
+ limit: m.limit,
2536
+ })
2537
+ }
2538
+ return filtered.length > 0 ? { ...attachment, memories: filtered } : null
2539
+ })
2540
+ .filter((a): a is Attachment => a !== null)
2541
+ }
2542
+
2543
+ /**
2544
+ * Processes skill directories that were discovered during file operations.
2545
+ * Uses dynamicSkillDirTriggers field from ToolUseContext
2546
+ */
2547
+ async function getDynamicSkillAttachments(
2548
+ toolUseContext: ToolUseContext,
2549
+ ): Promise<Attachment[]> {
2550
+ const attachments: Attachment[] = []
2551
+
2552
+ if (
2553
+ toolUseContext.dynamicSkillDirTriggers &&
2554
+ toolUseContext.dynamicSkillDirTriggers.size > 0
2555
+ ) {
2556
+ // Parallelize: readdir all skill dirs concurrently
2557
+ const perDirResults = await Promise.all(
2558
+ Array.from(toolUseContext.dynamicSkillDirTriggers).map(async skillDir => {
2559
+ try {
2560
+ const entries = await readdir(skillDir, { withFileTypes: true })
2561
+ const candidates = entries
2562
+ .filter(e => e.isDirectory() || e.isSymbolicLink())
2563
+ .map(e => e.name)
2564
+ // Parallelize: stat all SKILL.md candidates concurrently
2565
+ const checked = await Promise.all(
2566
+ candidates.map(async name => {
2567
+ try {
2568
+ await stat(resolve(skillDir, name, 'SKILL.md'))
2569
+ return name
2570
+ } catch {
2571
+ return null // SKILL.md doesn't exist, skip this entry
2572
+ }
2573
+ }),
2574
+ )
2575
+ return {
2576
+ skillDir,
2577
+ skillNames: checked.filter((n): n is string => n !== null),
2578
+ }
2579
+ } catch {
2580
+ // Ignore errors reading skill directories (e.g., directory doesn't exist)
2581
+ return { skillDir, skillNames: [] }
2582
+ }
2583
+ }),
2584
+ )
2585
+
2586
+ for (const { skillDir, skillNames } of perDirResults) {
2587
+ if (skillNames.length > 0) {
2588
+ attachments.push({
2589
+ type: 'dynamic_skill',
2590
+ skillDir,
2591
+ skillNames,
2592
+ displayPath: relative(getCwd(), skillDir),
2593
+ })
2594
+ }
2595
+ }
2596
+
2597
+ toolUseContext.dynamicSkillDirTriggers.clear()
2598
+ }
2599
+
2600
+ return attachments
2601
+ }
2602
+
2603
+ // Track which skills have been sent to avoid re-sending. Keyed by agentId
2604
+ // (empty string = main thread) so subagents get their own turn-0 listing —
2605
+ // without per-agent scoping, the main thread populating this Set would cause
2606
+ // every subagent's filterToBundledAndMcp result to dedup to empty.
2607
+ const sentSkillNames = new Map<string, Set<string>>()
2608
+
2609
+ // Called when the skill set genuinely changes (plugin reload, skill file
2610
+ // change on disk) so new skills get announced. NOT called on compact —
2611
+ // post-compact re-injection costs ~4K tokens/event for marginal benefit.
2612
+ export function resetSentSkillNames(): void {
2613
+ sentSkillNames.clear()
2614
+ suppressNext = false
2615
+ }
2616
+
2617
+ /**
2618
+ * Suppress the next skill-listing injection. Called by conversationRecovery
2619
+ * on --resume when a skill_listing attachment already exists in the
2620
+ * transcript.
2621
+ *
2622
+ * `sentSkillNames` is module-scope — process-local. Each `claude -p` spawn
2623
+ * starts with an empty Map, so without this every resume re-injects the
2624
+ * full ~600-token listing even though it's already in the conversation from
2625
+ * the prior process. Shows up on every --resume; particularly loud for
2626
+ * daemons that respawn frequently.
2627
+ *
2628
+ * Trade-off: skills added between sessions won't be announced until the
2629
+ * next non-resume session. Acceptable — skill_listing was never meant to
2630
+ * cover cross-process deltas, and the agent can still call them (they're
2631
+ * in the Skill tool's runtime registry regardless).
2632
+ */
2633
+ export function suppressNextSkillListing(): void {
2634
+ suppressNext = true
2635
+ }
2636
+ let suppressNext = false
2637
+
2638
+ // When skill-search is enabled and the filtered (bundled + MCP) listing exceeds
2639
+ // this count, fall back to bundled-only. Protects MCP-heavy users (100+ servers)
2640
+ // from truncation while keeping the turn-0 guarantee for typical setups.
2641
+ const FILTERED_LISTING_MAX = 30
2642
+
2643
+ /**
2644
+ * Filter skills to bundled (Anthropic-curated) + MCP (user-connected) only.
2645
+ * Used when skill-search is enabled to resolve the turn-0 gap for subagents:
2646
+ * these sources are small, intent-signaled, and won't hit the truncation budget.
2647
+ * User/project/plugin skills (the long tail — 200+) go through discovery instead.
2648
+ *
2649
+ * Falls back to bundled-only if bundled+mcp exceeds FILTERED_LISTING_MAX.
2650
+ */
2651
+ export function filterToBundledAndMcp(commands: Command[]): Command[] {
2652
+ const filtered = commands.filter(
2653
+ cmd => cmd.loadedFrom === 'bundled' || cmd.loadedFrom === 'mcp',
2654
+ )
2655
+ if (filtered.length > FILTERED_LISTING_MAX) {
2656
+ return filtered.filter(cmd => cmd.loadedFrom === 'bundled')
2657
+ }
2658
+ return filtered
2659
+ }
2660
+
2661
+ async function getSkillListingAttachments(
2662
+ toolUseContext: ToolUseContext,
2663
+ ): Promise<Attachment[]> {
2664
+ if (process.env.NODE_ENV === 'test') {
2665
+ return []
2666
+ }
2667
+
2668
+ // Skip skill listing for agents that don't have the Skill tool — they can't use skills directly.
2669
+ if (
2670
+ !toolUseContext.options.tools.some(t => toolMatchesName(t, SKILL_TOOL_NAME))
2671
+ ) {
2672
+ return []
2673
+ }
2674
+
2675
+ const cwd = getProjectRoot()
2676
+ const localCommands = await getSkillToolCommands(cwd)
2677
+ const mcpSkills = getMcpSkillCommands(
2678
+ toolUseContext.getAppState().mcp.commands,
2679
+ )
2680
+ let allCommands =
2681
+ mcpSkills.length > 0
2682
+ ? uniqBy([...localCommands, ...mcpSkills], 'name')
2683
+ : localCommands
2684
+
2685
+ // When skill search is active, filter to bundled + MCP instead of full
2686
+ // suppression. Resolves the turn-0 gap: main thread gets turn-0 discovery
2687
+ // via getTurnZeroSkillDiscovery (blocking), but subagents use the async
2688
+ // subagent_spawn signal (collected post-tools, visible turn 1). Bundled +
2689
+ // MCP are small and intent-signaled; user/project/plugin skills go through
2690
+ // discovery. feature() first for DCE — the property-access string leaks
2691
+ // otherwise even with ?. on null.
2692
+ if (
2693
+ feature('EXPERIMENTAL_SKILL_SEARCH') &&
2694
+ skillSearchModules?.featureCheck.isSkillSearchEnabled()
2695
+ ) {
2696
+ allCommands = filterToBundledAndMcp(allCommands)
2697
+ }
2698
+
2699
+ const agentKey = toolUseContext.agentId ?? ''
2700
+ let sent = sentSkillNames.get(agentKey)
2701
+ if (!sent) {
2702
+ sent = new Set()
2703
+ sentSkillNames.set(agentKey, sent)
2704
+ }
2705
+
2706
+ // Resume path: prior process already injected a listing; it's in the
2707
+ // transcript. Mark everything current as sent so only post-resume deltas
2708
+ // (skills loaded later via /reload-plugins etc) get announced.
2709
+ if (suppressNext) {
2710
+ suppressNext = false
2711
+ for (const cmd of allCommands) {
2712
+ sent.add(cmd.name)
2713
+ }
2714
+ return []
2715
+ }
2716
+
2717
+ // Find skills we haven't sent yet
2718
+ const newSkills = allCommands.filter(cmd => !sent.has(cmd.name))
2719
+
2720
+ if (newSkills.length === 0) {
2721
+ return []
2722
+ }
2723
+
2724
+ // If no skills have been sent yet, this is the initial batch
2725
+ const isInitial = sent.size === 0
2726
+
2727
+ // Mark as sent
2728
+ for (const cmd of newSkills) {
2729
+ sent.add(cmd.name)
2730
+ }
2731
+
2732
+ logForDebugging(
2733
+ `Sending ${newSkills.length} skills via attachment (${isInitial ? 'initial' : 'dynamic'}, ${sent.size} total sent)`,
2734
+ )
2735
+
2736
+ // Format within budget using existing logic
2737
+ const contextWindowTokens = getContextWindowForModel(
2738
+ toolUseContext.options.mainLoopModel,
2739
+ getSdkBetas(),
2740
+ )
2741
+ const content = formatCommandsWithinBudget(newSkills, contextWindowTokens)
2742
+
2743
+ return [
2744
+ {
2745
+ type: 'skill_listing',
2746
+ content,
2747
+ skillCount: newSkills.length,
2748
+ isInitial,
2749
+ },
2750
+ ]
2751
+ }
2752
+
2753
+ // getSkillDiscoveryAttachment moved to skillSearch/prefetch.ts as
2754
+ // getTurnZeroSkillDiscovery — keeps the 'skill_discovery' string literal inside
2755
+ // a feature-gated module so it doesn't leak into external builds.
2756
+
2757
+ export function extractAtMentionedFiles(content: string): string[] {
2758
+ // Extract filenames mentioned with @ symbol, including line range syntax: @file.txt#L10-20
2759
+ // Also supports quoted paths for files with spaces: @"my/file with spaces.txt"
2760
+ // Example: "foo bar @baz moo" would extract "baz"
2761
+ // Example: 'check @"my file.txt" please' would extract "my file.txt"
2762
+
2763
+ // Two patterns: quoted paths and regular paths
2764
+ const quotedAtMentionRegex = /(^|\s)@"([^"]+)"/g
2765
+ const regularAtMentionRegex = /(^|\s)@([^\s]+)\b/g
2766
+
2767
+ const quotedMatches: string[] = []
2768
+ const regularMatches: string[] = []
2769
+
2770
+ // Extract quoted mentions first (skip agent mentions like @"code-reviewer (agent)")
2771
+ let match
2772
+ while ((match = quotedAtMentionRegex.exec(content)) !== null) {
2773
+ if (match[2] && !match[2].endsWith(' (agent)')) {
2774
+ quotedMatches.push(match[2]) // The content inside quotes
2775
+ }
2776
+ }
2777
+
2778
+ // Extract regular mentions
2779
+ const regularMatchArray = content.match(regularAtMentionRegex) || []
2780
+ regularMatchArray.forEach(match => {
2781
+ const filename = match.slice(match.indexOf('@') + 1)
2782
+ // Don't include if it starts with a quote (already handled as quoted)
2783
+ if (!filename.startsWith('"')) {
2784
+ regularMatches.push(filename)
2785
+ }
2786
+ })
2787
+
2788
+ // Combine and deduplicate
2789
+ return uniq([...quotedMatches, ...regularMatches])
2790
+ }
2791
+
2792
+ export function extractMcpResourceMentions(content: string): string[] {
2793
+ // Extract MCP resources mentioned with @ symbol in format @server:uri
2794
+ // Example: "@server1:resource/path" would extract "server1:resource/path"
2795
+ const atMentionRegex = /(^|\s)@([^\s]+:[^\s]+)\b/g
2796
+ const matches = content.match(atMentionRegex) || []
2797
+
2798
+ // Remove the prefix (everything before @) from each match
2799
+ return uniq(matches.map(match => match.slice(match.indexOf('@') + 1)))
2800
+ }
2801
+
2802
+ export function extractAgentMentions(content: string): string[] {
2803
+ // Extract agent mentions in two formats:
2804
+ // 1. @agent-<agent-type> (legacy/manual typing)
2805
+ // Example: "@agent-code-elegance-refiner" → "agent-code-elegance-refiner"
2806
+ // 2. @"<agent-type> (agent)" (from autocomplete selection)
2807
+ // Example: '@"code-reviewer (agent)"' → "code-reviewer"
2808
+ // Supports colons, dots, and at-signs for plugin-scoped agents like "@agent-asana:project-status-updater"
2809
+ const results: string[] = []
2810
+
2811
+ // Match quoted format: @"<type> (agent)"
2812
+ const quotedAgentRegex = /(^|\s)@"([\w:.@-]+) \(agent\)"/g
2813
+ let match
2814
+ while ((match = quotedAgentRegex.exec(content)) !== null) {
2815
+ if (match[2]) {
2816
+ results.push(match[2])
2817
+ }
2818
+ }
2819
+
2820
+ // Match unquoted format: @agent-<type>
2821
+ const unquotedAgentRegex = /(^|\s)@(agent-[\w:.@-]+)/g
2822
+ const unquotedMatches = content.match(unquotedAgentRegex) || []
2823
+ for (const m of unquotedMatches) {
2824
+ results.push(m.slice(m.indexOf('@') + 1))
2825
+ }
2826
+
2827
+ return uniq(results)
2828
+ }
2829
+
2830
+ interface AtMentionedFileLines {
2831
+ filename: string
2832
+ lineStart?: number
2833
+ lineEnd?: number
2834
+ }
2835
+
2836
+ export function parseAtMentionedFileLines(
2837
+ mention: string,
2838
+ ): AtMentionedFileLines {
2839
+ // Parse mentions like "file.txt#L10-20", "file.txt#heading", or just "file.txt"
2840
+ // Supports line ranges (#L10, #L10-20) and strips non-line-range fragments (#heading)
2841
+ const match = mention.match(/^([^#]+)(?:#L(\d+)(?:-(\d+))?)?(?:#[^#]*)?$/)
2842
+
2843
+ if (!match) {
2844
+ return { filename: mention }
2845
+ }
2846
+
2847
+ const [, filename, lineStartStr, lineEndStr] = match
2848
+ const lineStart = lineStartStr ? parseInt(lineStartStr, 10) : undefined
2849
+ const lineEnd = lineEndStr ? parseInt(lineEndStr, 10) : lineStart
2850
+
2851
+ return { filename: filename ?? mention, lineStart, lineEnd }
2852
+ }
2853
+
2854
+ async function getDiagnosticAttachments(
2855
+ toolUseContext: ToolUseContext,
2856
+ ): Promise<Attachment[]> {
2857
+ // Diagnostics are only useful if the agent has the Bash tool to act on them
2858
+ if (
2859
+ !toolUseContext.options.tools.some(t => toolMatchesName(t, BASH_TOOL_NAME))
2860
+ ) {
2861
+ return []
2862
+ }
2863
+
2864
+ // Get new diagnostics from the tracker (IDE diagnostics via MCP)
2865
+ const newDiagnostics = await diagnosticTracker.getNewDiagnostics()
2866
+ if (newDiagnostics.length === 0) {
2867
+ return []
2868
+ }
2869
+
2870
+ return [
2871
+ {
2872
+ type: 'diagnostics',
2873
+ files: newDiagnostics,
2874
+ isNew: true,
2875
+ },
2876
+ ]
2877
+ }
2878
+
2879
+ /**
2880
+ * Get LSP diagnostic attachments from passive LSP servers.
2881
+ * Follows the AsyncHookRegistry pattern for consistent async attachment delivery.
2882
+ */
2883
+ async function getLSPDiagnosticAttachments(
2884
+ toolUseContext: ToolUseContext,
2885
+ ): Promise<Attachment[]> {
2886
+ // LSP diagnostics are only useful if the agent has the Bash tool to act on them
2887
+ if (
2888
+ !toolUseContext.options.tools.some(t => toolMatchesName(t, BASH_TOOL_NAME))
2889
+ ) {
2890
+ return []
2891
+ }
2892
+
2893
+ logForDebugging('LSP Diagnostics: getLSPDiagnosticAttachments called')
2894
+
2895
+ try {
2896
+ const diagnosticSets = checkForLSPDiagnostics()
2897
+
2898
+ if (diagnosticSets.length === 0) {
2899
+ return []
2900
+ }
2901
+
2902
+ logForDebugging(
2903
+ `LSP Diagnostics: Found ${diagnosticSets.length} pending diagnostic set(s)`,
2904
+ )
2905
+
2906
+ // Convert each diagnostic set to an attachment
2907
+ const attachments: Attachment[] = diagnosticSets.map(({ files }) => ({
2908
+ type: 'diagnostics' as const,
2909
+ files,
2910
+ isNew: true,
2911
+ }))
2912
+
2913
+ // Clear delivered diagnostics from registry to prevent memory leak
2914
+ // Follows same pattern as removeDeliveredAsyncHooks
2915
+ if (diagnosticSets.length > 0) {
2916
+ clearAllLSPDiagnostics()
2917
+ logForDebugging(
2918
+ `LSP Diagnostics: Cleared ${diagnosticSets.length} delivered diagnostic(s) from registry`,
2919
+ )
2920
+ }
2921
+
2922
+ logForDebugging(
2923
+ `LSP Diagnostics: Returning ${attachments.length} diagnostic attachment(s)`,
2924
+ )
2925
+
2926
+ return attachments
2927
+ } catch (error) {
2928
+ const err = toError(error)
2929
+ logError(
2930
+ new Error(`Failed to get LSP diagnostic attachments: ${err.message}`),
2931
+ )
2932
+ // Return empty array to allow other attachments to proceed
2933
+ return []
2934
+ }
2935
+ }
2936
+
2937
+ export async function* getAttachmentMessages(
2938
+ input: string | null,
2939
+ toolUseContext: ToolUseContext,
2940
+ ideSelection: IDESelection | null,
2941
+ queuedCommands: QueuedCommand[],
2942
+ messages?: Message[],
2943
+ querySource?: QuerySource,
2944
+ options?: { skipSkillDiscovery?: boolean },
2945
+ ): AsyncGenerator<AttachmentMessage, void> {
2946
+ // TODO: Compute this upstream
2947
+ const attachments = await getAttachments(
2948
+ input,
2949
+ toolUseContext,
2950
+ ideSelection,
2951
+ queuedCommands,
2952
+ messages,
2953
+ querySource,
2954
+ options,
2955
+ )
2956
+
2957
+ if (attachments.length === 0) {
2958
+ return
2959
+ }
2960
+
2961
+ logEvent('tengu_attachments', {
2962
+ attachment_types: attachments.map(
2963
+ _ => _.type,
2964
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2965
+ })
2966
+
2967
+ for (const attachment of attachments) {
2968
+ yield createAttachmentMessage(attachment)
2969
+ }
2970
+ }
2971
+
2972
+ /**
2973
+ * Generates a file attachment by reading a file with proper validation and truncation.
2974
+ * This is the core file reading logic shared between @-mentioned files and post-compact restoration.
2975
+ *
2976
+ * @param filename The absolute path to the file to read
2977
+ * @param toolUseContext The tool use context for calling FileReadTool
2978
+ * @param options Optional configuration for file reading
2979
+ * @returns A new_file attachment or null if the file couldn't be read
2980
+ */
2981
+ /**
2982
+ * Check if a PDF file should be represented as a lightweight reference
2983
+ * instead of being inlined. Returns a PDFReferenceAttachment for large PDFs
2984
+ * (more than PDF_AT_MENTION_INLINE_THRESHOLD pages), or null otherwise.
2985
+ */
2986
+ export async function tryGetPDFReference(
2987
+ filename: string,
2988
+ ): Promise<PDFReferenceAttachment | null> {
2989
+ const ext = parse(filename).ext.toLowerCase()
2990
+ if (!isPDFExtension(ext)) {
2991
+ return null
2992
+ }
2993
+ try {
2994
+ const [stats, pageCount] = await Promise.all([
2995
+ getFsImplementation().stat(filename),
2996
+ getPDFPageCount(filename),
2997
+ ])
2998
+ // Use page count if available, otherwise fall back to size heuristic (~100KB per page)
2999
+ const effectivePageCount = pageCount ?? Math.ceil(stats.size / (100 * 1024))
3000
+ if (effectivePageCount > PDF_AT_MENTION_INLINE_THRESHOLD) {
3001
+ logEvent('tengu_pdf_reference_attachment', {
3002
+ pageCount: effectivePageCount,
3003
+ fileSize: stats.size,
3004
+ hadPdfinfo: pageCount !== null,
3005
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
3006
+ return {
3007
+ type: 'pdf_reference',
3008
+ filename,
3009
+ pageCount: effectivePageCount,
3010
+ fileSize: stats.size,
3011
+ displayPath: relative(getCwd(), filename),
3012
+ }
3013
+ }
3014
+ } catch {
3015
+ // If we can't stat the file, return null to proceed with normal reading
3016
+ }
3017
+ return null
3018
+ }
3019
+
3020
+ export async function generateFileAttachment(
3021
+ filename: string,
3022
+ toolUseContext: ToolUseContext,
3023
+ successEventName: string,
3024
+ errorEventName: string,
3025
+ mode: 'compact' | 'at-mention',
3026
+ options?: {
3027
+ offset?: number
3028
+ limit?: number
3029
+ },
3030
+ ): Promise<
3031
+ | FileAttachment
3032
+ | CompactFileReferenceAttachment
3033
+ | PDFReferenceAttachment
3034
+ | AlreadyReadFileAttachment
3035
+ | null
3036
+ > {
3037
+ const { offset, limit } = options ?? {}
3038
+
3039
+ // Check if file has a deny rule configured
3040
+ const appState = toolUseContext.getAppState()
3041
+ if (isFileReadDenied(filename, appState.toolPermissionContext)) {
3042
+ return null
3043
+ }
3044
+
3045
+ // Check file size before attempting to read (skip for PDFs — they have their own size/page handling below)
3046
+ if (
3047
+ mode === 'at-mention' &&
3048
+ !isFileWithinReadSizeLimit(
3049
+ filename,
3050
+ getDefaultFileReadingLimits().maxSizeBytes,
3051
+ )
3052
+ ) {
3053
+ const ext = parse(filename).ext.toLowerCase()
3054
+ if (!isPDFExtension(ext)) {
3055
+ try {
3056
+ const stats = await getFsImplementation().stat(filename)
3057
+ logEvent('tengu_attachment_file_too_large', {
3058
+ size_bytes: stats.size,
3059
+ mode,
3060
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
3061
+ return null
3062
+ } catch {
3063
+ // If we can't stat the file, proceed with normal reading (will fail later if file doesn't exist)
3064
+ }
3065
+ }
3066
+ }
3067
+
3068
+ // For large PDFs on @ mention, return a lightweight reference instead of inlining
3069
+ if (mode === 'at-mention') {
3070
+ const pdfRef = await tryGetPDFReference(filename)
3071
+ if (pdfRef) {
3072
+ return pdfRef
3073
+ }
3074
+ }
3075
+
3076
+ // Check if file is already in context with latest version
3077
+ const existingFileState = toolUseContext.readFileState.get(filename)
3078
+ if (existingFileState && mode === 'at-mention') {
3079
+ try {
3080
+ // Check if the file has been modified since we last read it
3081
+ const mtimeMs = await getFileModificationTimeAsync(filename)
3082
+
3083
+ // Handle timestamp format inconsistency:
3084
+ // - FileReadTool stores Date.now() (current time when read)
3085
+ // - FileEdit/WriteTools store mtimeMs (file modification time)
3086
+ //
3087
+ // If timestamp > mtimeMs, it was stored by FileReadTool using Date.now()
3088
+ // In this case, we should not use the optimization since we can't reliably
3089
+ // compare modification times. Only use optimization when timestamp <= mtimeMs,
3090
+ // indicating it was stored by FileEdit/WriteTool with actual mtimeMs.
3091
+
3092
+ if (
3093
+ existingFileState.timestamp <= mtimeMs &&
3094
+ mtimeMs === existingFileState.timestamp
3095
+ ) {
3096
+ // File hasn't been modified, return already_read_file attachment
3097
+ // This tells the system the file is already in context and doesn't need to be sent to API
3098
+ logEvent(successEventName, {})
3099
+ return {
3100
+ type: 'already_read_file',
3101
+ filename,
3102
+ displayPath: relative(getCwd(), filename),
3103
+ content: {
3104
+ type: 'text',
3105
+ file: {
3106
+ filePath: filename,
3107
+ content: existingFileState.content,
3108
+ numLines: countCharInString(existingFileState.content, '\n') + 1,
3109
+ startLine: offset ?? 1,
3110
+ totalLines:
3111
+ countCharInString(existingFileState.content, '\n') + 1,
3112
+ },
3113
+ },
3114
+ }
3115
+ }
3116
+ } catch {
3117
+ // If we can't stat the file, proceed with normal reading
3118
+ }
3119
+ }
3120
+
3121
+ try {
3122
+ const fileInput = {
3123
+ file_path: filename,
3124
+ offset,
3125
+ limit,
3126
+ }
3127
+
3128
+ async function readTruncatedFile(): Promise<
3129
+ | FileAttachment
3130
+ | CompactFileReferenceAttachment
3131
+ | AlreadyReadFileAttachment
3132
+ | null
3133
+ > {
3134
+ if (mode === 'compact') {
3135
+ return {
3136
+ type: 'compact_file_reference',
3137
+ filename,
3138
+ displayPath: relative(getCwd(), filename),
3139
+ }
3140
+ }
3141
+
3142
+ // Check deny rules before reading truncated file
3143
+ const appState = toolUseContext.getAppState()
3144
+ if (isFileReadDenied(filename, appState.toolPermissionContext)) {
3145
+ return null
3146
+ }
3147
+
3148
+ try {
3149
+ // Read only the first MAX_LINES_TO_READ lines for files that are too large
3150
+ const truncatedInput = {
3151
+ file_path: filename,
3152
+ offset: offset ?? 1,
3153
+ limit: MAX_LINES_TO_READ,
3154
+ }
3155
+ const result = await FileReadTool.call(truncatedInput, toolUseContext)
3156
+ logEvent(successEventName, {})
3157
+
3158
+ return {
3159
+ type: 'file' as const,
3160
+ filename,
3161
+ content: result.data,
3162
+ truncated: true,
3163
+ displayPath: relative(getCwd(), filename),
3164
+ }
3165
+ } catch {
3166
+ logEvent(errorEventName, {})
3167
+ return null
3168
+ }
3169
+ }
3170
+
3171
+ // Validate file path is valid
3172
+ const isValid = await FileReadTool.validateInput(fileInput, toolUseContext)
3173
+ if (!isValid.result) {
3174
+ return null
3175
+ }
3176
+
3177
+ try {
3178
+ const result = await FileReadTool.call(fileInput, toolUseContext)
3179
+ logEvent(successEventName, {})
3180
+ return {
3181
+ type: 'file',
3182
+ filename,
3183
+ content: result.data,
3184
+ displayPath: relative(getCwd(), filename),
3185
+ }
3186
+ } catch (error) {
3187
+ if (
3188
+ error instanceof MaxFileReadTokenExceededError ||
3189
+ error instanceof FileTooLargeError
3190
+ ) {
3191
+ return await readTruncatedFile()
3192
+ }
3193
+ throw error
3194
+ }
3195
+ } catch {
3196
+ logEvent(errorEventName, {})
3197
+ return null
3198
+ }
3199
+ }
3200
+
3201
+ export function createAttachmentMessage(
3202
+ attachment: Attachment,
3203
+ ): AttachmentMessage {
3204
+ return {
3205
+ attachment,
3206
+ type: 'attachment',
3207
+ uuid: randomUUID(),
3208
+ timestamp: new Date().toISOString(),
3209
+ }
3210
+ }
3211
+
3212
+ function getTodoReminderTurnCounts(messages: Message[]): {
3213
+ turnsSinceLastTodoWrite: number
3214
+ turnsSinceLastReminder: number
3215
+ } {
3216
+ let lastTodoWriteIndex = -1
3217
+ let lastReminderIndex = -1
3218
+ let assistantTurnsSinceWrite = 0
3219
+ let assistantTurnsSinceReminder = 0
3220
+
3221
+ // Iterate backwards to find most recent events
3222
+ for (let i = messages.length - 1; i >= 0; i--) {
3223
+ const message = messages[i]
3224
+
3225
+ if (message?.type === 'assistant') {
3226
+ if (isThinkingMessage(message)) {
3227
+ // Skip thinking messages
3228
+ continue
3229
+ }
3230
+
3231
+ // Check for TodoWrite usage BEFORE incrementing counter
3232
+ // (we don't want to count the TodoWrite message itself as "1 turn since write")
3233
+ if (
3234
+ lastTodoWriteIndex === -1 &&
3235
+ 'message' in message &&
3236
+ Array.isArray(message.message?.content) &&
3237
+ message.message.content.some(
3238
+ block => block.type === 'tool_use' && block.name === 'TodoWrite',
3239
+ )
3240
+ ) {
3241
+ lastTodoWriteIndex = i
3242
+ }
3243
+
3244
+ // Count assistant turns before finding events
3245
+ if (lastTodoWriteIndex === -1) assistantTurnsSinceWrite++
3246
+ if (lastReminderIndex === -1) assistantTurnsSinceReminder++
3247
+ } else if (
3248
+ lastReminderIndex === -1 &&
3249
+ message?.type === 'attachment' &&
3250
+ message.attachment.type === 'todo_reminder'
3251
+ ) {
3252
+ lastReminderIndex = i
3253
+ }
3254
+
3255
+ if (lastTodoWriteIndex !== -1 && lastReminderIndex !== -1) {
3256
+ break
3257
+ }
3258
+ }
3259
+
3260
+ return {
3261
+ turnsSinceLastTodoWrite: assistantTurnsSinceWrite,
3262
+ turnsSinceLastReminder: assistantTurnsSinceReminder,
3263
+ }
3264
+ }
3265
+
3266
+ async function getTodoReminderAttachments(
3267
+ messages: Message[] | undefined,
3268
+ toolUseContext: ToolUseContext,
3269
+ ): Promise<Attachment[]> {
3270
+ // Skip if TodoWrite tool is not available
3271
+ if (
3272
+ !toolUseContext.options.tools.some(t =>
3273
+ toolMatchesName(t, TODO_WRITE_TOOL_NAME),
3274
+ )
3275
+ ) {
3276
+ return []
3277
+ }
3278
+
3279
+ // When SendUserMessage is in the toolkit, it's the primary communication
3280
+ // channel and the model is always told to use it (#20467). TodoWrite
3281
+ // becomes a side channel — nudging the model about it conflicts with the
3282
+ // brief workflow. The tool itself stays available; this only gates the
3283
+ // "you haven't used it in a while" nag.
3284
+ if (
3285
+ BRIEF_TOOL_NAME &&
3286
+ toolUseContext.options.tools.some(t => toolMatchesName(t, BRIEF_TOOL_NAME))
3287
+ ) {
3288
+ return []
3289
+ }
3290
+
3291
+ // Skip if no messages provided
3292
+ if (!messages || messages.length === 0) {
3293
+ return []
3294
+ }
3295
+
3296
+ const { turnsSinceLastTodoWrite, turnsSinceLastReminder } =
3297
+ getTodoReminderTurnCounts(messages)
3298
+
3299
+ // Check if we should show a reminder
3300
+ if (
3301
+ turnsSinceLastTodoWrite >= TODO_REMINDER_CONFIG.TURNS_SINCE_WRITE &&
3302
+ turnsSinceLastReminder >= TODO_REMINDER_CONFIG.TURNS_BETWEEN_REMINDERS
3303
+ ) {
3304
+ const todoKey = toolUseContext.agentId ?? getSessionId()
3305
+ const appState = toolUseContext.getAppState()
3306
+ const todos = appState.todos[todoKey] ?? []
3307
+ return [
3308
+ {
3309
+ type: 'todo_reminder',
3310
+ content: todos,
3311
+ itemCount: todos.length,
3312
+ },
3313
+ ]
3314
+ }
3315
+
3316
+ return []
3317
+ }
3318
+
3319
+ function getTaskReminderTurnCounts(messages: Message[]): {
3320
+ turnsSinceLastTaskManagement: number
3321
+ turnsSinceLastReminder: number
3322
+ } {
3323
+ let lastTaskManagementIndex = -1
3324
+ let lastReminderIndex = -1
3325
+ let assistantTurnsSinceTaskManagement = 0
3326
+ let assistantTurnsSinceReminder = 0
3327
+
3328
+ // Iterate backwards to find most recent events
3329
+ for (let i = messages.length - 1; i >= 0; i--) {
3330
+ const message = messages[i]
3331
+
3332
+ if (message?.type === 'assistant') {
3333
+ if (isThinkingMessage(message)) {
3334
+ // Skip thinking messages
3335
+ continue
3336
+ }
3337
+
3338
+ // Check for TaskCreate or TaskUpdate usage BEFORE incrementing counter
3339
+ if (
3340
+ lastTaskManagementIndex === -1 &&
3341
+ 'message' in message &&
3342
+ Array.isArray(message.message?.content) &&
3343
+ message.message.content.some(
3344
+ block =>
3345
+ block.type === 'tool_use' &&
3346
+ (block.name === TASK_CREATE_TOOL_NAME ||
3347
+ block.name === TASK_UPDATE_TOOL_NAME),
3348
+ )
3349
+ ) {
3350
+ lastTaskManagementIndex = i
3351
+ }
3352
+
3353
+ // Count assistant turns before finding events
3354
+ if (lastTaskManagementIndex === -1) assistantTurnsSinceTaskManagement++
3355
+ if (lastReminderIndex === -1) assistantTurnsSinceReminder++
3356
+ } else if (
3357
+ lastReminderIndex === -1 &&
3358
+ message?.type === 'attachment' &&
3359
+ message.attachment.type === 'task_reminder'
3360
+ ) {
3361
+ lastReminderIndex = i
3362
+ }
3363
+
3364
+ if (lastTaskManagementIndex !== -1 && lastReminderIndex !== -1) {
3365
+ break
3366
+ }
3367
+ }
3368
+
3369
+ return {
3370
+ turnsSinceLastTaskManagement: assistantTurnsSinceTaskManagement,
3371
+ turnsSinceLastReminder: assistantTurnsSinceReminder,
3372
+ }
3373
+ }
3374
+
3375
+ async function getTaskReminderAttachments(
3376
+ messages: Message[] | undefined,
3377
+ toolUseContext: ToolUseContext,
3378
+ ): Promise<Attachment[]> {
3379
+ if (!isTodoV2Enabled()) {
3380
+ return []
3381
+ }
3382
+
3383
+ // Skip for ant users
3384
+ if (process.env.USER_TYPE === 'ant') {
3385
+ return []
3386
+ }
3387
+
3388
+ // When SendUserMessage is in the toolkit, it's the primary communication
3389
+ // channel and the model is always told to use it (#20467). TaskUpdate
3390
+ // becomes a side channel — nudging the model about it conflicts with the
3391
+ // brief workflow. The tool itself stays available; this only gates the nag.
3392
+ if (
3393
+ BRIEF_TOOL_NAME &&
3394
+ toolUseContext.options.tools.some(t => toolMatchesName(t, BRIEF_TOOL_NAME))
3395
+ ) {
3396
+ return []
3397
+ }
3398
+
3399
+ // Skip if TaskUpdate tool is not available
3400
+ if (
3401
+ !toolUseContext.options.tools.some(t =>
3402
+ toolMatchesName(t, TASK_UPDATE_TOOL_NAME),
3403
+ )
3404
+ ) {
3405
+ return []
3406
+ }
3407
+
3408
+ // Skip if no messages provided
3409
+ if (!messages || messages.length === 0) {
3410
+ return []
3411
+ }
3412
+
3413
+ const { turnsSinceLastTaskManagement, turnsSinceLastReminder } =
3414
+ getTaskReminderTurnCounts(messages)
3415
+
3416
+ // Check if we should show a reminder
3417
+ if (
3418
+ turnsSinceLastTaskManagement >= TODO_REMINDER_CONFIG.TURNS_SINCE_WRITE &&
3419
+ turnsSinceLastReminder >= TODO_REMINDER_CONFIG.TURNS_BETWEEN_REMINDERS
3420
+ ) {
3421
+ const tasks = await listTasks(getTaskListId())
3422
+ return [
3423
+ {
3424
+ type: 'task_reminder',
3425
+ content: tasks,
3426
+ itemCount: tasks.length,
3427
+ },
3428
+ ]
3429
+ }
3430
+
3431
+ return []
3432
+ }
3433
+
3434
+ /**
3435
+ * Get attachments for all unified tasks using the Task framework.
3436
+ * Replaces the old getBackgroundShellAttachments, getBackgroundRemoteSessionAttachments,
3437
+ * and getAsyncAgentAttachments functions.
3438
+ */
3439
+ async function getUnifiedTaskAttachments(
3440
+ toolUseContext: ToolUseContext,
3441
+ ): Promise<Attachment[]> {
3442
+ const appState = toolUseContext.getAppState()
3443
+ const { attachments, updatedTaskOffsets, evictedTaskIds } =
3444
+ await generateTaskAttachments(appState)
3445
+
3446
+ applyTaskOffsetsAndEvictions(
3447
+ toolUseContext.setAppState,
3448
+ updatedTaskOffsets,
3449
+ evictedTaskIds,
3450
+ )
3451
+
3452
+ // Convert TaskAttachment to Attachment format
3453
+ return attachments.map(taskAttachment => ({
3454
+ type: 'task_status' as const,
3455
+ taskId: taskAttachment.taskId,
3456
+ taskType: taskAttachment.taskType,
3457
+ status: taskAttachment.status,
3458
+ description: taskAttachment.description,
3459
+ deltaSummary: taskAttachment.deltaSummary,
3460
+ outputFilePath: getTaskOutputPath(taskAttachment.taskId),
3461
+ }))
3462
+ }
3463
+
3464
+ async function getAsyncHookResponseAttachments(): Promise<Attachment[]> {
3465
+ const responses = await checkForAsyncHookResponses()
3466
+
3467
+ if (responses.length === 0) {
3468
+ return []
3469
+ }
3470
+
3471
+ logForDebugging(
3472
+ `Hooks: getAsyncHookResponseAttachments found ${responses.length} responses`,
3473
+ )
3474
+
3475
+ const attachments = responses.map(
3476
+ ({
3477
+ processId,
3478
+ response,
3479
+ hookName,
3480
+ hookEvent,
3481
+ toolName,
3482
+ pluginId,
3483
+ stdout,
3484
+ stderr,
3485
+ exitCode,
3486
+ }) => {
3487
+ logForDebugging(
3488
+ `Hooks: Creating attachment for ${processId} (${hookName}): ${jsonStringify(response)}`,
3489
+ )
3490
+ return {
3491
+ type: 'async_hook_response' as const,
3492
+ processId,
3493
+ hookName,
3494
+ hookEvent,
3495
+ toolName,
3496
+ response,
3497
+ stdout,
3498
+ stderr,
3499
+ exitCode,
3500
+ }
3501
+ },
3502
+ )
3503
+
3504
+ // Remove delivered hooks from registry to prevent re-processing
3505
+ if (responses.length > 0) {
3506
+ const processIds = responses.map(r => r.processId)
3507
+ removeDeliveredAsyncHooks(processIds)
3508
+ logForDebugging(
3509
+ `Hooks: Removed ${processIds.length} delivered hooks from registry`,
3510
+ )
3511
+ }
3512
+
3513
+ logForDebugging(
3514
+ `Hooks: getAsyncHookResponseAttachments found ${attachments.length} attachments`,
3515
+ )
3516
+
3517
+ return attachments
3518
+ }
3519
+
3520
+ /**
3521
+ * Get teammate mailbox attachments for agent swarm communication
3522
+ * Teammates are independent Claude Code sessions running in parallel (swarms),
3523
+ * not parent-child subagent relationships.
3524
+ *
3525
+ * This function checks two sources for messages:
3526
+ * 1. File-based mailbox (for messages that arrived between polls)
3527
+ * 2. AppState.inbox (for messages queued mid-turn by useInboxPoller)
3528
+ *
3529
+ * Messages from AppState.inbox are delivered mid-turn as attachments,
3530
+ * allowing teammates to receive messages without waiting for the turn to end.
3531
+ */
3532
+ async function getTeammateMailboxAttachments(
3533
+ toolUseContext: ToolUseContext,
3534
+ ): Promise<Attachment[]> {
3535
+ if (!isAgentSwarmsEnabled()) {
3536
+ return []
3537
+ }
3538
+ if (process.env.USER_TYPE !== 'ant') {
3539
+ return []
3540
+ }
3541
+
3542
+ // Get AppState early to check for team lead status
3543
+ const appState = toolUseContext.getAppState()
3544
+
3545
+ // Use agent name from helper (checks AsyncLocalStorage, then dynamicTeamContext)
3546
+ const envAgentName = getAgentName()
3547
+
3548
+ // Get team name (checks AsyncLocalStorage, dynamicTeamContext, then AppState)
3549
+ const teamName = getTeamName(appState.teamContext)
3550
+
3551
+ // Check if we're the team lead (uses shared logic from swarm utils)
3552
+ const teamLeadStatus = isTeamLead(appState.teamContext)
3553
+
3554
+ // Check if viewing a teammate's transcript (for in-process teammates)
3555
+ const viewedTeammate = getViewedTeammateTask(appState)
3556
+
3557
+ // Resolve agent name based on who we're VIEWING:
3558
+ // - If viewing a teammate, use THEIR name (to read from their mailbox)
3559
+ // - Otherwise use env var if set, or leader's name if we're the team lead
3560
+ let agentName = viewedTeammate?.identity.agentName ?? envAgentName
3561
+ if (!agentName && teamLeadStatus && appState.teamContext) {
3562
+ const leadAgentId = appState.teamContext.leadAgentId
3563
+ // Look up the lead's name from agents map (not the UUID)
3564
+ agentName = appState.teamContext.teammates[leadAgentId]?.name || 'team-lead'
3565
+ }
3566
+
3567
+ logForDebugging(
3568
+ `[SwarmMailbox] getTeammateMailboxAttachments called: envAgentName=${envAgentName}, isTeamLead=${teamLeadStatus}, resolved agentName=${agentName}, teamName=${teamName}`,
3569
+ )
3570
+
3571
+ // Only check inbox if running as an agent in a swarm or team lead
3572
+ if (!agentName) {
3573
+ logForDebugging(
3574
+ `[SwarmMailbox] Not checking inbox - not in a swarm or team lead`,
3575
+ )
3576
+ return []
3577
+ }
3578
+
3579
+ logForDebugging(
3580
+ `[SwarmMailbox] Checking inbox for agent="${agentName}" team="${teamName || 'default'}"`,
3581
+ )
3582
+
3583
+ // Check mailbox for unread messages (routes to in-process or file-based)
3584
+ // Filter out structured protocol messages (permission requests/responses, shutdown
3585
+ // messages, etc.) — these must be left unread for useInboxPoller to route to their
3586
+ // proper handlers (workerPermissions queue, sandbox queue, etc.). Without filtering,
3587
+ // attachment generation races with InboxPoller: whichever reads first marks all
3588
+ // messages as read, and if attachments wins, protocol messages get bundled as raw
3589
+ // LLM context text instead of being routed to their UI handlers.
3590
+ const allUnreadMessages = await readUnreadMessages(agentName, teamName)
3591
+ const unreadMessages = allUnreadMessages.filter(
3592
+ m => !isStructuredProtocolMessage(m.text),
3593
+ )
3594
+ logForDebugging(
3595
+ `[MailboxBridge] Found ${allUnreadMessages.length} unread message(s) for "${agentName}" (${allUnreadMessages.length - unreadMessages.length} structured protocol messages filtered out)`,
3596
+ )
3597
+
3598
+ // Also check AppState.inbox for pending messages (queued mid-turn by useInboxPoller)
3599
+ // IMPORTANT: appState.inbox contains messages FROM teammates TO the leader.
3600
+ // Only show these when viewing the leader's transcript (not a teammate's).
3601
+ // When viewing a teammate, their messages come from the file-based mailbox above.
3602
+ // In-process teammates share AppState with the leader — appState.inbox contains
3603
+ // the LEADER's queued messages, not the teammate's. Skip it to prevent leakage
3604
+ // (including self-echo from broadcasts). Teammates receive messages exclusively
3605
+ // through their file-based mailbox + waitForNextPromptOrShutdown.
3606
+ // Note: viewedTeammate was already computed above for agentName resolution
3607
+ const pendingInboxMessages =
3608
+ viewedTeammate || isInProcessTeammate()
3609
+ ? [] // Viewing teammate or running as in-process teammate - don't show leader's inbox
3610
+ : appState.inbox.messages.filter(m => m.status === 'pending')
3611
+ logForDebugging(
3612
+ `[SwarmMailbox] Found ${pendingInboxMessages.length} pending message(s) in AppState.inbox`,
3613
+ )
3614
+
3615
+ // Combine both sources of messages WITH DEDUPLICATION
3616
+ // The same message could exist in both file mailbox and AppState.inbox due to race conditions:
3617
+ // 1. getTeammateMailboxAttachments reads file -> finds message M
3618
+ // 2. InboxPoller reads same file -> queues M in AppState.inbox
3619
+ // 3. getTeammateMailboxAttachments reads AppState -> finds M again
3620
+ // We deduplicate using from+timestamp+text prefix as the key
3621
+ const seen = new Set<string>()
3622
+ let allMessages: Array<{
3623
+ from: string
3624
+ text: string
3625
+ timestamp: string
3626
+ color?: string
3627
+ summary?: string
3628
+ }> = []
3629
+
3630
+ for (const m of [...unreadMessages, ...pendingInboxMessages]) {
3631
+ const key = `${m.from}|${m.timestamp}|${m.text.slice(0, 100)}`
3632
+ if (!seen.has(key)) {
3633
+ seen.add(key)
3634
+ allMessages.push({
3635
+ from: m.from,
3636
+ text: m.text,
3637
+ timestamp: m.timestamp,
3638
+ color: m.color,
3639
+ summary: m.summary,
3640
+ })
3641
+ }
3642
+ }
3643
+
3644
+ // Collapse multiple idle notifications per agent — keep only the latest.
3645
+ // Single pass to parse, then filter without re-parsing.
3646
+ const idleAgentByIndex = new Map<number, string>()
3647
+ const latestIdleByAgent = new Map<string, number>()
3648
+ for (let i = 0; i < allMessages.length; i++) {
3649
+ const idle = isIdleNotification(allMessages[i]!.text)
3650
+ if (idle) {
3651
+ idleAgentByIndex.set(i, idle.from)
3652
+ latestIdleByAgent.set(idle.from, i)
3653
+ }
3654
+ }
3655
+ if (idleAgentByIndex.size > latestIdleByAgent.size) {
3656
+ const beforeCount = allMessages.length
3657
+ allMessages = allMessages.filter((_m, i) => {
3658
+ const agent = idleAgentByIndex.get(i)
3659
+ if (agent === undefined) return true
3660
+ return latestIdleByAgent.get(agent) === i
3661
+ })
3662
+ logForDebugging(
3663
+ `[SwarmMailbox] Collapsed ${beforeCount - allMessages.length} duplicate idle notification(s)`,
3664
+ )
3665
+ }
3666
+
3667
+ if (allMessages.length === 0) {
3668
+ logForDebugging(`[SwarmMailbox] No messages to deliver, returning empty`)
3669
+ return []
3670
+ }
3671
+
3672
+ logForDebugging(
3673
+ `[SwarmMailbox] Returning ${allMessages.length} message(s) as attachment for "${agentName}" (${unreadMessages.length} from file, ${pendingInboxMessages.length} from AppState, after dedup)`,
3674
+ )
3675
+
3676
+ // Build the attachment BEFORE marking messages as processed
3677
+ // This prevents message loss if any operation below fails
3678
+ const attachment: Attachment[] = [
3679
+ {
3680
+ type: 'teammate_mailbox',
3681
+ messages: allMessages,
3682
+ },
3683
+ ]
3684
+
3685
+ // Mark only non-structured mailbox messages as read after attachment is built.
3686
+ // Structured protocol messages stay unread for useInboxPoller to handle.
3687
+ if (unreadMessages.length > 0) {
3688
+ await markMessagesAsReadByPredicate(
3689
+ agentName,
3690
+ m => !isStructuredProtocolMessage(m.text),
3691
+ teamName,
3692
+ )
3693
+ logForDebugging(
3694
+ `[MailboxBridge] marked ${unreadMessages.length} non-structured message(s) as read for agent="${agentName}" team="${teamName || 'default'}"`,
3695
+ )
3696
+ }
3697
+
3698
+ // Process shutdown_approved messages - remove teammates from team file
3699
+ // This mirrors what useInboxPoller does in interactive mode (lines 546-606)
3700
+ // In -p mode, useInboxPoller doesn't run, so we must handle this here
3701
+ if (teamLeadStatus && teamName) {
3702
+ for (const m of allMessages) {
3703
+ const shutdownApproval = isShutdownApproved(m.text)
3704
+ if (shutdownApproval) {
3705
+ const teammateToRemove = shutdownApproval.from
3706
+ logForDebugging(
3707
+ `[SwarmMailbox] Processing shutdown_approved from ${teammateToRemove}`,
3708
+ )
3709
+
3710
+ // Find the teammate ID by name
3711
+ const teammateId = appState.teamContext?.teammates
3712
+ ? Object.entries(appState.teamContext.teammates).find(
3713
+ ([, t]) => t.name === teammateToRemove,
3714
+ )?.[0]
3715
+ : undefined
3716
+
3717
+ if (teammateId) {
3718
+ // Remove from team file
3719
+ removeTeammateFromTeamFile(teamName, {
3720
+ agentId: teammateId,
3721
+ name: teammateToRemove,
3722
+ })
3723
+ logForDebugging(
3724
+ `[SwarmMailbox] Removed ${teammateToRemove} from team file`,
3725
+ )
3726
+
3727
+ // Unassign tasks owned by this teammate
3728
+ await unassignTeammateTasks(
3729
+ teamName,
3730
+ teammateId,
3731
+ teammateToRemove,
3732
+ 'shutdown',
3733
+ )
3734
+
3735
+ // Remove from teamContext in AppState
3736
+ toolUseContext.setAppState(prev => {
3737
+ if (!prev.teamContext?.teammates) return prev
3738
+ if (!(teammateId in prev.teamContext.teammates)) return prev
3739
+ const { [teammateId]: _, ...remainingTeammates } =
3740
+ prev.teamContext.teammates
3741
+ return {
3742
+ ...prev,
3743
+ teamContext: {
3744
+ ...prev.teamContext,
3745
+ teammates: remainingTeammates,
3746
+ },
3747
+ }
3748
+ })
3749
+ }
3750
+ }
3751
+ }
3752
+ }
3753
+
3754
+ // Mark AppState inbox messages as processed LAST, after attachment is built
3755
+ // This ensures messages aren't lost if earlier operations fail
3756
+ if (pendingInboxMessages.length > 0) {
3757
+ const pendingIds = new Set(pendingInboxMessages.map(m => m.id))
3758
+ toolUseContext.setAppState(prev => ({
3759
+ ...prev,
3760
+ inbox: {
3761
+ messages: prev.inbox.messages.map(m =>
3762
+ pendingIds.has(m.id) ? { ...m, status: 'processed' as const } : m,
3763
+ ),
3764
+ },
3765
+ }))
3766
+ }
3767
+
3768
+ return attachment
3769
+ }
3770
+
3771
+ /**
3772
+ * Get team context attachment for teammates in a swarm.
3773
+ * Only injected on the first turn to provide team coordination instructions.
3774
+ */
3775
+ function getTeamContextAttachment(messages: Message[]): Attachment[] {
3776
+ const teamName = getTeamName()
3777
+ const agentId = getAgentId()
3778
+ const agentName = getAgentName()
3779
+
3780
+ // Only inject for teammates (not team lead or non-team sessions)
3781
+ if (!teamName || !agentId) {
3782
+ return []
3783
+ }
3784
+
3785
+ // Only inject on first turn - check if there are no assistant messages yet
3786
+ const hasAssistantMessage = messages.some(m => m.type === 'assistant')
3787
+ if (hasAssistantMessage) {
3788
+ return []
3789
+ }
3790
+
3791
+ const configDir = getClaudeConfigHomeDir()
3792
+ const teamConfigPath = `${configDir}/teams/${teamName}/config.json`
3793
+ const taskListPath = `${configDir}/tasks/${teamName}/`
3794
+
3795
+ return [
3796
+ {
3797
+ type: 'team_context',
3798
+ agentId,
3799
+ agentName: agentName || agentId,
3800
+ teamName,
3801
+ teamConfigPath,
3802
+ taskListPath,
3803
+ },
3804
+ ]
3805
+ }
3806
+
3807
+ function getTokenUsageAttachment(
3808
+ messages: Message[],
3809
+ model: string,
3810
+ ): Attachment[] {
3811
+ if (!isEnvTruthy(process.env.CLAUDE_CODE_ENABLE_TOKEN_USAGE_ATTACHMENT)) {
3812
+ return []
3813
+ }
3814
+
3815
+ const contextWindow = getEffectiveContextWindowSize(model)
3816
+ const usedTokens = tokenCountFromLastAPIResponse(messages)
3817
+
3818
+ return [
3819
+ {
3820
+ type: 'token_usage',
3821
+ used: usedTokens,
3822
+ total: contextWindow,
3823
+ remaining: contextWindow - usedTokens,
3824
+ },
3825
+ ]
3826
+ }
3827
+
3828
+ function getOutputTokenUsageAttachment(): Attachment[] {
3829
+ if (feature('TOKEN_BUDGET')) {
3830
+ const budget = getCurrentTurnTokenBudget()
3831
+ if (budget === null || budget <= 0) {
3832
+ return []
3833
+ }
3834
+ return [
3835
+ {
3836
+ type: 'output_token_usage',
3837
+ turn: getTurnOutputTokens(),
3838
+ session: getTotalOutputTokens(),
3839
+ budget,
3840
+ },
3841
+ ]
3842
+ }
3843
+ return []
3844
+ }
3845
+
3846
+ function getMaxBudgetUsdAttachment(maxBudgetUsd?: number): Attachment[] {
3847
+ if (maxBudgetUsd === undefined) {
3848
+ return []
3849
+ }
3850
+
3851
+ const usedCost = getTotalCostUSD()
3852
+ const remainingBudget = maxBudgetUsd - usedCost
3853
+
3854
+ return [
3855
+ {
3856
+ type: 'budget_usd',
3857
+ used: usedCost,
3858
+ total: maxBudgetUsd,
3859
+ remaining: remainingBudget,
3860
+ },
3861
+ ]
3862
+ }
3863
+
3864
+ /**
3865
+ * Count human turns since plan mode exit (plan_mode_exit attachment).
3866
+ * Returns 0 if no plan_mode_exit attachment found.
3867
+ *
3868
+ * tool_result messages are type:'user' without isMeta, so filter by
3869
+ * toolUseResult to avoid counting them — otherwise the 10-turn reminder
3870
+ * interval fires every ~10 tool calls instead of ~10 human turns.
3871
+ */
3872
+ export function getVerifyPlanReminderTurnCount(messages: Message[]): number {
3873
+ let turnCount = 0
3874
+ for (let i = messages.length - 1; i >= 0; i--) {
3875
+ const message = messages[i]
3876
+ if (message && isHumanTurn(message)) {
3877
+ turnCount++
3878
+ }
3879
+ // Stop counting at plan_mode_exit attachment (marks when implementation started)
3880
+ if (
3881
+ message?.type === 'attachment' &&
3882
+ message.attachment.type === 'plan_mode_exit'
3883
+ ) {
3884
+ return turnCount
3885
+ }
3886
+ }
3887
+ // No plan_mode_exit found
3888
+ return 0
3889
+ }
3890
+
3891
+ /**
3892
+ * Get verify plan reminder attachment if the model hasn't called VerifyPlanExecution yet.
3893
+ */
3894
+ async function getVerifyPlanReminderAttachment(
3895
+ messages: Message[] | undefined,
3896
+ toolUseContext: ToolUseContext,
3897
+ ): Promise<Attachment[]> {
3898
+ if (
3899
+ process.env.USER_TYPE !== 'ant' ||
3900
+ !isEnvTruthy(process.env.CLAUDE_CODE_VERIFY_PLAN)
3901
+ ) {
3902
+ return []
3903
+ }
3904
+
3905
+ const appState = toolUseContext.getAppState()
3906
+ const pending = appState.pendingPlanVerification
3907
+
3908
+ // Only remind if plan exists and verification not started or completed
3909
+ if (
3910
+ !pending ||
3911
+ pending.verificationStarted ||
3912
+ pending.verificationCompleted
3913
+ ) {
3914
+ return []
3915
+ }
3916
+
3917
+ // Only remind every N turns
3918
+ if (messages && messages.length > 0) {
3919
+ const turnCount = getVerifyPlanReminderTurnCount(messages)
3920
+ if (
3921
+ turnCount === 0 ||
3922
+ turnCount % VERIFY_PLAN_REMINDER_CONFIG.TURNS_BETWEEN_REMINDERS !== 0
3923
+ ) {
3924
+ return []
3925
+ }
3926
+ }
3927
+
3928
+ return [{ type: 'verify_plan_reminder' }]
3929
+ }
3930
+
3931
+ export function getCompactionReminderAttachment(
3932
+ messages: Message[],
3933
+ model: string,
3934
+ ): Attachment[] {
3935
+ if (!getFeatureValue_CACHED_MAY_BE_STALE('tengu_marble_fox', false)) {
3936
+ return []
3937
+ }
3938
+
3939
+ if (!isAutoCompactEnabled()) {
3940
+ return []
3941
+ }
3942
+
3943
+ const contextWindow = getContextWindowForModel(model, getSdkBetas())
3944
+ if (contextWindow < 1_000_000) {
3945
+ return []
3946
+ }
3947
+
3948
+ const effectiveWindow = getEffectiveContextWindowSize(model)
3949
+ const usedTokens = tokenCountWithEstimation(messages)
3950
+ if (usedTokens < effectiveWindow * 0.25) {
3951
+ return []
3952
+ }
3953
+
3954
+ return [{ type: 'compaction_reminder' }]
3955
+ }
3956
+
3957
+ /**
3958
+ * Context-efficiency nudge. Injected after every N tokens of growth without
3959
+ * a snip. Pacing is handled entirely by shouldNudgeForSnips — the 10k
3960
+ * interval resets on prior nudges, snip markers, snip boundaries, and
3961
+ * compact boundaries.
3962
+ */
3963
+ export function getContextEfficiencyAttachment(
3964
+ messages: Message[],
3965
+ ): Attachment[] {
3966
+ if (!feature('HISTORY_SNIP')) {
3967
+ return []
3968
+ }
3969
+ // Gate must match SnipTool.isEnabled() — don't nudge toward a tool that
3970
+ // isn't in the tool list. Lazy require keeps this file snip-string-free.
3971
+ const { isSnipRuntimeEnabled, shouldNudgeForSnips } =
3972
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
3973
+ require('../services/compact/snipCompact.js') as typeof import('../services/compact/snipCompact.js')
3974
+ if (!isSnipRuntimeEnabled()) {
3975
+ return []
3976
+ }
3977
+
3978
+ if (!shouldNudgeForSnips(messages)) {
3979
+ return []
3980
+ }
3981
+
3982
+ return [{ type: 'context_efficiency' }]
3983
+ }
3984
+
3985
+
3986
+ function isFileReadDenied(
3987
+ filePath: string,
3988
+ toolPermissionContext: ToolPermissionContext,
3989
+ ): boolean {
3990
+ const denyRule = matchingRuleForInput(
3991
+ filePath,
3992
+ toolPermissionContext,
3993
+ 'read',
3994
+ 'deny',
3995
+ )
3996
+ return denyRule !== null
3997
+ }