mumucc 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 (2090) hide show
  1. package/README.md +337 -0
  2. package/bin/mumucc +58 -0
  3. package/bin/opencc +50 -0
  4. package/bun.lock +1625 -0
  5. package/defaults/settings.json +67 -0
  6. package/package.json +149 -0
  7. package/scripts/patch-node-modules.sh +100 -0
  8. package/scripts/teammate-wrapper.sh +12 -0
  9. package/shims/bun-bundle.ts +43 -0
  10. package/shims/globals.ts +14 -0
  11. package/src/QueryEngine.ts +1295 -0
  12. package/src/Task.ts +125 -0
  13. package/src/Tool.ts +792 -0
  14. package/src/assistant/AssistantSessionChooser.tsx +5 -0
  15. package/src/assistant/gate.ts +2 -0
  16. package/src/assistant/index.ts +2 -0
  17. package/src/assistant/sessionDiscovery.ts +4 -0
  18. package/src/assistant/sessionHistory.ts +87 -0
  19. package/src/bootstrap/state.ts +1758 -0
  20. package/src/bridge/bridgeApi.ts +539 -0
  21. package/src/bridge/bridgeConfig.ts +48 -0
  22. package/src/bridge/bridgeDebug.ts +135 -0
  23. package/src/bridge/bridgeEnabled.ts +202 -0
  24. package/src/bridge/bridgeMain.ts +2999 -0
  25. package/src/bridge/bridgeMessaging.ts +461 -0
  26. package/src/bridge/bridgePermissionCallbacks.ts +43 -0
  27. package/src/bridge/bridgePointer.ts +210 -0
  28. package/src/bridge/bridgeStatusUtil.ts +163 -0
  29. package/src/bridge/bridgeUI.ts +530 -0
  30. package/src/bridge/capacityWake.ts +56 -0
  31. package/src/bridge/codeSessionApi.ts +168 -0
  32. package/src/bridge/createSession.ts +384 -0
  33. package/src/bridge/debugUtils.ts +141 -0
  34. package/src/bridge/envLessBridgeConfig.ts +165 -0
  35. package/src/bridge/flushGate.ts +71 -0
  36. package/src/bridge/inboundAttachments.ts +175 -0
  37. package/src/bridge/inboundMessages.ts +80 -0
  38. package/src/bridge/initReplBridge.ts +569 -0
  39. package/src/bridge/jwtUtils.ts +256 -0
  40. package/src/bridge/peerSessions.ts +2 -0
  41. package/src/bridge/pollConfig.ts +110 -0
  42. package/src/bridge/pollConfigDefaults.ts +82 -0
  43. package/src/bridge/remoteBridgeCore.ts +1008 -0
  44. package/src/bridge/replBridge.ts +2406 -0
  45. package/src/bridge/replBridgeHandle.ts +36 -0
  46. package/src/bridge/replBridgeTransport.ts +370 -0
  47. package/src/bridge/sessionIdCompat.ts +57 -0
  48. package/src/bridge/sessionRunner.ts +550 -0
  49. package/src/bridge/trustedDevice.ts +210 -0
  50. package/src/bridge/types.ts +262 -0
  51. package/src/bridge/webhookSanitizer.ts +2 -0
  52. package/src/bridge/workSecret.ts +127 -0
  53. package/src/buddy/CompanionSprite.tsx +371 -0
  54. package/src/buddy/companion.ts +133 -0
  55. package/src/buddy/prompt.ts +36 -0
  56. package/src/buddy/sprites.ts +514 -0
  57. package/src/buddy/types.ts +148 -0
  58. package/src/buddy/useBuddyNotification.tsx +98 -0
  59. package/src/cachedMicrocompact.ts +3 -0
  60. package/src/cli/exit.ts +31 -0
  61. package/src/cli/handlers/agents.ts +70 -0
  62. package/src/cli/handlers/auth.ts +330 -0
  63. package/src/cli/handlers/autoMode.ts +170 -0
  64. package/src/cli/handlers/mcp.tsx +362 -0
  65. package/src/cli/handlers/plugins.ts +878 -0
  66. package/src/cli/handlers/util.tsx +110 -0
  67. package/src/cli/ndjsonSafeStringify.ts +32 -0
  68. package/src/cli/print.ts +5594 -0
  69. package/src/cli/remoteIO.ts +255 -0
  70. package/src/cli/structuredIO.ts +859 -0
  71. package/src/cli/transports/HybridTransport.ts +282 -0
  72. package/src/cli/transports/SSETransport.ts +711 -0
  73. package/src/cli/transports/SerialBatchEventUploader.ts +275 -0
  74. package/src/cli/transports/Transport.ts +4 -0
  75. package/src/cli/transports/WebSocketTransport.ts +800 -0
  76. package/src/cli/transports/WorkerStateUploader.ts +131 -0
  77. package/src/cli/transports/ccrClient.ts +998 -0
  78. package/src/cli/transports/transportUtils.ts +45 -0
  79. package/src/cli/update.ts +422 -0
  80. package/src/commands/add-dir/add-dir.tsx +126 -0
  81. package/src/commands/add-dir/index.ts +11 -0
  82. package/src/commands/add-dir/validation.ts +110 -0
  83. package/src/commands/advisor.ts +109 -0
  84. package/src/commands/agents/agents.tsx +12 -0
  85. package/src/commands/agents/index.ts +10 -0
  86. package/src/commands/agents-platform/index.ts +2 -0
  87. package/src/commands/ant-trace/index.js +1 -0
  88. package/src/commands/assistant/assistant.ts +3 -0
  89. package/src/commands/assistant/index.ts +2 -0
  90. package/src/commands/autofix-pr/index.js +1 -0
  91. package/src/commands/backfill-sessions/index.js +1 -0
  92. package/src/commands/branch/branch.ts +296 -0
  93. package/src/commands/branch/index.ts +14 -0
  94. package/src/commands/break-cache/index.js +1 -0
  95. package/src/commands/bridge/bridge.tsx +509 -0
  96. package/src/commands/bridge/index.ts +26 -0
  97. package/src/commands/bridge-kick.ts +200 -0
  98. package/src/commands/brief.ts +130 -0
  99. package/src/commands/btw/btw.tsx +243 -0
  100. package/src/commands/btw/index.ts +13 -0
  101. package/src/commands/buddy/index.ts +2 -0
  102. package/src/commands/bughunter/index.js +1 -0
  103. package/src/commands/chrome/chrome.tsx +285 -0
  104. package/src/commands/chrome/index.ts +13 -0
  105. package/src/commands/clear/caches.ts +144 -0
  106. package/src/commands/clear/clear/caches.ts +2 -0
  107. package/src/commands/clear/clear/conversation.ts +2 -0
  108. package/src/commands/clear/clear.ts +7 -0
  109. package/src/commands/clear/conversation.ts +251 -0
  110. package/src/commands/clear/index.ts +19 -0
  111. package/src/commands/color/color.ts +93 -0
  112. package/src/commands/color/index.ts +16 -0
  113. package/src/commands/commit-push-pr.ts +158 -0
  114. package/src/commands/commit.ts +92 -0
  115. package/src/commands/compact/compact.ts +287 -0
  116. package/src/commands/compact/index.ts +15 -0
  117. package/src/commands/config/config.tsx +7 -0
  118. package/src/commands/config/index.ts +11 -0
  119. package/src/commands/context/context-noninteractive.ts +325 -0
  120. package/src/commands/context/context.tsx +64 -0
  121. package/src/commands/context/index.ts +24 -0
  122. package/src/commands/copy/copy.tsx +371 -0
  123. package/src/commands/copy/index.ts +15 -0
  124. package/src/commands/cost/cost.ts +24 -0
  125. package/src/commands/cost/index.ts +23 -0
  126. package/src/commands/createMovedToPluginCommand.ts +65 -0
  127. package/src/commands/ctx_viz/index.js +1 -0
  128. package/src/commands/debug-tool-call/index.js +1 -0
  129. package/src/commands/desktop/desktop.tsx +9 -0
  130. package/src/commands/desktop/index.ts +26 -0
  131. package/src/commands/diff/diff.tsx +9 -0
  132. package/src/commands/diff/index.ts +8 -0
  133. package/src/commands/doctor/doctor.tsx +7 -0
  134. package/src/commands/doctor/index.ts +12 -0
  135. package/src/commands/effort/effort.tsx +183 -0
  136. package/src/commands/effort/index.ts +13 -0
  137. package/src/commands/env/index.js +1 -0
  138. package/src/commands/exit/exit.tsx +33 -0
  139. package/src/commands/exit/index.ts +12 -0
  140. package/src/commands/export/export.tsx +91 -0
  141. package/src/commands/export/index.ts +11 -0
  142. package/src/commands/extra-usage/extra-usage-core.ts +118 -0
  143. package/src/commands/extra-usage/extra-usage-noninteractive.ts +16 -0
  144. package/src/commands/extra-usage/extra-usage.tsx +17 -0
  145. package/src/commands/extra-usage/index.ts +31 -0
  146. package/src/commands/fast/fast.tsx +269 -0
  147. package/src/commands/fast/index.ts +26 -0
  148. package/src/commands/feedback/feedback.tsx +25 -0
  149. package/src/commands/feedback/index.ts +26 -0
  150. package/src/commands/files/files.ts +19 -0
  151. package/src/commands/files/index.ts +12 -0
  152. package/src/commands/force-snip.ts +2 -0
  153. package/src/commands/fork/index.ts +2 -0
  154. package/src/commands/good-claude/index.js +1 -0
  155. package/src/commands/heapdump/heapdump.ts +17 -0
  156. package/src/commands/heapdump/index.ts +12 -0
  157. package/src/commands/help/help.tsx +11 -0
  158. package/src/commands/help/index.ts +10 -0
  159. package/src/commands/hooks/hooks.tsx +13 -0
  160. package/src/commands/hooks/index.ts +11 -0
  161. package/src/commands/ide/ide.tsx +646 -0
  162. package/src/commands/ide/index.ts +11 -0
  163. package/src/commands/init-verifiers.ts +262 -0
  164. package/src/commands/init.ts +256 -0
  165. package/src/commands/insights.ts +3200 -0
  166. package/src/commands/install-github-app/ApiKeyStep.tsx +231 -0
  167. package/src/commands/install-github-app/CheckExistingSecretStep.tsx +190 -0
  168. package/src/commands/install-github-app/CheckGitHubStep.tsx +15 -0
  169. package/src/commands/install-github-app/ChooseRepoStep.tsx +211 -0
  170. package/src/commands/install-github-app/CreatingStep.tsx +65 -0
  171. package/src/commands/install-github-app/ErrorStep.tsx +85 -0
  172. package/src/commands/install-github-app/ExistingWorkflowStep.tsx +103 -0
  173. package/src/commands/install-github-app/InstallAppStep.tsx +94 -0
  174. package/src/commands/install-github-app/OAuthFlowStep.tsx +276 -0
  175. package/src/commands/install-github-app/SuccessStep.tsx +96 -0
  176. package/src/commands/install-github-app/WarningsStep.tsx +73 -0
  177. package/src/commands/install-github-app/index.ts +13 -0
  178. package/src/commands/install-github-app/install-github-app.tsx +587 -0
  179. package/src/commands/install-github-app/setupGitHubActions.ts +325 -0
  180. package/src/commands/install-github-app/types.ts +6 -0
  181. package/src/commands/install-slack-app/index.ts +12 -0
  182. package/src/commands/install-slack-app/install-slack-app.ts +30 -0
  183. package/src/commands/install.tsx +300 -0
  184. package/src/commands/issue/index.js +1 -0
  185. package/src/commands/keybindings/index.ts +13 -0
  186. package/src/commands/keybindings/keybindings.ts +53 -0
  187. package/src/commands/login/CodexLogin.tsx +424 -0
  188. package/src/commands/login/CustomProviderLogin.tsx +298 -0
  189. package/src/commands/login/ThirdPartyLogin.tsx +149 -0
  190. package/src/commands/login/index.ts +14 -0
  191. package/src/commands/login/login.tsx +219 -0
  192. package/src/commands/logout/index.ts +10 -0
  193. package/src/commands/logout/logout.tsx +82 -0
  194. package/src/commands/mcp/addCommand.ts +280 -0
  195. package/src/commands/mcp/index.ts +12 -0
  196. package/src/commands/mcp/mcp.tsx +85 -0
  197. package/src/commands/mcp/xaaIdpCommand.ts +266 -0
  198. package/src/commands/memory/index.ts +10 -0
  199. package/src/commands/memory/memory.tsx +90 -0
  200. package/src/commands/mobile/index.ts +11 -0
  201. package/src/commands/mobile/mobile.tsx +274 -0
  202. package/src/commands/mock-limits/index.js +1 -0
  203. package/src/commands/model/index.ts +17 -0
  204. package/src/commands/model/model.tsx +304 -0
  205. package/src/commands/oauth-refresh/index.js +1 -0
  206. package/src/commands/onboarding/index.js +1 -0
  207. package/src/commands/output-style/index.ts +11 -0
  208. package/src/commands/output-style/output-style.tsx +7 -0
  209. package/src/commands/passes/index.ts +22 -0
  210. package/src/commands/passes/passes.tsx +24 -0
  211. package/src/commands/peers/index.ts +2 -0
  212. package/src/commands/perf-issue/index.js +1 -0
  213. package/src/commands/permissions/index.ts +11 -0
  214. package/src/commands/permissions/permissions.tsx +10 -0
  215. package/src/commands/plan/index.ts +11 -0
  216. package/src/commands/plan/plan.tsx +122 -0
  217. package/src/commands/plugin/AddMarketplace.tsx +162 -0
  218. package/src/commands/plugin/BrowseMarketplace.tsx +802 -0
  219. package/src/commands/plugin/DiscoverPlugins.tsx +781 -0
  220. package/src/commands/plugin/ManageMarketplaces.tsx +838 -0
  221. package/src/commands/plugin/ManagePlugins.tsx +2215 -0
  222. package/src/commands/plugin/PluginErrors.tsx +124 -0
  223. package/src/commands/plugin/PluginOptionsDialog.tsx +357 -0
  224. package/src/commands/plugin/PluginOptionsFlow.tsx +135 -0
  225. package/src/commands/plugin/PluginSettings.tsx +1072 -0
  226. package/src/commands/plugin/PluginTrustWarning.tsx +32 -0
  227. package/src/commands/plugin/UnifiedInstalledCell.tsx +565 -0
  228. package/src/commands/plugin/ValidatePlugin.tsx +98 -0
  229. package/src/commands/plugin/index.tsx +11 -0
  230. package/src/commands/plugin/parseArgs.ts +103 -0
  231. package/src/commands/plugin/plugin.tsx +7 -0
  232. package/src/commands/plugin/pluginDetailsHelpers.tsx +117 -0
  233. package/src/commands/plugin/types.ts +5 -0
  234. package/src/commands/plugin/unifiedTypes.ts +4 -0
  235. package/src/commands/plugin/usePagination.ts +171 -0
  236. package/src/commands/pr_comments/index.ts +50 -0
  237. package/src/commands/privacy-settings/index.ts +14 -0
  238. package/src/commands/privacy-settings/privacy-settings.tsx +58 -0
  239. package/src/commands/proactive.ts +2 -0
  240. package/src/commands/rate-limit-options/index.ts +19 -0
  241. package/src/commands/rate-limit-options/rate-limit-options.tsx +210 -0
  242. package/src/commands/release-notes/index.ts +11 -0
  243. package/src/commands/release-notes/release-notes.ts +50 -0
  244. package/src/commands/reload-plugins/index.ts +18 -0
  245. package/src/commands/reload-plugins/reload-plugins.ts +61 -0
  246. package/src/commands/remote-env/index.ts +15 -0
  247. package/src/commands/remote-env/remote-env.tsx +7 -0
  248. package/src/commands/remote-setup/api.ts +182 -0
  249. package/src/commands/remote-setup/index.ts +20 -0
  250. package/src/commands/remote-setup/remote-setup.tsx +187 -0
  251. package/src/commands/remoteControlServer/index.ts +2 -0
  252. package/src/commands/rename/generateSessionName.ts +67 -0
  253. package/src/commands/rename/index.ts +12 -0
  254. package/src/commands/rename/rename.ts +87 -0
  255. package/src/commands/reset-limits/index.js +4 -0
  256. package/src/commands/resume/index.ts +12 -0
  257. package/src/commands/resume/resume.tsx +275 -0
  258. package/src/commands/review/UltrareviewOverageDialog.tsx +96 -0
  259. package/src/commands/review/reviewRemote.ts +316 -0
  260. package/src/commands/review/ultrareviewCommand.tsx +58 -0
  261. package/src/commands/review/ultrareviewEnabled.ts +14 -0
  262. package/src/commands/review.ts +57 -0
  263. package/src/commands/rewind/index.ts +13 -0
  264. package/src/commands/rewind/rewind.ts +13 -0
  265. package/src/commands/sandbox-toggle/index.ts +50 -0
  266. package/src/commands/sandbox-toggle/sandbox-toggle.tsx +83 -0
  267. package/src/commands/security-review.ts +243 -0
  268. package/src/commands/session/index.ts +16 -0
  269. package/src/commands/session/session.tsx +140 -0
  270. package/src/commands/share/index.js +1 -0
  271. package/src/commands/skills/index.ts +10 -0
  272. package/src/commands/skills/skills.tsx +8 -0
  273. package/src/commands/stats/index.ts +10 -0
  274. package/src/commands/stats/stats.tsx +7 -0
  275. package/src/commands/status/index.ts +12 -0
  276. package/src/commands/status/status.tsx +8 -0
  277. package/src/commands/statusline.tsx +24 -0
  278. package/src/commands/stickers/index.ts +11 -0
  279. package/src/commands/stickers/stickers.ts +16 -0
  280. package/src/commands/subscribe-pr.ts +2 -0
  281. package/src/commands/summary/index.js +1 -0
  282. package/src/commands/tag/index.ts +12 -0
  283. package/src/commands/tag/tag.tsx +215 -0
  284. package/src/commands/tasks/index.ts +11 -0
  285. package/src/commands/tasks/tasks.tsx +8 -0
  286. package/src/commands/teleport/index.js +1 -0
  287. package/src/commands/terminalSetup/index.ts +23 -0
  288. package/src/commands/terminalSetup/terminalSetup.tsx +531 -0
  289. package/src/commands/theme/index.ts +10 -0
  290. package/src/commands/theme/theme.tsx +57 -0
  291. package/src/commands/thinkback/index.ts +13 -0
  292. package/src/commands/thinkback/thinkback.tsx +554 -0
  293. package/src/commands/thinkback-play/index.ts +17 -0
  294. package/src/commands/thinkback-play/thinkback-play.ts +43 -0
  295. package/src/commands/torch.ts +2 -0
  296. package/src/commands/ultraplan.tsx +471 -0
  297. package/src/commands/upgrade/index.ts +16 -0
  298. package/src/commands/upgrade/upgrade.tsx +38 -0
  299. package/src/commands/usage/index.ts +9 -0
  300. package/src/commands/usage/usage.tsx +7 -0
  301. package/src/commands/version.ts +22 -0
  302. package/src/commands/vim/index.ts +11 -0
  303. package/src/commands/vim/vim.ts +38 -0
  304. package/src/commands/voice/index.ts +20 -0
  305. package/src/commands/voice/voice.ts +150 -0
  306. package/src/commands/workflows/index.ts +2 -0
  307. package/src/commands.ts +754 -0
  308. package/src/components/AgentProgressLine.tsx +136 -0
  309. package/src/components/AntModelSwitchCallout.ts +2 -0
  310. package/src/components/App.tsx +56 -0
  311. package/src/components/ApproveApiKey.tsx +123 -0
  312. package/src/components/AutoModeOptInDialog.tsx +142 -0
  313. package/src/components/AutoUpdater.tsx +198 -0
  314. package/src/components/AutoUpdaterWrapper.tsx +91 -0
  315. package/src/components/AwsAuthStatusBox.tsx +82 -0
  316. package/src/components/BaseTextInput.tsx +136 -0
  317. package/src/components/BashModeProgress.tsx +56 -0
  318. package/src/components/BridgeDialog.tsx +401 -0
  319. package/src/components/BypassPermissionsModeDialog.tsx +87 -0
  320. package/src/components/ChannelDowngradeDialog.tsx +102 -0
  321. package/src/components/ClaudeCodeHint/PluginHintMenu.tsx +78 -0
  322. package/src/components/ClaudeInChromeOnboarding.tsx +121 -0
  323. package/src/components/ClaudeMdExternalIncludesDialog.tsx +137 -0
  324. package/src/components/ClickableImageRef.tsx +73 -0
  325. package/src/components/CompactSummary.tsx +118 -0
  326. package/src/components/ConfigurableShortcutHint.tsx +57 -0
  327. package/src/components/ConsoleOAuthFlow.tsx +631 -0
  328. package/src/components/ContextSuggestions.tsx +47 -0
  329. package/src/components/ContextVisualization.tsx +489 -0
  330. package/src/components/CoordinatorAgentStatus.tsx +273 -0
  331. package/src/components/CostThresholdDialog.tsx +50 -0
  332. package/src/components/CtrlOToExpand.tsx +51 -0
  333. package/src/components/CustomSelect/SelectMulti.tsx +213 -0
  334. package/src/components/CustomSelect/index.ts +3 -0
  335. package/src/components/CustomSelect/option-map.ts +50 -0
  336. package/src/components/CustomSelect/select-input-option.tsx +488 -0
  337. package/src/components/CustomSelect/select-option.tsx +68 -0
  338. package/src/components/CustomSelect/select.tsx +690 -0
  339. package/src/components/CustomSelect/use-multi-select-state.ts +414 -0
  340. package/src/components/CustomSelect/use-select-input.ts +287 -0
  341. package/src/components/CustomSelect/use-select-navigation.ts +653 -0
  342. package/src/components/CustomSelect/use-select-state.ts +157 -0
  343. package/src/components/DesktopHandoff.tsx +193 -0
  344. package/src/components/DesktopUpsell/DesktopUpsellStartup.tsx +171 -0
  345. package/src/components/DevBar.tsx +49 -0
  346. package/src/components/DevChannelsDialog.tsx +105 -0
  347. package/src/components/DiagnosticsDisplay.tsx +95 -0
  348. package/src/components/EffortCallout.tsx +265 -0
  349. package/src/components/EffortIndicator.ts +42 -0
  350. package/src/components/ExitFlow.tsx +48 -0
  351. package/src/components/ExportDialog.tsx +128 -0
  352. package/src/components/FallbackToolUseErrorMessage.tsx +116 -0
  353. package/src/components/FallbackToolUseRejectedMessage.tsx +16 -0
  354. package/src/components/FastIcon.tsx +46 -0
  355. package/src/components/Feedback.tsx +592 -0
  356. package/src/components/FeedbackSurvey/FeedbackSurvey.tsx +174 -0
  357. package/src/components/FeedbackSurvey/FeedbackSurveyView.tsx +108 -0
  358. package/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx +88 -0
  359. package/src/components/FeedbackSurvey/submitTranscriptShare.ts +112 -0
  360. package/src/components/FeedbackSurvey/useDebouncedDigitInput.ts +82 -0
  361. package/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +296 -0
  362. package/src/components/FeedbackSurvey/useFrustrationDetection.ts +2 -0
  363. package/src/components/FeedbackSurvey/useMemorySurvey.tsx +213 -0
  364. package/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +206 -0
  365. package/src/components/FeedbackSurvey/useSurveyState.tsx +100 -0
  366. package/src/components/FeedbackSurvey/utils.ts +5 -0
  367. package/src/components/FileEditToolDiff.tsx +181 -0
  368. package/src/components/FileEditToolUpdatedMessage.tsx +124 -0
  369. package/src/components/FileEditToolUseRejectedMessage.tsx +170 -0
  370. package/src/components/FilePathLink.tsx +43 -0
  371. package/src/components/FullscreenLayout.tsx +637 -0
  372. package/src/components/GlobalSearchDialog.tsx +343 -0
  373. package/src/components/HelpV2/Commands.tsx +82 -0
  374. package/src/components/HelpV2/General.tsx +23 -0
  375. package/src/components/HelpV2/HelpV2.tsx +184 -0
  376. package/src/components/HighlightedCode/Fallback.tsx +193 -0
  377. package/src/components/HighlightedCode.tsx +190 -0
  378. package/src/components/HistorySearchDialog.tsx +118 -0
  379. package/src/components/IdeAutoConnectDialog.tsx +154 -0
  380. package/src/components/IdeOnboardingDialog.tsx +167 -0
  381. package/src/components/IdeStatusIndicator.tsx +58 -0
  382. package/src/components/IdleReturnDialog.tsx +118 -0
  383. package/src/components/InterruptedByUser.tsx +15 -0
  384. package/src/components/InvalidConfigDialog.tsx +156 -0
  385. package/src/components/InvalidSettingsDialog.tsx +89 -0
  386. package/src/components/KeybindingWarnings.tsx +55 -0
  387. package/src/components/LanguagePicker.tsx +86 -0
  388. package/src/components/LogSelector.tsx +1575 -0
  389. package/src/components/LogoV2/AnimatedAsterisk.tsx +50 -0
  390. package/src/components/LogoV2/AnimatedClawd.tsx +124 -0
  391. package/src/components/LogoV2/ChannelsNotice.tsx +266 -0
  392. package/src/components/LogoV2/Clawd.tsx +240 -0
  393. package/src/components/LogoV2/CondensedLogo.tsx +161 -0
  394. package/src/components/LogoV2/EmergencyTip.tsx +58 -0
  395. package/src/components/LogoV2/Feed.tsx +112 -0
  396. package/src/components/LogoV2/FeedColumn.tsx +59 -0
  397. package/src/components/LogoV2/GuestPassesUpsell.tsx +70 -0
  398. package/src/components/LogoV2/LogoV2.tsx +543 -0
  399. package/src/components/LogoV2/Opus1mMergeNotice.tsx +55 -0
  400. package/src/components/LogoV2/OverageCreditUpsell.tsx +166 -0
  401. package/src/components/LogoV2/VoiceModeNotice.tsx +68 -0
  402. package/src/components/LogoV2/WelcomeV2.tsx +433 -0
  403. package/src/components/LogoV2/feedConfigs.tsx +92 -0
  404. package/src/components/LspRecommendation/LspRecommendationMenu.tsx +88 -0
  405. package/src/components/MCPServerApprovalDialog.tsx +115 -0
  406. package/src/components/MCPServerDesktopImportDialog.tsx +203 -0
  407. package/src/components/MCPServerDialogCopy.tsx +15 -0
  408. package/src/components/MCPServerMultiselectDialog.tsx +133 -0
  409. package/src/components/ManagedSettingsSecurityDialog/ManagedSettingsSecurityDialog.tsx +149 -0
  410. package/src/components/ManagedSettingsSecurityDialog/utils.ts +144 -0
  411. package/src/components/Markdown.tsx +236 -0
  412. package/src/components/MarkdownTable.tsx +322 -0
  413. package/src/components/MemoryUsageIndicator.tsx +37 -0
  414. package/src/components/Message.tsx +627 -0
  415. package/src/components/MessageModel.tsx +43 -0
  416. package/src/components/MessageResponse.tsx +78 -0
  417. package/src/components/MessageRow.tsx +383 -0
  418. package/src/components/MessageSelector.tsx +831 -0
  419. package/src/components/MessageTimestamp.tsx +63 -0
  420. package/src/components/Messages.tsx +834 -0
  421. package/src/components/ModelPicker.tsx +448 -0
  422. package/src/components/NativeAutoUpdater.tsx +193 -0
  423. package/src/components/NotebookEditToolUseRejectedMessage.tsx +92 -0
  424. package/src/components/OffscreenFreeze.tsx +44 -0
  425. package/src/components/Onboarding.tsx +244 -0
  426. package/src/components/OutputStylePicker.tsx +112 -0
  427. package/src/components/PackageManagerAutoUpdater.tsx +104 -0
  428. package/src/components/Passes/Passes.tsx +184 -0
  429. package/src/components/PrBadge.tsx +97 -0
  430. package/src/components/PressEnterToContinue.tsx +15 -0
  431. package/src/components/PromptInput/HistorySearchInput.tsx +51 -0
  432. package/src/components/PromptInput/IssueFlagBanner.tsx +12 -0
  433. package/src/components/PromptInput/Notifications.tsx +332 -0
  434. package/src/components/PromptInput/PromptInput.tsx +2339 -0
  435. package/src/components/PromptInput/PromptInputFooter.tsx +191 -0
  436. package/src/components/PromptInput/PromptInputFooterLeftSide.tsx +517 -0
  437. package/src/components/PromptInput/PromptInputFooterSuggestions.tsx +293 -0
  438. package/src/components/PromptInput/PromptInputHelpMenu.tsx +358 -0
  439. package/src/components/PromptInput/PromptInputModeIndicator.tsx +93 -0
  440. package/src/components/PromptInput/PromptInputQueuedCommands.tsx +117 -0
  441. package/src/components/PromptInput/PromptInputStashNotice.tsx +25 -0
  442. package/src/components/PromptInput/SandboxPromptFooterHint.tsx +64 -0
  443. package/src/components/PromptInput/ShimmeredInput.tsx +143 -0
  444. package/src/components/PromptInput/VoiceIndicator.tsx +137 -0
  445. package/src/components/PromptInput/inputModes.ts +33 -0
  446. package/src/components/PromptInput/inputPaste.ts +90 -0
  447. package/src/components/PromptInput/useMaybeTruncateInput.ts +58 -0
  448. package/src/components/PromptInput/usePromptInputPlaceholder.ts +76 -0
  449. package/src/components/PromptInput/useShowFastIconHint.ts +31 -0
  450. package/src/components/PromptInput/useSwarmBanner.ts +155 -0
  451. package/src/components/PromptInput/utils.ts +60 -0
  452. package/src/components/QuickOpenDialog.tsx +244 -0
  453. package/src/components/RemoteCallout.tsx +76 -0
  454. package/src/components/RemoteEnvironmentDialog.tsx +340 -0
  455. package/src/components/ResumeTask.tsx +268 -0
  456. package/src/components/SandboxViolationExpandedView.tsx +99 -0
  457. package/src/components/ScrollKeybindingHandler.tsx +1012 -0
  458. package/src/components/SearchBox.tsx +72 -0
  459. package/src/components/SentryErrorBoundary.ts +28 -0
  460. package/src/components/SessionBackgroundHint.tsx +108 -0
  461. package/src/components/SessionPreview.tsx +194 -0
  462. package/src/components/Settings/Config.tsx +1822 -0
  463. package/src/components/Settings/Settings.tsx +137 -0
  464. package/src/components/Settings/Status.tsx +241 -0
  465. package/src/components/Settings/Usage.tsx +377 -0
  466. package/src/components/ShowInIDEPrompt.tsx +170 -0
  467. package/src/components/SkillImprovementSurvey.tsx +152 -0
  468. package/src/components/Spinner/FlashingChar.tsx +61 -0
  469. package/src/components/Spinner/GlimmerMessage.tsx +328 -0
  470. package/src/components/Spinner/ShimmerChar.tsx +36 -0
  471. package/src/components/Spinner/SpinnerAnimationRow.tsx +265 -0
  472. package/src/components/Spinner/SpinnerGlyph.tsx +80 -0
  473. package/src/components/Spinner/TeammateSpinnerLine.tsx +233 -0
  474. package/src/components/Spinner/TeammateSpinnerTree.tsx +272 -0
  475. package/src/components/Spinner/index.ts +10 -0
  476. package/src/components/Spinner/teammateSelectHint.ts +1 -0
  477. package/src/components/Spinner/types.ts +5 -0
  478. package/src/components/Spinner/useShimmerAnimation.ts +31 -0
  479. package/src/components/Spinner/useStalledAnimation.ts +75 -0
  480. package/src/components/Spinner/utils.ts +84 -0
  481. package/src/components/Spinner.tsx +562 -0
  482. package/src/components/Stats.tsx +1228 -0
  483. package/src/components/StatusLine.tsx +324 -0
  484. package/src/components/StatusNotices.tsx +55 -0
  485. package/src/components/StructuredDiff/Fallback.tsx +487 -0
  486. package/src/components/StructuredDiff/colorDiff.ts +37 -0
  487. package/src/components/StructuredDiff.tsx +190 -0
  488. package/src/components/StructuredDiffList.tsx +30 -0
  489. package/src/components/TagTabs.tsx +139 -0
  490. package/src/components/TaskListV2.tsx +378 -0
  491. package/src/components/TeammateViewHeader.tsx +82 -0
  492. package/src/components/TeleportError.tsx +189 -0
  493. package/src/components/TeleportProgress.tsx +140 -0
  494. package/src/components/TeleportRepoMismatchDialog.tsx +104 -0
  495. package/src/components/TeleportResumeWrapper.tsx +167 -0
  496. package/src/components/TeleportStash.tsx +116 -0
  497. package/src/components/TextInput.tsx +124 -0
  498. package/src/components/ThemePicker.tsx +333 -0
  499. package/src/components/ThinkingToggle.tsx +153 -0
  500. package/src/components/TokenWarning.tsx +179 -0
  501. package/src/components/ToolUseLoader.tsx +42 -0
  502. package/src/components/TrustDialog/TrustDialog.tsx +290 -0
  503. package/src/components/TrustDialog/utils.ts +245 -0
  504. package/src/components/UndercoverAutoCallout.ts +2 -0
  505. package/src/components/ValidationErrorsList.tsx +148 -0
  506. package/src/components/VimTextInput.tsx +140 -0
  507. package/src/components/VirtualMessageList.tsx +1082 -0
  508. package/src/components/WorkflowMultiselectDialog.tsx +128 -0
  509. package/src/components/WorktreeExitDialog.tsx +231 -0
  510. package/src/components/agents/AgentDetail.tsx +220 -0
  511. package/src/components/agents/AgentEditor.tsx +178 -0
  512. package/src/components/agents/AgentNavigationFooter.tsx +26 -0
  513. package/src/components/agents/AgentsList.tsx +440 -0
  514. package/src/components/agents/AgentsMenu.tsx +800 -0
  515. package/src/components/agents/ColorPicker.tsx +112 -0
  516. package/src/components/agents/ModelSelector.tsx +68 -0
  517. package/src/components/agents/SnapshotUpdateDialog.tsx +5 -0
  518. package/src/components/agents/ToolSelector.tsx +562 -0
  519. package/src/components/agents/agentFileUtils.ts +272 -0
  520. package/src/components/agents/generateAgent.ts +197 -0
  521. package/src/components/agents/new-agent-creation/CreateAgentWizard.tsx +97 -0
  522. package/src/components/agents/new-agent-creation/types.ts +4 -0
  523. package/src/components/agents/new-agent-creation/wizard-steps/ColorStep.tsx +84 -0
  524. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStep.tsx +378 -0
  525. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStepWrapper.tsx +74 -0
  526. package/src/components/agents/new-agent-creation/wizard-steps/DescriptionStep.tsx +123 -0
  527. package/src/components/agents/new-agent-creation/wizard-steps/GenerateStep.tsx +143 -0
  528. package/src/components/agents/new-agent-creation/wizard-steps/LocationStep.tsx +80 -0
  529. package/src/components/agents/new-agent-creation/wizard-steps/MemoryStep.tsx +113 -0
  530. package/src/components/agents/new-agent-creation/wizard-steps/MethodStep.tsx +80 -0
  531. package/src/components/agents/new-agent-creation/wizard-steps/ModelStep.tsx +52 -0
  532. package/src/components/agents/new-agent-creation/wizard-steps/PromptStep.tsx +128 -0
  533. package/src/components/agents/new-agent-creation/wizard-steps/ToolsStep.tsx +61 -0
  534. package/src/components/agents/new-agent-creation/wizard-steps/TypeStep.tsx +103 -0
  535. package/src/components/agents/types.ts +27 -0
  536. package/src/components/agents/utils.ts +18 -0
  537. package/src/components/agents/validateAgent.ts +109 -0
  538. package/src/components/design-system/Byline.tsx +77 -0
  539. package/src/components/design-system/Dialog.tsx +138 -0
  540. package/src/components/design-system/Divider.tsx +149 -0
  541. package/src/components/design-system/FuzzyPicker.tsx +312 -0
  542. package/src/components/design-system/KeyboardShortcutHint.tsx +81 -0
  543. package/src/components/design-system/ListItem.tsx +244 -0
  544. package/src/components/design-system/LoadingState.tsx +94 -0
  545. package/src/components/design-system/Pane.tsx +77 -0
  546. package/src/components/design-system/ProgressBar.tsx +86 -0
  547. package/src/components/design-system/Ratchet.tsx +80 -0
  548. package/src/components/design-system/StatusIcon.tsx +95 -0
  549. package/src/components/design-system/Tabs.tsx +340 -0
  550. package/src/components/design-system/ThemeProvider.tsx +170 -0
  551. package/src/components/design-system/ThemedBox.tsx +156 -0
  552. package/src/components/design-system/ThemedText.tsx +124 -0
  553. package/src/components/design-system/color.ts +30 -0
  554. package/src/components/diff/DiffDetailView.tsx +281 -0
  555. package/src/components/diff/DiffDialog.tsx +383 -0
  556. package/src/components/diff/DiffFileList.tsx +292 -0
  557. package/src/components/grove/Grove.tsx +463 -0
  558. package/src/components/hooks/HooksConfigMenu.tsx +578 -0
  559. package/src/components/hooks/PromptDialog.tsx +90 -0
  560. package/src/components/hooks/SelectEventMode.tsx +127 -0
  561. package/src/components/hooks/SelectHookMode.tsx +112 -0
  562. package/src/components/hooks/SelectMatcherMode.tsx +144 -0
  563. package/src/components/hooks/ViewHookMode.tsx +199 -0
  564. package/src/components/mcp/CapabilitiesSection.tsx +61 -0
  565. package/src/components/mcp/ElicitationDialog.tsx +1169 -0
  566. package/src/components/mcp/MCPAgentServerMenu.tsx +183 -0
  567. package/src/components/mcp/MCPListPanel.tsx +504 -0
  568. package/src/components/mcp/MCPReconnect.tsx +167 -0
  569. package/src/components/mcp/MCPRemoteServerMenu.tsx +649 -0
  570. package/src/components/mcp/MCPSettings.tsx +398 -0
  571. package/src/components/mcp/MCPStdioServerMenu.tsx +177 -0
  572. package/src/components/mcp/MCPToolDetailView.tsx +212 -0
  573. package/src/components/mcp/MCPToolListView.tsx +141 -0
  574. package/src/components/mcp/McpParsingWarnings.tsx +213 -0
  575. package/src/components/mcp/index.ts +9 -0
  576. package/src/components/mcp/types.ts +10 -0
  577. package/src/components/mcp/utils/reconnectHelpers.tsx +49 -0
  578. package/src/components/memory/MemoryFileSelector.tsx +438 -0
  579. package/src/components/memory/MemoryUpdateNotification.tsx +45 -0
  580. package/src/components/messageActions.tsx +450 -0
  581. package/src/components/messages/AdvisorMessage.tsx +158 -0
  582. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +31 -0
  583. package/src/components/messages/AssistantTextMessage.tsx +270 -0
  584. package/src/components/messages/AssistantThinkingMessage.tsx +86 -0
  585. package/src/components/messages/AssistantToolUseMessage.tsx +368 -0
  586. package/src/components/messages/AttachmentMessage.tsx +536 -0
  587. package/src/components/messages/CollapsedReadSearchContent.tsx +484 -0
  588. package/src/components/messages/CompactBoundaryMessage.tsx +18 -0
  589. package/src/components/messages/GroupedToolUseContent.tsx +58 -0
  590. package/src/components/messages/HighlightedThinkingText.tsx +162 -0
  591. package/src/components/messages/HookProgressMessage.tsx +116 -0
  592. package/src/components/messages/PlanApprovalMessage.tsx +222 -0
  593. package/src/components/messages/RateLimitMessage.tsx +161 -0
  594. package/src/components/messages/ShutdownMessage.tsx +132 -0
  595. package/src/components/messages/SnipBoundaryMessage.ts +2 -0
  596. package/src/components/messages/SystemAPIErrorMessage.tsx +141 -0
  597. package/src/components/messages/SystemTextMessage.tsx +827 -0
  598. package/src/components/messages/TaskAssignmentMessage.tsx +76 -0
  599. package/src/components/messages/UserAgentNotificationMessage.tsx +83 -0
  600. package/src/components/messages/UserBashInputMessage.tsx +58 -0
  601. package/src/components/messages/UserBashOutputMessage.tsx +54 -0
  602. package/src/components/messages/UserChannelMessage.tsx +137 -0
  603. package/src/components/messages/UserCommandMessage.tsx +108 -0
  604. package/src/components/messages/UserCrossSessionMessage.ts +2 -0
  605. package/src/components/messages/UserForkBoilerplateMessage.ts +2 -0
  606. package/src/components/messages/UserGitHubWebhookMessage.ts +2 -0
  607. package/src/components/messages/UserImageMessage.tsx +59 -0
  608. package/src/components/messages/UserLocalCommandOutputMessage.tsx +167 -0
  609. package/src/components/messages/UserMemoryInputMessage.tsx +75 -0
  610. package/src/components/messages/UserPlanMessage.tsx +42 -0
  611. package/src/components/messages/UserPromptMessage.tsx +80 -0
  612. package/src/components/messages/UserResourceUpdateMessage.tsx +121 -0
  613. package/src/components/messages/UserTeammateMessage.tsx +206 -0
  614. package/src/components/messages/UserTextMessage.tsx +275 -0
  615. package/src/components/messages/UserToolResultMessage/RejectedPlanMessage.tsx +31 -0
  616. package/src/components/messages/UserToolResultMessage/RejectedToolUseMessage.tsx +16 -0
  617. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +16 -0
  618. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +103 -0
  619. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +95 -0
  620. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +106 -0
  621. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +104 -0
  622. package/src/components/messages/UserToolResultMessage/utils.tsx +44 -0
  623. package/src/components/messages/nullRenderingAttachments.ts +70 -0
  624. package/src/components/messages/teamMemCollapsed.tsx +140 -0
  625. package/src/components/messages/teamMemSaved.ts +19 -0
  626. package/src/components/permissions/AskUserQuestionPermissionRequest/AskUserQuestionPermissionRequest.tsx +645 -0
  627. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewBox.tsx +229 -0
  628. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewQuestionView.tsx +328 -0
  629. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionNavigationBar.tsx +178 -0
  630. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionView.tsx +465 -0
  631. package/src/components/permissions/AskUserQuestionPermissionRequest/SubmitQuestionsView.tsx +144 -0
  632. package/src/components/permissions/AskUserQuestionPermissionRequest/use-multiple-choice-state.ts +179 -0
  633. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +482 -0
  634. package/src/components/permissions/BashPermissionRequest/bashToolUseOptions.tsx +147 -0
  635. package/src/components/permissions/ComputerUseApproval/ComputerUseApproval.tsx +441 -0
  636. package/src/components/permissions/EnterPlanModePermissionRequest/EnterPlanModePermissionRequest.tsx +122 -0
  637. package/src/components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.tsx +768 -0
  638. package/src/components/permissions/FallbackPermissionRequest.tsx +333 -0
  639. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  640. package/src/components/permissions/FilePermissionDialog/FilePermissionDialog.tsx +204 -0
  641. package/src/components/permissions/FilePermissionDialog/ideDiffConfig.ts +42 -0
  642. package/src/components/permissions/FilePermissionDialog/permissionOptions.tsx +177 -0
  643. package/src/components/permissions/FilePermissionDialog/useFilePermissionDialog.ts +212 -0
  644. package/src/components/permissions/FilePermissionDialog/usePermissionHandler.ts +185 -0
  645. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +161 -0
  646. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +89 -0
  647. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +115 -0
  648. package/src/components/permissions/MonitorPermissionRequest/MonitorPermissionRequest.ts +2 -0
  649. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditPermissionRequest.tsx +166 -0
  650. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditToolDiff.tsx +235 -0
  651. package/src/components/permissions/PermissionDecisionDebugInfo.tsx +460 -0
  652. package/src/components/permissions/PermissionDialog.tsx +72 -0
  653. package/src/components/permissions/PermissionExplanation.tsx +272 -0
  654. package/src/components/permissions/PermissionPrompt.tsx +336 -0
  655. package/src/components/permissions/PermissionRequest.tsx +217 -0
  656. package/src/components/permissions/PermissionRequestTitle.tsx +66 -0
  657. package/src/components/permissions/PermissionRuleExplanation.tsx +121 -0
  658. package/src/components/permissions/PowerShellPermissionRequest/PowerShellPermissionRequest.tsx +235 -0
  659. package/src/components/permissions/PowerShellPermissionRequest/powershellToolUseOptions.tsx +91 -0
  660. package/src/components/permissions/ReviewArtifactPermissionRequest/ReviewArtifactPermissionRequest.ts +2 -0
  661. package/src/components/permissions/SandboxPermissionRequest.tsx +163 -0
  662. package/src/components/permissions/SedEditPermissionRequest/SedEditPermissionRequest.tsx +230 -0
  663. package/src/components/permissions/SkillPermissionRequest/SkillPermissionRequest.tsx +369 -0
  664. package/src/components/permissions/WebFetchPermissionRequest/WebFetchPermissionRequest.tsx +258 -0
  665. package/src/components/permissions/WorkerBadge.tsx +49 -0
  666. package/src/components/permissions/WorkerPendingPermission.tsx +105 -0
  667. package/src/components/permissions/hooks.ts +209 -0
  668. package/src/components/permissions/rules/AddPermissionRules.tsx +180 -0
  669. package/src/components/permissions/rules/AddWorkspaceDirectory.tsx +340 -0
  670. package/src/components/permissions/rules/PermissionRuleDescription.tsx +76 -0
  671. package/src/components/permissions/rules/PermissionRuleInput.tsx +138 -0
  672. package/src/components/permissions/rules/PermissionRuleList.tsx +1179 -0
  673. package/src/components/permissions/rules/RecentDenialsTab.tsx +207 -0
  674. package/src/components/permissions/rules/RemoveWorkspaceDirectory.tsx +110 -0
  675. package/src/components/permissions/rules/WorkspaceTab.tsx +150 -0
  676. package/src/components/permissions/shellPermissionHelpers.tsx +164 -0
  677. package/src/components/permissions/useShellPermissionFeedback.ts +148 -0
  678. package/src/components/permissions/utils.ts +25 -0
  679. package/src/components/sandbox/SandboxConfigTab.tsx +45 -0
  680. package/src/components/sandbox/SandboxDependenciesTab.tsx +120 -0
  681. package/src/components/sandbox/SandboxDoctorSection.tsx +46 -0
  682. package/src/components/sandbox/SandboxOverridesTab.tsx +193 -0
  683. package/src/components/sandbox/SandboxSettings.tsx +296 -0
  684. package/src/components/shell/ExpandShellOutputContext.tsx +36 -0
  685. package/src/components/shell/OutputLine.tsx +118 -0
  686. package/src/components/shell/ShellProgressMessage.tsx +150 -0
  687. package/src/components/shell/ShellTimeDisplay.tsx +74 -0
  688. package/src/components/skills/SkillsMenu.tsx +237 -0
  689. package/src/components/tasks/AsyncAgentDetailDialog.tsx +229 -0
  690. package/src/components/tasks/BackgroundTask.tsx +345 -0
  691. package/src/components/tasks/BackgroundTaskStatus.tsx +429 -0
  692. package/src/components/tasks/BackgroundTasksDialog.tsx +652 -0
  693. package/src/components/tasks/DreamDetailDialog.tsx +251 -0
  694. package/src/components/tasks/InProcessTeammateDetailDialog.tsx +266 -0
  695. package/src/components/tasks/MonitorMcpDetailDialog.ts +2 -0
  696. package/src/components/tasks/RemoteSessionDetailDialog.tsx +904 -0
  697. package/src/components/tasks/RemoteSessionProgress.tsx +243 -0
  698. package/src/components/tasks/ShellDetailDialog.tsx +404 -0
  699. package/src/components/tasks/ShellProgress.tsx +87 -0
  700. package/src/components/tasks/WorkflowDetailDialog.ts +2 -0
  701. package/src/components/tasks/renderToolActivity.tsx +33 -0
  702. package/src/components/tasks/taskStatusUtils.tsx +107 -0
  703. package/src/components/teams/TeamStatus.tsx +80 -0
  704. package/src/components/teams/TeamsDialog.tsx +715 -0
  705. package/src/components/ui/OrderedList.tsx +71 -0
  706. package/src/components/ui/OrderedListItem.tsx +45 -0
  707. package/src/components/ui/TreeSelect.tsx +397 -0
  708. package/src/components/ui/option.ts +4 -0
  709. package/src/components/wizard/WizardDialogLayout.tsx +65 -0
  710. package/src/components/wizard/WizardNavigationFooter.tsx +24 -0
  711. package/src/components/wizard/WizardProvider.tsx +213 -0
  712. package/src/components/wizard/index.ts +9 -0
  713. package/src/components/wizard/types.ts +6 -0
  714. package/src/components/wizard/useWizard.ts +13 -0
  715. package/src/constants/apiLimits.ts +94 -0
  716. package/src/constants/betas.ts +52 -0
  717. package/src/constants/common.ts +33 -0
  718. package/src/constants/cyberRiskInstruction.ts +24 -0
  719. package/src/constants/errorIds.ts +15 -0
  720. package/src/constants/figures.ts +45 -0
  721. package/src/constants/files.ts +156 -0
  722. package/src/constants/github-app.ts +144 -0
  723. package/src/constants/keys.ts +11 -0
  724. package/src/constants/messages.ts +1 -0
  725. package/src/constants/oauth.ts +234 -0
  726. package/src/constants/outputStyles.ts +216 -0
  727. package/src/constants/product.ts +76 -0
  728. package/src/constants/prompts.ts +911 -0
  729. package/src/constants/querySource.ts +52 -0
  730. package/src/constants/spinnerVerbs.ts +204 -0
  731. package/src/constants/system.ts +95 -0
  732. package/src/constants/systemPromptSections.ts +68 -0
  733. package/src/constants/toolLimits.ts +56 -0
  734. package/src/constants/tools.ts +112 -0
  735. package/src/constants/turnCompletionVerbs.ts +12 -0
  736. package/src/constants/xml.ts +86 -0
  737. package/src/context/QueuedMessageContext.tsx +63 -0
  738. package/src/context/fpsMetrics.tsx +30 -0
  739. package/src/context/mailbox.tsx +38 -0
  740. package/src/context/modalContext.tsx +58 -0
  741. package/src/context/notifications.tsx +240 -0
  742. package/src/context/overlayContext.tsx +151 -0
  743. package/src/context/promptOverlayContext.tsx +125 -0
  744. package/src/context/stats.tsx +220 -0
  745. package/src/context/voice.tsx +88 -0
  746. package/src/context.ts +189 -0
  747. package/src/coordinator/coordinatorMode.ts +369 -0
  748. package/src/coordinator/workerAgent.ts +2 -0
  749. package/src/cost-tracker.ts +323 -0
  750. package/src/costHook.ts +22 -0
  751. package/src/dialogLaunchers.tsx +133 -0
  752. package/src/entrypoints/agentSdkTypes.ts +443 -0
  753. package/src/entrypoints/cli.tsx +303 -0
  754. package/src/entrypoints/init.ts +340 -0
  755. package/src/entrypoints/mcp.ts +196 -0
  756. package/src/entrypoints/sandboxTypes.ts +156 -0
  757. package/src/entrypoints/sdk/controlSchemas.ts +663 -0
  758. package/src/entrypoints/sdk/controlTypes.ts +15 -0
  759. package/src/entrypoints/sdk/coreSchemas.ts +1889 -0
  760. package/src/entrypoints/sdk/coreTypes.generated.ts +2 -0
  761. package/src/entrypoints/sdk/coreTypes.ts +62 -0
  762. package/src/entrypoints/sdk/runtimeTypes.ts +21 -0
  763. package/src/entrypoints/sdk/sdkUtilityTypes.ts +4 -0
  764. package/src/entrypoints/sdk/settingsTypes.generated.ts +2 -0
  765. package/src/entrypoints/sdk/toolTypes.ts +2 -0
  766. package/src/history.ts +464 -0
  767. package/src/hooks/fileSuggestions.ts +811 -0
  768. package/src/hooks/notifs/useAntOrgWarningNotification.ts +2 -0
  769. package/src/hooks/notifs/useAutoModeUnavailableNotification.ts +56 -0
  770. package/src/hooks/notifs/useCanSwitchToExistingSubscription.tsx +60 -0
  771. package/src/hooks/notifs/useDeprecationWarningNotification.tsx +44 -0
  772. package/src/hooks/notifs/useFastModeNotification.tsx +162 -0
  773. package/src/hooks/notifs/useIDEStatusIndicator.tsx +186 -0
  774. package/src/hooks/notifs/useInstallMessages.tsx +26 -0
  775. package/src/hooks/notifs/useLspInitializationNotification.tsx +143 -0
  776. package/src/hooks/notifs/useMcpConnectivityStatus.tsx +88 -0
  777. package/src/hooks/notifs/useModelMigrationNotifications.tsx +52 -0
  778. package/src/hooks/notifs/useNpmDeprecationNotification.tsx +25 -0
  779. package/src/hooks/notifs/usePluginAutoupdateNotification.tsx +83 -0
  780. package/src/hooks/notifs/usePluginInstallationStatus.tsx +128 -0
  781. package/src/hooks/notifs/useRateLimitWarningNotification.tsx +114 -0
  782. package/src/hooks/notifs/useSettingsErrors.tsx +69 -0
  783. package/src/hooks/notifs/useStartupNotification.ts +41 -0
  784. package/src/hooks/notifs/useTeammateShutdownNotification.ts +78 -0
  785. package/src/hooks/renderPlaceholder.ts +51 -0
  786. package/src/hooks/toolPermission/PermissionContext.ts +388 -0
  787. package/src/hooks/toolPermission/handlers/coordinatorHandler.ts +65 -0
  788. package/src/hooks/toolPermission/handlers/interactiveHandler.ts +536 -0
  789. package/src/hooks/toolPermission/handlers/swarmWorkerHandler.ts +159 -0
  790. package/src/hooks/toolPermission/permissionLogging.ts +238 -0
  791. package/src/hooks/unifiedSuggestions.ts +202 -0
  792. package/src/hooks/useAfterFirstRender.ts +17 -0
  793. package/src/hooks/useApiKeyVerification.ts +84 -0
  794. package/src/hooks/useArrowKeyHistory.tsx +229 -0
  795. package/src/hooks/useAssistantHistory.ts +250 -0
  796. package/src/hooks/useAwaySummary.ts +125 -0
  797. package/src/hooks/useBackgroundTaskNavigation.ts +251 -0
  798. package/src/hooks/useBlink.ts +34 -0
  799. package/src/hooks/useCanUseTool.tsx +204 -0
  800. package/src/hooks/useCancelRequest.ts +276 -0
  801. package/src/hooks/useChromeExtensionNotification.tsx +50 -0
  802. package/src/hooks/useClaudeCodeHintRecommendation.tsx +129 -0
  803. package/src/hooks/useClipboardImageHint.ts +77 -0
  804. package/src/hooks/useCommandKeybindings.tsx +108 -0
  805. package/src/hooks/useCommandQueue.ts +15 -0
  806. package/src/hooks/useCopyOnSelect.ts +98 -0
  807. package/src/hooks/useDeferredHookMessages.ts +46 -0
  808. package/src/hooks/useDiffData.ts +110 -0
  809. package/src/hooks/useDiffInIDE.ts +379 -0
  810. package/src/hooks/useDirectConnect.ts +229 -0
  811. package/src/hooks/useDoublePress.ts +62 -0
  812. package/src/hooks/useDynamicConfig.ts +22 -0
  813. package/src/hooks/useElapsedTime.ts +37 -0
  814. package/src/hooks/useExitOnCtrlCD.ts +95 -0
  815. package/src/hooks/useExitOnCtrlCDWithKeybindings.ts +24 -0
  816. package/src/hooks/useFileHistorySnapshotInit.ts +25 -0
  817. package/src/hooks/useGlobalKeybindings.tsx +249 -0
  818. package/src/hooks/useHistorySearch.ts +303 -0
  819. package/src/hooks/useIDEIntegration.tsx +70 -0
  820. package/src/hooks/useIdeAtMentioned.ts +76 -0
  821. package/src/hooks/useIdeConnectionStatus.ts +33 -0
  822. package/src/hooks/useIdeLogging.ts +41 -0
  823. package/src/hooks/useIdeSelection.ts +150 -0
  824. package/src/hooks/useInboxPoller.ts +969 -0
  825. package/src/hooks/useInputBuffer.ts +132 -0
  826. package/src/hooks/useIssueFlagBanner.ts +133 -0
  827. package/src/hooks/useLogMessages.ts +119 -0
  828. package/src/hooks/useLspPluginRecommendation.tsx +194 -0
  829. package/src/hooks/useMailboxBridge.ts +21 -0
  830. package/src/hooks/useMainLoopModel.ts +34 -0
  831. package/src/hooks/useManagePlugins.ts +304 -0
  832. package/src/hooks/useMemoryUsage.ts +39 -0
  833. package/src/hooks/useMergedClients.ts +23 -0
  834. package/src/hooks/useMergedCommands.ts +15 -0
  835. package/src/hooks/useMergedTools.ts +44 -0
  836. package/src/hooks/useMinDisplayTime.ts +35 -0
  837. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  838. package/src/hooks/useOfficialMarketplaceNotification.tsx +48 -0
  839. package/src/hooks/usePasteHandler.ts +285 -0
  840. package/src/hooks/usePluginRecommendationBase.tsx +105 -0
  841. package/src/hooks/usePrStatus.ts +106 -0
  842. package/src/hooks/usePromptSuggestion.ts +177 -0
  843. package/src/hooks/usePromptsFromClaudeInChrome.tsx +71 -0
  844. package/src/hooks/useQueueProcessor.ts +68 -0
  845. package/src/hooks/useRemoteSession.ts +605 -0
  846. package/src/hooks/useReplBridge.tsx +723 -0
  847. package/src/hooks/useSSHSession.ts +241 -0
  848. package/src/hooks/useScheduledTasks.ts +139 -0
  849. package/src/hooks/useSearchInput.ts +364 -0
  850. package/src/hooks/useSessionBackgrounding.ts +158 -0
  851. package/src/hooks/useSettings.ts +17 -0
  852. package/src/hooks/useSettingsChange.ts +25 -0
  853. package/src/hooks/useSkillImprovementSurvey.ts +105 -0
  854. package/src/hooks/useSkillsChange.ts +62 -0
  855. package/src/hooks/useSwarmInitialization.ts +81 -0
  856. package/src/hooks/useSwarmPermissionPoller.ts +330 -0
  857. package/src/hooks/useTaskListWatcher.ts +221 -0
  858. package/src/hooks/useTasksV2.ts +250 -0
  859. package/src/hooks/useTeammateViewAutoExit.ts +63 -0
  860. package/src/hooks/useTeleportResume.tsx +85 -0
  861. package/src/hooks/useTerminalSize.ts +15 -0
  862. package/src/hooks/useTextInput.ts +529 -0
  863. package/src/hooks/useTimeout.ts +14 -0
  864. package/src/hooks/useTurnDiffs.ts +213 -0
  865. package/src/hooks/useTypeahead.tsx +1385 -0
  866. package/src/hooks/useUpdateNotification.ts +34 -0
  867. package/src/hooks/useVimInput.ts +316 -0
  868. package/src/hooks/useVirtualScroll.ts +721 -0
  869. package/src/hooks/useVoice.ts +1144 -0
  870. package/src/hooks/useVoiceEnabled.ts +25 -0
  871. package/src/hooks/useVoiceIntegration.tsx +677 -0
  872. package/src/ink/Ansi.tsx +292 -0
  873. package/src/ink/bidi.ts +139 -0
  874. package/src/ink/clearTerminal.ts +74 -0
  875. package/src/ink/colorize.ts +231 -0
  876. package/src/ink/components/AlternateScreen.tsx +80 -0
  877. package/src/ink/components/App.tsx +658 -0
  878. package/src/ink/components/AppContext.ts +21 -0
  879. package/src/ink/components/Box.tsx +214 -0
  880. package/src/ink/components/Button.tsx +192 -0
  881. package/src/ink/components/ClockContext.tsx +112 -0
  882. package/src/ink/components/CursorDeclarationContext.ts +32 -0
  883. package/src/ink/components/ErrorOverview.tsx +109 -0
  884. package/src/ink/components/Link.tsx +42 -0
  885. package/src/ink/components/Newline.tsx +39 -0
  886. package/src/ink/components/NoSelect.tsx +68 -0
  887. package/src/ink/components/RawAnsi.tsx +57 -0
  888. package/src/ink/components/ScrollBox.tsx +237 -0
  889. package/src/ink/components/Spacer.tsx +20 -0
  890. package/src/ink/components/StdinContext.ts +49 -0
  891. package/src/ink/components/TerminalFocusContext.tsx +52 -0
  892. package/src/ink/components/TerminalSizeContext.tsx +7 -0
  893. package/src/ink/components/Text.tsx +254 -0
  894. package/src/ink/constants.ts +2 -0
  895. package/src/ink/cursor.ts +4 -0
  896. package/src/ink/devtools-window-polyfill.js +7 -0
  897. package/src/ink/devtools.js +9 -0
  898. package/src/ink/dom.ts +484 -0
  899. package/src/ink/events/click-event.ts +38 -0
  900. package/src/ink/events/dispatcher.ts +233 -0
  901. package/src/ink/events/emitter.ts +39 -0
  902. package/src/ink/events/event-handlers.ts +73 -0
  903. package/src/ink/events/event.ts +11 -0
  904. package/src/ink/events/focus-event.ts +21 -0
  905. package/src/ink/events/input-event.ts +205 -0
  906. package/src/ink/events/keyboard-event.ts +51 -0
  907. package/src/ink/events/paste-event.ts +4 -0
  908. package/src/ink/events/resize-event.ts +4 -0
  909. package/src/ink/events/terminal-event.ts +107 -0
  910. package/src/ink/events/terminal-focus-event.ts +19 -0
  911. package/src/ink/focus.ts +181 -0
  912. package/src/ink/frame.ts +124 -0
  913. package/src/ink/get-max-width.ts +27 -0
  914. package/src/ink/global.d.ts +3 -0
  915. package/src/ink/hit-test.ts +130 -0
  916. package/src/ink/hooks/use-animation-frame.ts +57 -0
  917. package/src/ink/hooks/use-app.ts +8 -0
  918. package/src/ink/hooks/use-declared-cursor.ts +73 -0
  919. package/src/ink/hooks/use-input.ts +92 -0
  920. package/src/ink/hooks/use-interval.ts +67 -0
  921. package/src/ink/hooks/use-search-highlight.ts +53 -0
  922. package/src/ink/hooks/use-selection.ts +104 -0
  923. package/src/ink/hooks/use-stdin.ts +8 -0
  924. package/src/ink/hooks/use-tab-status.ts +72 -0
  925. package/src/ink/hooks/use-terminal-focus.ts +16 -0
  926. package/src/ink/hooks/use-terminal-title.ts +31 -0
  927. package/src/ink/hooks/use-terminal-viewport.ts +96 -0
  928. package/src/ink/ink.tsx +1723 -0
  929. package/src/ink/instances.ts +10 -0
  930. package/src/ink/layout/engine.ts +6 -0
  931. package/src/ink/layout/geometry.ts +97 -0
  932. package/src/ink/layout/node.ts +152 -0
  933. package/src/ink/layout/yoga.ts +308 -0
  934. package/src/ink/line-width-cache.ts +24 -0
  935. package/src/ink/log-update.ts +773 -0
  936. package/src/ink/measure-element.ts +23 -0
  937. package/src/ink/measure-text.ts +47 -0
  938. package/src/ink/node-cache.ts +54 -0
  939. package/src/ink/optimizer.ts +93 -0
  940. package/src/ink/output.ts +797 -0
  941. package/src/ink/parse-keypress.ts +801 -0
  942. package/src/ink/reconciler.ts +512 -0
  943. package/src/ink/render-border.ts +231 -0
  944. package/src/ink/render-node-to-output.ts +1462 -0
  945. package/src/ink/render-to-screen.ts +231 -0
  946. package/src/ink/renderer.ts +178 -0
  947. package/src/ink/root.ts +184 -0
  948. package/src/ink/screen.ts +1486 -0
  949. package/src/ink/searchHighlight.ts +93 -0
  950. package/src/ink/selection.ts +917 -0
  951. package/src/ink/squash-text-nodes.ts +92 -0
  952. package/src/ink/stringWidth.ts +222 -0
  953. package/src/ink/styles.ts +771 -0
  954. package/src/ink/supports-hyperlinks.ts +57 -0
  955. package/src/ink/tabstops.ts +46 -0
  956. package/src/ink/terminal-focus-state.ts +47 -0
  957. package/src/ink/terminal-querier.ts +212 -0
  958. package/src/ink/terminal.ts +248 -0
  959. package/src/ink/termio/ansi.ts +75 -0
  960. package/src/ink/termio/csi.ts +319 -0
  961. package/src/ink/termio/dec.ts +60 -0
  962. package/src/ink/termio/esc.ts +67 -0
  963. package/src/ink/termio/osc.ts +493 -0
  964. package/src/ink/termio/parser.ts +394 -0
  965. package/src/ink/termio/sgr.ts +308 -0
  966. package/src/ink/termio/tokenize.ts +319 -0
  967. package/src/ink/termio/types.ts +236 -0
  968. package/src/ink/termio.ts +42 -0
  969. package/src/ink/useTerminalNotification.ts +126 -0
  970. package/src/ink/warn.ts +9 -0
  971. package/src/ink/widest-line.ts +19 -0
  972. package/src/ink/wrap-text.ts +74 -0
  973. package/src/ink/wrapAnsi.ts +20 -0
  974. package/src/ink.ts +85 -0
  975. package/src/interactiveHelpers.tsx +366 -0
  976. package/src/jobs/classifier.ts +2 -0
  977. package/src/keybindings/KeybindingContext.tsx +243 -0
  978. package/src/keybindings/KeybindingProviderSetup.tsx +308 -0
  979. package/src/keybindings/defaultBindings.ts +340 -0
  980. package/src/keybindings/loadUserBindings.ts +472 -0
  981. package/src/keybindings/match.ts +120 -0
  982. package/src/keybindings/parser.ts +203 -0
  983. package/src/keybindings/reservedShortcuts.ts +127 -0
  984. package/src/keybindings/resolver.ts +244 -0
  985. package/src/keybindings/schema.ts +236 -0
  986. package/src/keybindings/shortcutFormat.ts +63 -0
  987. package/src/keybindings/template.ts +52 -0
  988. package/src/keybindings/types.ts +9 -0
  989. package/src/keybindings/useKeybinding.ts +196 -0
  990. package/src/keybindings/useShortcutDisplay.ts +59 -0
  991. package/src/keybindings/validate.ts +498 -0
  992. package/src/main.tsx +4684 -0
  993. package/src/memdir/findRelevantMemories.ts +141 -0
  994. package/src/memdir/memdir.ts +507 -0
  995. package/src/memdir/memoryAge.ts +53 -0
  996. package/src/memdir/memoryScan.ts +94 -0
  997. package/src/memdir/memoryShapeTelemetry.ts +2 -0
  998. package/src/memdir/memoryTypes.ts +271 -0
  999. package/src/memdir/paths.ts +278 -0
  1000. package/src/memdir/teamMemPaths.ts +292 -0
  1001. package/src/memdir/teamMemPrompts.ts +100 -0
  1002. package/src/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  1003. package/src/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  1004. package/src/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  1005. package/src/migrations/migrateFennecToOpus.ts +45 -0
  1006. package/src/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  1007. package/src/migrations/migrateOpusToOpus1m.ts +43 -0
  1008. package/src/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  1009. package/src/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  1010. package/src/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  1011. package/src/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  1012. package/src/migrations/resetProToOpusDefault.ts +51 -0
  1013. package/src/moreright/useMoreRight.tsx +26 -0
  1014. package/src/native-ts/color-diff/index.ts +999 -0
  1015. package/src/native-ts/file-index/index.ts +370 -0
  1016. package/src/native-ts/yoga-layout/enums.ts +134 -0
  1017. package/src/native-ts/yoga-layout/index.ts +2578 -0
  1018. package/src/outputStyles/loadOutputStylesDir.ts +98 -0
  1019. package/src/plugins/builtinPlugins.ts +159 -0
  1020. package/src/plugins/bundled/index.ts +23 -0
  1021. package/src/proactive/index.ts +2 -0
  1022. package/src/proactive/useProactive.ts +2 -0
  1023. package/src/projectOnboardingState.ts +83 -0
  1024. package/src/query/config.ts +46 -0
  1025. package/src/query/deps.ts +40 -0
  1026. package/src/query/stopHooks.ts +473 -0
  1027. package/src/query/tokenBudget.ts +93 -0
  1028. package/src/query/transitions.ts +5 -0
  1029. package/src/query.ts +1729 -0
  1030. package/src/remote/RemoteSessionManager.ts +343 -0
  1031. package/src/remote/SessionsWebSocket.ts +404 -0
  1032. package/src/remote/remotePermissionBridge.ts +78 -0
  1033. package/src/remote/sdkMessageAdapter.ts +302 -0
  1034. package/src/replLauncher.tsx +23 -0
  1035. package/src/schemas/hooks.ts +222 -0
  1036. package/src/screens/Doctor.tsx +575 -0
  1037. package/src/screens/REPL.tsx +5006 -0
  1038. package/src/screens/ResumeConversation.tsx +399 -0
  1039. package/src/server/createDirectConnectSession.ts +88 -0
  1040. package/src/server/directConnectManager.ts +213 -0
  1041. package/src/server/types.ts +57 -0
  1042. package/src/services/AgentSummary/agentSummary.ts +179 -0
  1043. package/src/services/MagicDocs/magicDocs.ts +254 -0
  1044. package/src/services/MagicDocs/prompts.ts +127 -0
  1045. package/src/services/PromptSuggestion/promptSuggestion.ts +523 -0
  1046. package/src/services/PromptSuggestion/speculation.ts +991 -0
  1047. package/src/services/SessionMemory/prompts.ts +324 -0
  1048. package/src/services/SessionMemory/sessionMemory.ts +495 -0
  1049. package/src/services/SessionMemory/sessionMemoryUtils.ts +207 -0
  1050. package/src/services/analytics/config.ts +38 -0
  1051. package/src/services/analytics/datadog.ts +307 -0
  1052. package/src/services/analytics/firstPartyEventLogger.ts +449 -0
  1053. package/src/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
  1054. package/src/services/analytics/growthbook.ts +1155 -0
  1055. package/src/services/analytics/index.ts +173 -0
  1056. package/src/services/analytics/metadata.ts +973 -0
  1057. package/src/services/analytics/sink.ts +114 -0
  1058. package/src/services/analytics/sinkKillswitch.ts +25 -0
  1059. package/src/services/api/adminRequests.ts +119 -0
  1060. package/src/services/api/bootstrap.ts +141 -0
  1061. package/src/services/api/claude.ts +3450 -0
  1062. package/src/services/api/client.ts +459 -0
  1063. package/src/services/api/codexShim.ts +833 -0
  1064. package/src/services/api/dumpPrompts.ts +226 -0
  1065. package/src/services/api/emptyUsage.ts +22 -0
  1066. package/src/services/api/errorUtils.ts +260 -0
  1067. package/src/services/api/errors.ts +1220 -0
  1068. package/src/services/api/filesApi.ts +748 -0
  1069. package/src/services/api/firstTokenDate.ts +60 -0
  1070. package/src/services/api/grove.ts +357 -0
  1071. package/src/services/api/logging.ts +788 -0
  1072. package/src/services/api/metricsOptOut.ts +159 -0
  1073. package/src/services/api/openaiProviderConfig.ts +333 -0
  1074. package/src/services/api/openaiShim.ts +882 -0
  1075. package/src/services/api/overageCreditGrant.ts +137 -0
  1076. package/src/services/api/promptCacheBreakDetection.ts +727 -0
  1077. package/src/services/api/referral.ts +281 -0
  1078. package/src/services/api/sessionIngress.ts +514 -0
  1079. package/src/services/api/ultrareviewQuota.ts +38 -0
  1080. package/src/services/api/usage.ts +63 -0
  1081. package/src/services/api/withRetry.ts +856 -0
  1082. package/src/services/autoDream/autoDream.ts +324 -0
  1083. package/src/services/autoDream/config.ts +21 -0
  1084. package/src/services/autoDream/consolidationLock.ts +140 -0
  1085. package/src/services/autoDream/consolidationPrompt.ts +65 -0
  1086. package/src/services/awaySummary.ts +74 -0
  1087. package/src/services/claudeAiLimits.ts +515 -0
  1088. package/src/services/claudeAiLimitsHook.ts +23 -0
  1089. package/src/services/compact/apiMicrocompact.ts +153 -0
  1090. package/src/services/compact/autoCompact.ts +351 -0
  1091. package/src/services/compact/cachedMCConfig.ts +2 -0
  1092. package/src/services/compact/cachedMicrocompact.ts +18 -0
  1093. package/src/services/compact/compact.ts +1705 -0
  1094. package/src/services/compact/compactWarningHook.ts +16 -0
  1095. package/src/services/compact/compactWarningState.ts +18 -0
  1096. package/src/services/compact/grouping.ts +63 -0
  1097. package/src/services/compact/microCompact.ts +530 -0
  1098. package/src/services/compact/postCompactCleanup.ts +77 -0
  1099. package/src/services/compact/prompt.ts +374 -0
  1100. package/src/services/compact/reactiveCompact.ts +2 -0
  1101. package/src/services/compact/sessionMemoryCompact.ts +630 -0
  1102. package/src/services/compact/snipCompact.ts +2 -0
  1103. package/src/services/compact/snipProjection.ts +2 -0
  1104. package/src/services/compact/timeBasedMCConfig.ts +43 -0
  1105. package/src/services/contextCollapse/index.ts +2 -0
  1106. package/src/services/contextCollapse/operations.ts +2 -0
  1107. package/src/services/contextCollapse/persist.ts +2 -0
  1108. package/src/services/diagnosticTracking.ts +397 -0
  1109. package/src/services/extractMemories/extractMemories.ts +615 -0
  1110. package/src/services/extractMemories/prompts.ts +154 -0
  1111. package/src/services/internalLogging.ts +90 -0
  1112. package/src/services/lsp/LSPClient.ts +447 -0
  1113. package/src/services/lsp/LSPDiagnosticRegistry.ts +386 -0
  1114. package/src/services/lsp/LSPServerInstance.ts +511 -0
  1115. package/src/services/lsp/LSPServerManager.ts +420 -0
  1116. package/src/services/lsp/config.ts +79 -0
  1117. package/src/services/lsp/manager.ts +289 -0
  1118. package/src/services/lsp/passiveFeedback.ts +328 -0
  1119. package/src/services/lsp/types.ts +6 -0
  1120. package/src/services/mcp/InProcessTransport.ts +63 -0
  1121. package/src/services/mcp/MCPConnectionManager.tsx +73 -0
  1122. package/src/services/mcp/SdkControlTransport.ts +136 -0
  1123. package/src/services/mcp/auth.ts +2465 -0
  1124. package/src/services/mcp/channelAllowlist.ts +76 -0
  1125. package/src/services/mcp/channelNotification.ts +316 -0
  1126. package/src/services/mcp/channelPermissions.ts +240 -0
  1127. package/src/services/mcp/claudeai.ts +164 -0
  1128. package/src/services/mcp/client.ts +3348 -0
  1129. package/src/services/mcp/config.ts +1578 -0
  1130. package/src/services/mcp/elicitationHandler.ts +313 -0
  1131. package/src/services/mcp/envExpansion.ts +38 -0
  1132. package/src/services/mcp/headersHelper.ts +138 -0
  1133. package/src/services/mcp/mcpStringUtils.ts +106 -0
  1134. package/src/services/mcp/normalization.ts +23 -0
  1135. package/src/services/mcp/oauthPort.ts +78 -0
  1136. package/src/services/mcp/officialRegistry.ts +72 -0
  1137. package/src/services/mcp/types.ts +258 -0
  1138. package/src/services/mcp/useManageMCPConnections.ts +1141 -0
  1139. package/src/services/mcp/utils.ts +575 -0
  1140. package/src/services/mcp/vscodeSdkMcp.ts +112 -0
  1141. package/src/services/mcp/xaa.ts +511 -0
  1142. package/src/services/mcp/xaaIdpLogin.ts +487 -0
  1143. package/src/services/mcpServerApproval.tsx +41 -0
  1144. package/src/services/mockRateLimits.ts +882 -0
  1145. package/src/services/notifier.ts +156 -0
  1146. package/src/services/oauth/auth-code-listener.ts +211 -0
  1147. package/src/services/oauth/client.ts +566 -0
  1148. package/src/services/oauth/crypto.ts +23 -0
  1149. package/src/services/oauth/getOauthProfile.ts +53 -0
  1150. package/src/services/oauth/index.ts +198 -0
  1151. package/src/services/oauth/types.ts +14 -0
  1152. package/src/services/plugins/PluginInstallationManager.ts +184 -0
  1153. package/src/services/plugins/pluginCliCommands.ts +344 -0
  1154. package/src/services/plugins/pluginOperations.ts +1088 -0
  1155. package/src/services/policyLimits/index.ts +663 -0
  1156. package/src/services/policyLimits/types.ts +27 -0
  1157. package/src/services/preventSleep.ts +165 -0
  1158. package/src/services/rateLimitMessages.ts +344 -0
  1159. package/src/services/rateLimitMocking.ts +144 -0
  1160. package/src/services/remoteManagedSettings/index.ts +638 -0
  1161. package/src/services/remoteManagedSettings/securityCheck.jsx.ts +5 -0
  1162. package/src/services/remoteManagedSettings/securityCheck.tsx +74 -0
  1163. package/src/services/remoteManagedSettings/syncCache.ts +112 -0
  1164. package/src/services/remoteManagedSettings/syncCacheState.ts +96 -0
  1165. package/src/services/remoteManagedSettings/types.ts +31 -0
  1166. package/src/services/sessionTranscript/sessionTranscript.ts +2 -0
  1167. package/src/services/settingsSync/index.ts +581 -0
  1168. package/src/services/settingsSync/types.ts +67 -0
  1169. package/src/services/skillSearch/featureCheck.ts +2 -0
  1170. package/src/services/skillSearch/localSearch.ts +2 -0
  1171. package/src/services/skillSearch/prefetch.ts +2 -0
  1172. package/src/services/skillSearch/remoteSkillLoader.ts +2 -0
  1173. package/src/services/skillSearch/remoteSkillState.ts +2 -0
  1174. package/src/services/skillSearch/signals.ts +4 -0
  1175. package/src/services/skillSearch/telemetry.ts +2 -0
  1176. package/src/services/teamMemorySync/index.ts +1256 -0
  1177. package/src/services/teamMemorySync/secretScanner.ts +324 -0
  1178. package/src/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
  1179. package/src/services/teamMemorySync/types.ts +156 -0
  1180. package/src/services/teamMemorySync/watcher.ts +387 -0
  1181. package/src/services/tips/tipHistory.ts +17 -0
  1182. package/src/services/tips/tipRegistry.ts +686 -0
  1183. package/src/services/tips/tipScheduler.ts +58 -0
  1184. package/src/services/tips/types.ts +5 -0
  1185. package/src/services/tokenEstimation.ts +495 -0
  1186. package/src/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
  1187. package/src/services/tools/StreamingToolExecutor.ts +530 -0
  1188. package/src/services/tools/toolExecution.ts +1745 -0
  1189. package/src/services/tools/toolHooks.ts +650 -0
  1190. package/src/services/tools/toolOrchestration.ts +188 -0
  1191. package/src/services/vcr.ts +406 -0
  1192. package/src/services/voice.ts +525 -0
  1193. package/src/services/voiceKeyterms.ts +106 -0
  1194. package/src/services/voiceStreamSTT.ts +544 -0
  1195. package/src/setup.ts +477 -0
  1196. package/src/skills/bundled/batch.ts +124 -0
  1197. package/src/skills/bundled/claude-api/python/agent-sdk/README.md +1 -0
  1198. package/src/skills/bundled/claude-api/python/claude-api/README.md +1 -0
  1199. package/src/skills/bundled/claude-api/typescript/agent-sdk/README.md +1 -0
  1200. package/src/skills/bundled/claude-api/typescript/claude-api/README.md +1 -0
  1201. package/src/skills/bundled/claudeApi.ts +196 -0
  1202. package/src/skills/bundled/claudeApiContent.ts +75 -0
  1203. package/src/skills/bundled/claudeInChrome.ts +34 -0
  1204. package/src/skills/bundled/debug.ts +103 -0
  1205. package/src/skills/bundled/dream.ts +2 -0
  1206. package/src/skills/bundled/hunter.ts +2 -0
  1207. package/src/skills/bundled/index.ts +79 -0
  1208. package/src/skills/bundled/keybindings.ts +339 -0
  1209. package/src/skills/bundled/loop.ts +92 -0
  1210. package/src/skills/bundled/loremIpsum.ts +282 -0
  1211. package/src/skills/bundled/remember.ts +82 -0
  1212. package/src/skills/bundled/runSkillGenerator.ts +2 -0
  1213. package/src/skills/bundled/scheduleRemoteAgents.ts +447 -0
  1214. package/src/skills/bundled/simplify.ts +69 -0
  1215. package/src/skills/bundled/skillify.ts +197 -0
  1216. package/src/skills/bundled/stuck.ts +79 -0
  1217. package/src/skills/bundled/updateConfig.ts +475 -0
  1218. package/src/skills/bundled/verify.ts +30 -0
  1219. package/src/skills/bundled/verifyContent.ts +13 -0
  1220. package/src/skills/bundledSkills.ts +220 -0
  1221. package/src/skills/loadSkillsDir.ts +1086 -0
  1222. package/src/skills/mcpSkillBuilders.ts +44 -0
  1223. package/src/skills/mcpSkills.ts +2 -0
  1224. package/src/ssh/SSHSessionManager.ts +4 -0
  1225. package/src/ssh/createSSHSession.ts +4 -0
  1226. package/src/state/AppState.tsx +200 -0
  1227. package/src/state/AppStateStore.ts +569 -0
  1228. package/src/state/onChangeAppState.ts +171 -0
  1229. package/src/state/selectors.ts +76 -0
  1230. package/src/state/store.ts +34 -0
  1231. package/src/state/teammateViewHelpers.ts +141 -0
  1232. package/src/tasks/DreamTask/DreamTask.ts +157 -0
  1233. package/src/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  1234. package/src/tasks/InProcessTeammateTask/types.ts +121 -0
  1235. package/src/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  1236. package/src/tasks/LocalMainSessionTask.ts +479 -0
  1237. package/src/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  1238. package/src/tasks/LocalShellTask/guards.ts +41 -0
  1239. package/src/tasks/LocalShellTask/killShellTasks.ts +76 -0
  1240. package/src/tasks/LocalWorkflowTask/LocalWorkflowTask.ts +4 -0
  1241. package/src/tasks/MonitorMcpTask/MonitorMcpTask.ts +4 -0
  1242. package/src/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  1243. package/src/tasks/pillLabel.ts +82 -0
  1244. package/src/tasks/stopTask.ts +100 -0
  1245. package/src/tasks/types.ts +46 -0
  1246. package/src/tasks.ts +39 -0
  1247. package/src/tools/AgentTool/AgentTool.tsx +1414 -0
  1248. package/src/tools/AgentTool/UI.tsx +872 -0
  1249. package/src/tools/AgentTool/agentColorManager.ts +66 -0
  1250. package/src/tools/AgentTool/agentDisplay.ts +104 -0
  1251. package/src/tools/AgentTool/agentMemory.ts +177 -0
  1252. package/src/tools/AgentTool/agentMemorySnapshot.ts +197 -0
  1253. package/src/tools/AgentTool/agentToolUtils.ts +686 -0
  1254. package/src/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
  1255. package/src/tools/AgentTool/built-in/exploreAgent.ts +83 -0
  1256. package/src/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
  1257. package/src/tools/AgentTool/built-in/planAgent.ts +92 -0
  1258. package/src/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
  1259. package/src/tools/AgentTool/built-in/verificationAgent.ts +152 -0
  1260. package/src/tools/AgentTool/builtInAgents.ts +72 -0
  1261. package/src/tools/AgentTool/constants.ts +12 -0
  1262. package/src/tools/AgentTool/forkSubagent.ts +210 -0
  1263. package/src/tools/AgentTool/loadAgentsDir.ts +755 -0
  1264. package/src/tools/AgentTool/prompt.ts +287 -0
  1265. package/src/tools/AgentTool/resumeAgent.ts +265 -0
  1266. package/src/tools/AgentTool/runAgent.ts +973 -0
  1267. package/src/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
  1268. package/src/tools/AskUserQuestionTool/prompt.ts +44 -0
  1269. package/src/tools/BashTool/BashTool.tsx +1144 -0
  1270. package/src/tools/BashTool/BashToolResultMessage.tsx +191 -0
  1271. package/src/tools/BashTool/UI.tsx +185 -0
  1272. package/src/tools/BashTool/bashCommandHelpers.ts +265 -0
  1273. package/src/tools/BashTool/bashPermissions.ts +2621 -0
  1274. package/src/tools/BashTool/bashSecurity.ts +2592 -0
  1275. package/src/tools/BashTool/commandSemantics.ts +140 -0
  1276. package/src/tools/BashTool/commentLabel.ts +13 -0
  1277. package/src/tools/BashTool/destructiveCommandWarning.ts +102 -0
  1278. package/src/tools/BashTool/modeValidation.ts +115 -0
  1279. package/src/tools/BashTool/pathValidation.ts +1303 -0
  1280. package/src/tools/BashTool/prompt.ts +369 -0
  1281. package/src/tools/BashTool/readOnlyValidation.ts +1990 -0
  1282. package/src/tools/BashTool/sedEditParser.ts +322 -0
  1283. package/src/tools/BashTool/sedValidation.ts +684 -0
  1284. package/src/tools/BashTool/shouldUseSandbox.ts +153 -0
  1285. package/src/tools/BashTool/toolName.ts +2 -0
  1286. package/src/tools/BashTool/utils.ts +223 -0
  1287. package/src/tools/BriefTool/BriefTool.ts +204 -0
  1288. package/src/tools/BriefTool/UI.tsx +101 -0
  1289. package/src/tools/BriefTool/attachments.ts +110 -0
  1290. package/src/tools/BriefTool/prompt.ts +22 -0
  1291. package/src/tools/BriefTool/upload.ts +174 -0
  1292. package/src/tools/ConfigTool/ConfigTool.ts +467 -0
  1293. package/src/tools/ConfigTool/UI.tsx +38 -0
  1294. package/src/tools/ConfigTool/constants.ts +1 -0
  1295. package/src/tools/ConfigTool/prompt.ts +93 -0
  1296. package/src/tools/ConfigTool/supportedSettings.ts +211 -0
  1297. package/src/tools/CtxInspectTool/CtxInspectTool.ts +2 -0
  1298. package/src/tools/DiscoverSkillsTool/prompt.ts +2 -0
  1299. package/src/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
  1300. package/src/tools/EnterPlanModeTool/UI.tsx +33 -0
  1301. package/src/tools/EnterPlanModeTool/constants.ts +1 -0
  1302. package/src/tools/EnterPlanModeTool/prompt.ts +170 -0
  1303. package/src/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
  1304. package/src/tools/EnterWorktreeTool/UI.tsx +20 -0
  1305. package/src/tools/EnterWorktreeTool/constants.ts +1 -0
  1306. package/src/tools/EnterWorktreeTool/prompt.ts +30 -0
  1307. package/src/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
  1308. package/src/tools/ExitPlanModeTool/UI.tsx +82 -0
  1309. package/src/tools/ExitPlanModeTool/constants.ts +2 -0
  1310. package/src/tools/ExitPlanModeTool/prompt.ts +29 -0
  1311. package/src/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
  1312. package/src/tools/ExitWorktreeTool/UI.tsx +25 -0
  1313. package/src/tools/ExitWorktreeTool/constants.ts +1 -0
  1314. package/src/tools/ExitWorktreeTool/prompt.ts +32 -0
  1315. package/src/tools/FileEditTool/FileEditTool.ts +625 -0
  1316. package/src/tools/FileEditTool/UI.tsx +289 -0
  1317. package/src/tools/FileEditTool/constants.ts +11 -0
  1318. package/src/tools/FileEditTool/prompt.ts +28 -0
  1319. package/src/tools/FileEditTool/types.ts +85 -0
  1320. package/src/tools/FileEditTool/utils.ts +775 -0
  1321. package/src/tools/FileReadTool/FileReadTool.ts +1183 -0
  1322. package/src/tools/FileReadTool/UI.tsx +185 -0
  1323. package/src/tools/FileReadTool/imageProcessor.ts +94 -0
  1324. package/src/tools/FileReadTool/limits.ts +92 -0
  1325. package/src/tools/FileReadTool/prompt.ts +49 -0
  1326. package/src/tools/FileWriteTool/FileWriteTool.ts +434 -0
  1327. package/src/tools/FileWriteTool/UI.tsx +405 -0
  1328. package/src/tools/FileWriteTool/prompt.ts +18 -0
  1329. package/src/tools/GlobTool/GlobTool.ts +198 -0
  1330. package/src/tools/GlobTool/UI.tsx +63 -0
  1331. package/src/tools/GlobTool/prompt.ts +7 -0
  1332. package/src/tools/GrepTool/GrepTool.ts +577 -0
  1333. package/src/tools/GrepTool/UI.tsx +201 -0
  1334. package/src/tools/GrepTool/prompt.ts +18 -0
  1335. package/src/tools/LSPTool/LSPTool.ts +860 -0
  1336. package/src/tools/LSPTool/UI.tsx +228 -0
  1337. package/src/tools/LSPTool/formatters.ts +592 -0
  1338. package/src/tools/LSPTool/prompt.ts +21 -0
  1339. package/src/tools/LSPTool/schemas.ts +215 -0
  1340. package/src/tools/LSPTool/symbolContext.ts +90 -0
  1341. package/src/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
  1342. package/src/tools/ListMcpResourcesTool/UI.tsx +29 -0
  1343. package/src/tools/ListMcpResourcesTool/prompt.ts +20 -0
  1344. package/src/tools/ListPeersTool/ListPeersTool.ts +2 -0
  1345. package/src/tools/MCPTool/MCPTool.ts +77 -0
  1346. package/src/tools/MCPTool/UI.tsx +403 -0
  1347. package/src/tools/MCPTool/classifyForCollapse.ts +604 -0
  1348. package/src/tools/MCPTool/prompt.ts +3 -0
  1349. package/src/tools/McpAuthTool/McpAuthTool.ts +215 -0
  1350. package/src/tools/MonitorTool/MonitorTool.ts +2 -0
  1351. package/src/tools/NotebookEditTool/NotebookEditTool.ts +490 -0
  1352. package/src/tools/NotebookEditTool/UI.tsx +93 -0
  1353. package/src/tools/NotebookEditTool/constants.ts +2 -0
  1354. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  1355. package/src/tools/OverflowTestTool/OverflowTestTool.ts +2 -0
  1356. package/src/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
  1357. package/src/tools/PowerShellTool/UI.tsx +131 -0
  1358. package/src/tools/PowerShellTool/clmTypes.ts +211 -0
  1359. package/src/tools/PowerShellTool/commandSemantics.ts +142 -0
  1360. package/src/tools/PowerShellTool/commonParameters.ts +30 -0
  1361. package/src/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
  1362. package/src/tools/PowerShellTool/gitSafety.ts +176 -0
  1363. package/src/tools/PowerShellTool/modeValidation.ts +404 -0
  1364. package/src/tools/PowerShellTool/pathValidation.ts +2049 -0
  1365. package/src/tools/PowerShellTool/powershellPermissions.ts +1648 -0
  1366. package/src/tools/PowerShellTool/powershellSecurity.ts +1090 -0
  1367. package/src/tools/PowerShellTool/prompt.ts +145 -0
  1368. package/src/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
  1369. package/src/tools/PowerShellTool/toolName.ts +2 -0
  1370. package/src/tools/PushNotificationTool/PushNotificationTool.ts +2 -0
  1371. package/src/tools/REPLTool/REPLTool.ts +2 -0
  1372. package/src/tools/REPLTool/constants.ts +46 -0
  1373. package/src/tools/REPLTool/primitiveTools.ts +39 -0
  1374. package/src/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
  1375. package/src/tools/ReadMcpResourceTool/UI.tsx +37 -0
  1376. package/src/tools/ReadMcpResourceTool/prompt.ts +16 -0
  1377. package/src/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
  1378. package/src/tools/RemoteTriggerTool/UI.tsx +17 -0
  1379. package/src/tools/RemoteTriggerTool/prompt.ts +15 -0
  1380. package/src/tools/ReviewArtifactTool/ReviewArtifactTool.ts +2 -0
  1381. package/src/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
  1382. package/src/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
  1383. package/src/tools/ScheduleCronTool/CronListTool.ts +97 -0
  1384. package/src/tools/ScheduleCronTool/UI.tsx +60 -0
  1385. package/src/tools/ScheduleCronTool/prompt.ts +135 -0
  1386. package/src/tools/SendMessageTool/SendMessageTool.ts +930 -0
  1387. package/src/tools/SendMessageTool/UI.tsx +31 -0
  1388. package/src/tools/SendMessageTool/constants.ts +1 -0
  1389. package/src/tools/SendMessageTool/prompt.ts +49 -0
  1390. package/src/tools/SendUserFileTool/SendUserFileTool.ts +2 -0
  1391. package/src/tools/SendUserFileTool/prompt.ts +2 -0
  1392. package/src/tools/SkillTool/SkillTool.ts +1108 -0
  1393. package/src/tools/SkillTool/UI.tsx +128 -0
  1394. package/src/tools/SkillTool/constants.ts +1 -0
  1395. package/src/tools/SkillTool/prompt.ts +241 -0
  1396. package/src/tools/SleepTool/SleepTool.ts +2 -0
  1397. package/src/tools/SleepTool/prompt.ts +17 -0
  1398. package/src/tools/SnipTool/SnipTool.ts +2 -0
  1399. package/src/tools/SnipTool/prompt.ts +2 -0
  1400. package/src/tools/SubscribePRTool/SubscribePRTool.ts +2 -0
  1401. package/src/tools/SuggestBackgroundPRTool/SuggestBackgroundPRTool.ts +2 -0
  1402. package/src/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
  1403. package/src/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
  1404. package/src/tools/TaskCreateTool/constants.ts +1 -0
  1405. package/src/tools/TaskCreateTool/prompt.ts +56 -0
  1406. package/src/tools/TaskGetTool/TaskGetTool.ts +128 -0
  1407. package/src/tools/TaskGetTool/constants.ts +1 -0
  1408. package/src/tools/TaskGetTool/prompt.ts +24 -0
  1409. package/src/tools/TaskListTool/TaskListTool.ts +116 -0
  1410. package/src/tools/TaskListTool/constants.ts +1 -0
  1411. package/src/tools/TaskListTool/prompt.ts +49 -0
  1412. package/src/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
  1413. package/src/tools/TaskOutputTool/constants.ts +1 -0
  1414. package/src/tools/TaskStopTool/TaskStopTool.ts +131 -0
  1415. package/src/tools/TaskStopTool/UI.tsx +41 -0
  1416. package/src/tools/TaskStopTool/prompt.ts +8 -0
  1417. package/src/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
  1418. package/src/tools/TaskUpdateTool/constants.ts +1 -0
  1419. package/src/tools/TaskUpdateTool/prompt.ts +77 -0
  1420. package/src/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
  1421. package/src/tools/TeamCreateTool/UI.tsx +6 -0
  1422. package/src/tools/TeamCreateTool/constants.ts +1 -0
  1423. package/src/tools/TeamCreateTool/prompt.ts +113 -0
  1424. package/src/tools/TeamDeleteTool/TeamDeleteTool.ts +169 -0
  1425. package/src/tools/TeamDeleteTool/UI.tsx +20 -0
  1426. package/src/tools/TeamDeleteTool/constants.ts +1 -0
  1427. package/src/tools/TeamDeleteTool/prompt.ts +16 -0
  1428. package/src/tools/TerminalCaptureTool/TerminalCaptureTool.ts +2 -0
  1429. package/src/tools/TerminalCaptureTool/prompt.ts +2 -0
  1430. package/src/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
  1431. package/src/tools/TodoWriteTool/constants.ts +1 -0
  1432. package/src/tools/TodoWriteTool/prompt.ts +184 -0
  1433. package/src/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
  1434. package/src/tools/ToolSearchTool/constants.ts +1 -0
  1435. package/src/tools/ToolSearchTool/prompt.ts +121 -0
  1436. package/src/tools/TungstenTool/TungstenLiveMonitor.ts +4 -0
  1437. package/src/tools/TungstenTool/TungstenTool.ts +4 -0
  1438. package/src/tools/VerifyPlanExecutionTool/VerifyPlanExecutionTool.ts +2 -0
  1439. package/src/tools/VerifyPlanExecutionTool/constants.ts +2 -0
  1440. package/src/tools/WebBrowserTool/WebBrowserPanel.ts +2 -0
  1441. package/src/tools/WebBrowserTool/WebBrowserTool.ts +2 -0
  1442. package/src/tools/WebFetchTool/UI.tsx +72 -0
  1443. package/src/tools/WebFetchTool/WebFetchTool.ts +318 -0
  1444. package/src/tools/WebFetchTool/preapproved.ts +166 -0
  1445. package/src/tools/WebFetchTool/prompt.ts +46 -0
  1446. package/src/tools/WebFetchTool/utils.ts +530 -0
  1447. package/src/tools/WebSearchTool/UI.tsx +101 -0
  1448. package/src/tools/WebSearchTool/WebSearchTool.ts +435 -0
  1449. package/src/tools/WebSearchTool/prompt.ts +34 -0
  1450. package/src/tools/WorkflowTool/WorkflowPermissionRequest.ts +2 -0
  1451. package/src/tools/WorkflowTool/WorkflowTool.ts +2 -0
  1452. package/src/tools/WorkflowTool/bundled/index.ts +2 -0
  1453. package/src/tools/WorkflowTool/constants.ts +4 -0
  1454. package/src/tools/WorkflowTool/createWorkflowCommand.ts +2 -0
  1455. package/src/tools/shared/gitOperationTracking.ts +277 -0
  1456. package/src/tools/shared/spawnMultiAgent.ts +1093 -0
  1457. package/src/tools/testing/TestingPermissionTool.tsx +74 -0
  1458. package/src/tools/utils.ts +40 -0
  1459. package/src/tools.ts +395 -0
  1460. package/src/types/command.ts +216 -0
  1461. package/src/types/connectorText.ts +6 -0
  1462. package/src/types/fileSuggestion.ts +4 -0
  1463. package/src/types/generated/events_mono/claude_code/v1/claude_code_internal_event.ts +865 -0
  1464. package/src/types/generated/events_mono/common/v1/auth.ts +100 -0
  1465. package/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +223 -0
  1466. package/src/types/generated/google/protobuf/timestamp.ts +187 -0
  1467. package/src/types/hooks.ts +290 -0
  1468. package/src/types/ids.ts +44 -0
  1469. package/src/types/logs.ts +330 -0
  1470. package/src/types/message.ts +132 -0
  1471. package/src/types/messageQueueTypes.ts +5 -0
  1472. package/src/types/notebook.ts +10 -0
  1473. package/src/types/permissions.ts +441 -0
  1474. package/src/types/plugin.ts +363 -0
  1475. package/src/types/statusLine.ts +4 -0
  1476. package/src/types/textInputTypes.ts +387 -0
  1477. package/src/types/tools.ts +77 -0
  1478. package/src/types/utils.ts +5 -0
  1479. package/src/upstreamproxy/relay.ts +455 -0
  1480. package/src/upstreamproxy/upstreamproxy.ts +285 -0
  1481. package/src/utils/CircularBuffer.ts +84 -0
  1482. package/src/utils/Cursor.ts +1530 -0
  1483. package/src/utils/QueryGuard.ts +121 -0
  1484. package/src/utils/Shell.ts +474 -0
  1485. package/src/utils/ShellCommand.ts +465 -0
  1486. package/src/utils/abortController.ts +99 -0
  1487. package/src/utils/activityManager.ts +164 -0
  1488. package/src/utils/advisor.ts +145 -0
  1489. package/src/utils/agentContext.ts +178 -0
  1490. package/src/utils/agentId.ts +99 -0
  1491. package/src/utils/agentSwarmsEnabled.ts +44 -0
  1492. package/src/utils/agenticSessionSearch.ts +307 -0
  1493. package/src/utils/analyzeContext.ts +1382 -0
  1494. package/src/utils/ansiToPng.ts +334 -0
  1495. package/src/utils/ansiToSvg.ts +272 -0
  1496. package/src/utils/api.ts +718 -0
  1497. package/src/utils/apiPreconnect.ts +71 -0
  1498. package/src/utils/appleTerminalBackup.ts +124 -0
  1499. package/src/utils/argumentSubstitution.ts +145 -0
  1500. package/src/utils/array.ts +13 -0
  1501. package/src/utils/asciicast.ts +239 -0
  1502. package/src/utils/attachments.ts +3997 -0
  1503. package/src/utils/attribution.ts +393 -0
  1504. package/src/utils/auth.ts +2002 -0
  1505. package/src/utils/authFileDescriptor.ts +196 -0
  1506. package/src/utils/authPortable.ts +19 -0
  1507. package/src/utils/autoModeDenials.ts +26 -0
  1508. package/src/utils/autoRunIssue.tsx +122 -0
  1509. package/src/utils/autoUpdater.ts +561 -0
  1510. package/src/utils/aws.ts +74 -0
  1511. package/src/utils/awsAuthStatusManager.ts +81 -0
  1512. package/src/utils/background/remote/preconditions.ts +235 -0
  1513. package/src/utils/background/remote/remoteSession.ts +98 -0
  1514. package/src/utils/backgroundHousekeeping.ts +94 -0
  1515. package/src/utils/bash/ParsedCommand.ts +318 -0
  1516. package/src/utils/bash/ShellSnapshot.ts +582 -0
  1517. package/src/utils/bash/ast.ts +2679 -0
  1518. package/src/utils/bash/bashParser.ts +4436 -0
  1519. package/src/utils/bash/bashPipeCommand.ts +294 -0
  1520. package/src/utils/bash/commands.ts +1339 -0
  1521. package/src/utils/bash/heredoc.ts +733 -0
  1522. package/src/utils/bash/parser.ts +230 -0
  1523. package/src/utils/bash/prefix.ts +204 -0
  1524. package/src/utils/bash/registry.ts +53 -0
  1525. package/src/utils/bash/shellCompletion.ts +259 -0
  1526. package/src/utils/bash/shellPrefix.ts +28 -0
  1527. package/src/utils/bash/shellQuote.ts +304 -0
  1528. package/src/utils/bash/shellQuoting.ts +128 -0
  1529. package/src/utils/bash/specs/alias.ts +14 -0
  1530. package/src/utils/bash/specs/index.ts +18 -0
  1531. package/src/utils/bash/specs/nohup.ts +13 -0
  1532. package/src/utils/bash/specs/pyright.ts +91 -0
  1533. package/src/utils/bash/specs/sleep.ts +13 -0
  1534. package/src/utils/bash/specs/srun.ts +31 -0
  1535. package/src/utils/bash/specs/time.ts +13 -0
  1536. package/src/utils/bash/specs/timeout.ts +20 -0
  1537. package/src/utils/bash/treeSitterAnalysis.ts +506 -0
  1538. package/src/utils/betas.ts +449 -0
  1539. package/src/utils/billing.ts +78 -0
  1540. package/src/utils/binaryCheck.ts +53 -0
  1541. package/src/utils/browser.ts +68 -0
  1542. package/src/utils/bufferedWriter.ts +100 -0
  1543. package/src/utils/bundledMode.ts +22 -0
  1544. package/src/utils/caCerts.ts +115 -0
  1545. package/src/utils/caCertsConfig.ts +88 -0
  1546. package/src/utils/cachePaths.ts +38 -0
  1547. package/src/utils/classifierApprovals.ts +88 -0
  1548. package/src/utils/classifierApprovalsHook.ts +17 -0
  1549. package/src/utils/claudeCodeHints.ts +193 -0
  1550. package/src/utils/claudeDesktop.ts +152 -0
  1551. package/src/utils/claudeInChrome/chromeNativeHost.ts +527 -0
  1552. package/src/utils/claudeInChrome/common.ts +540 -0
  1553. package/src/utils/claudeInChrome/mcpServer.ts +293 -0
  1554. package/src/utils/claudeInChrome/prompt.ts +83 -0
  1555. package/src/utils/claudeInChrome/setup.ts +400 -0
  1556. package/src/utils/claudeInChrome/setupPortable.ts +233 -0
  1557. package/src/utils/claudeInChrome/toolRendering.tsx +262 -0
  1558. package/src/utils/claudemd.ts +1479 -0
  1559. package/src/utils/cleanup.ts +602 -0
  1560. package/src/utils/cleanupRegistry.ts +25 -0
  1561. package/src/utils/cliArgs.ts +60 -0
  1562. package/src/utils/cliHighlight.ts +54 -0
  1563. package/src/utils/codeIndexing.ts +206 -0
  1564. package/src/utils/collapseBackgroundBashNotifications.ts +84 -0
  1565. package/src/utils/collapseHookSummaries.ts +59 -0
  1566. package/src/utils/collapseReadSearch.ts +1109 -0
  1567. package/src/utils/collapseTeammateShutdowns.ts +55 -0
  1568. package/src/utils/combinedAbortSignal.ts +47 -0
  1569. package/src/utils/commandLifecycle.ts +21 -0
  1570. package/src/utils/commitAttribution.ts +961 -0
  1571. package/src/utils/completionCache.ts +166 -0
  1572. package/src/utils/computerUse/appNames.ts +196 -0
  1573. package/src/utils/computerUse/cleanup.ts +86 -0
  1574. package/src/utils/computerUse/common.ts +61 -0
  1575. package/src/utils/computerUse/computerUseLock.ts +215 -0
  1576. package/src/utils/computerUse/drainRunLoop.ts +79 -0
  1577. package/src/utils/computerUse/escHotkey.ts +54 -0
  1578. package/src/utils/computerUse/executor.ts +658 -0
  1579. package/src/utils/computerUse/gates.ts +72 -0
  1580. package/src/utils/computerUse/hostAdapter.ts +69 -0
  1581. package/src/utils/computerUse/inputLoader.ts +30 -0
  1582. package/src/utils/computerUse/mcpServer.ts +106 -0
  1583. package/src/utils/computerUse/setup.ts +53 -0
  1584. package/src/utils/computerUse/swiftLoader.ts +23 -0
  1585. package/src/utils/computerUse/toolRendering.tsx +125 -0
  1586. package/src/utils/computerUse/wrapper.tsx +336 -0
  1587. package/src/utils/concurrentSessions.ts +204 -0
  1588. package/src/utils/config.ts +1817 -0
  1589. package/src/utils/configConstants.ts +21 -0
  1590. package/src/utils/contentArray.ts +51 -0
  1591. package/src/utils/context.ts +221 -0
  1592. package/src/utils/contextAnalysis.ts +272 -0
  1593. package/src/utils/contextSuggestions.ts +235 -0
  1594. package/src/utils/controlMessageCompat.ts +32 -0
  1595. package/src/utils/conversationRecovery.ts +597 -0
  1596. package/src/utils/cron.ts +308 -0
  1597. package/src/utils/cronJitterConfig.ts +75 -0
  1598. package/src/utils/cronScheduler.ts +565 -0
  1599. package/src/utils/cronTasks.ts +458 -0
  1600. package/src/utils/cronTasksLock.ts +195 -0
  1601. package/src/utils/crossProjectResume.ts +75 -0
  1602. package/src/utils/crypto.ts +13 -0
  1603. package/src/utils/cwd.ts +32 -0
  1604. package/src/utils/debug.ts +268 -0
  1605. package/src/utils/debugFilter.ts +157 -0
  1606. package/src/utils/deepLink/banner.ts +123 -0
  1607. package/src/utils/deepLink/parseDeepLink.ts +170 -0
  1608. package/src/utils/deepLink/protocolHandler.ts +136 -0
  1609. package/src/utils/deepLink/registerProtocol.ts +348 -0
  1610. package/src/utils/deepLink/terminalLauncher.ts +557 -0
  1611. package/src/utils/deepLink/terminalPreference.ts +54 -0
  1612. package/src/utils/desktopDeepLink.ts +236 -0
  1613. package/src/utils/detectRepository.ts +178 -0
  1614. package/src/utils/diagLogs.ts +94 -0
  1615. package/src/utils/diff.ts +177 -0
  1616. package/src/utils/directMemberMessage.ts +69 -0
  1617. package/src/utils/displayTags.ts +51 -0
  1618. package/src/utils/doctorContextWarnings.ts +265 -0
  1619. package/src/utils/doctorDiagnostic.ts +625 -0
  1620. package/src/utils/dxt/helpers.ts +88 -0
  1621. package/src/utils/dxt/zip.ts +226 -0
  1622. package/src/utils/earlyInput.ts +191 -0
  1623. package/src/utils/editor.ts +183 -0
  1624. package/src/utils/effort.ts +333 -0
  1625. package/src/utils/embeddedTools.ts +29 -0
  1626. package/src/utils/env.ts +347 -0
  1627. package/src/utils/envDynamic.ts +151 -0
  1628. package/src/utils/envUtils.ts +192 -0
  1629. package/src/utils/envValidation.ts +38 -0
  1630. package/src/utils/errorLogSink.ts +235 -0
  1631. package/src/utils/errors.ts +238 -0
  1632. package/src/utils/exampleCommands.ts +184 -0
  1633. package/src/utils/execFileNoThrow.ts +150 -0
  1634. package/src/utils/execFileNoThrowPortable.ts +89 -0
  1635. package/src/utils/execSyncWrapper.ts +38 -0
  1636. package/src/utils/exportRenderer.tsx +98 -0
  1637. package/src/utils/extraUsage.ts +23 -0
  1638. package/src/utils/fastMode.ts +532 -0
  1639. package/src/utils/file.ts +584 -0
  1640. package/src/utils/fileHistory.ts +1115 -0
  1641. package/src/utils/fileOperationAnalytics.ts +71 -0
  1642. package/src/utils/filePersistence/filePersistence.ts +287 -0
  1643. package/src/utils/filePersistence/outputsScanner.ts +126 -0
  1644. package/src/utils/filePersistence/types.ts +10 -0
  1645. package/src/utils/fileRead.ts +102 -0
  1646. package/src/utils/fileReadCache.ts +96 -0
  1647. package/src/utils/fileStateCache.ts +142 -0
  1648. package/src/utils/findExecutable.ts +17 -0
  1649. package/src/utils/fingerprint.ts +76 -0
  1650. package/src/utils/forkedAgent.ts +689 -0
  1651. package/src/utils/format.ts +308 -0
  1652. package/src/utils/formatBriefTimestamp.ts +81 -0
  1653. package/src/utils/fpsTracker.ts +47 -0
  1654. package/src/utils/frontmatterParser.ts +370 -0
  1655. package/src/utils/fsOperations.ts +770 -0
  1656. package/src/utils/fullscreen.ts +202 -0
  1657. package/src/utils/generatedFiles.ts +136 -0
  1658. package/src/utils/generators.ts +88 -0
  1659. package/src/utils/genericProcessUtils.ts +184 -0
  1660. package/src/utils/getWorktreePaths.ts +70 -0
  1661. package/src/utils/getWorktreePathsPortable.ts +27 -0
  1662. package/src/utils/ghPrStatus.ts +106 -0
  1663. package/src/utils/git/gitConfigParser.ts +277 -0
  1664. package/src/utils/git/gitFilesystem.ts +699 -0
  1665. package/src/utils/git/gitignore.ts +99 -0
  1666. package/src/utils/git.ts +926 -0
  1667. package/src/utils/gitDiff.ts +532 -0
  1668. package/src/utils/gitSettings.ts +18 -0
  1669. package/src/utils/github/ghAuthStatus.ts +29 -0
  1670. package/src/utils/githubRepoPathMapping.ts +162 -0
  1671. package/src/utils/glob.ts +130 -0
  1672. package/src/utils/gracefulShutdown.ts +529 -0
  1673. package/src/utils/groupToolUses.ts +182 -0
  1674. package/src/utils/handlePromptSubmit.ts +610 -0
  1675. package/src/utils/hash.ts +46 -0
  1676. package/src/utils/headlessProfiler.ts +178 -0
  1677. package/src/utils/heapDumpService.ts +303 -0
  1678. package/src/utils/heatmap.ts +198 -0
  1679. package/src/utils/highlightMatch.tsx +28 -0
  1680. package/src/utils/hooks/AsyncHookRegistry.ts +309 -0
  1681. package/src/utils/hooks/apiQueryHookHelper.ts +141 -0
  1682. package/src/utils/hooks/execAgentHook.ts +339 -0
  1683. package/src/utils/hooks/execHttpHook.ts +242 -0
  1684. package/src/utils/hooks/execPromptHook.ts +211 -0
  1685. package/src/utils/hooks/fileChangedWatcher.ts +191 -0
  1686. package/src/utils/hooks/hookEvents.ts +192 -0
  1687. package/src/utils/hooks/hookHelpers.ts +83 -0
  1688. package/src/utils/hooks/hooksConfigManager.ts +400 -0
  1689. package/src/utils/hooks/hooksConfigSnapshot.ts +133 -0
  1690. package/src/utils/hooks/hooksSettings.ts +271 -0
  1691. package/src/utils/hooks/postSamplingHooks.ts +70 -0
  1692. package/src/utils/hooks/registerFrontmatterHooks.ts +67 -0
  1693. package/src/utils/hooks/registerSkillHooks.ts +64 -0
  1694. package/src/utils/hooks/sessionHooks.ts +447 -0
  1695. package/src/utils/hooks/skillImprovement.ts +267 -0
  1696. package/src/utils/hooks/ssrfGuard.ts +294 -0
  1697. package/src/utils/hooks.ts +5022 -0
  1698. package/src/utils/horizontalScroll.ts +137 -0
  1699. package/src/utils/http.ts +136 -0
  1700. package/src/utils/hyperlink.ts +39 -0
  1701. package/src/utils/iTermBackup.ts +73 -0
  1702. package/src/utils/ide.ts +1494 -0
  1703. package/src/utils/idePathConversion.ts +90 -0
  1704. package/src/utils/idleTimeout.ts +53 -0
  1705. package/src/utils/imagePaste.ts +416 -0
  1706. package/src/utils/imageResizer.ts +880 -0
  1707. package/src/utils/imageStore.ts +167 -0
  1708. package/src/utils/imageValidation.ts +104 -0
  1709. package/src/utils/immediateCommand.ts +15 -0
  1710. package/src/utils/inProcessTeammateHelpers.ts +102 -0
  1711. package/src/utils/ink.ts +26 -0
  1712. package/src/utils/intl.ts +94 -0
  1713. package/src/utils/jetbrains.ts +191 -0
  1714. package/src/utils/json.ts +277 -0
  1715. package/src/utils/jsonRead.ts +16 -0
  1716. package/src/utils/keyboardShortcuts.ts +14 -0
  1717. package/src/utils/lazySchema.ts +8 -0
  1718. package/src/utils/listSessionsImpl.ts +454 -0
  1719. package/src/utils/localInstaller.ts +162 -0
  1720. package/src/utils/lockfile.ts +43 -0
  1721. package/src/utils/log.ts +362 -0
  1722. package/src/utils/logoV2Utils.ts +373 -0
  1723. package/src/utils/mailbox.ts +73 -0
  1724. package/src/utils/managedEnv.ts +199 -0
  1725. package/src/utils/managedEnvConstants.ts +191 -0
  1726. package/src/utils/markdown.ts +381 -0
  1727. package/src/utils/markdownConfigLoader.ts +600 -0
  1728. package/src/utils/mcp/dateTimeParser.ts +121 -0
  1729. package/src/utils/mcp/elicitationValidation.ts +336 -0
  1730. package/src/utils/mcpInstructionsDelta.ts +130 -0
  1731. package/src/utils/mcpOutputStorage.ts +189 -0
  1732. package/src/utils/mcpValidation.ts +208 -0
  1733. package/src/utils/mcpWebSocketTransport.ts +200 -0
  1734. package/src/utils/memoize.ts +269 -0
  1735. package/src/utils/memory/types.ts +12 -0
  1736. package/src/utils/memory/versions.ts +8 -0
  1737. package/src/utils/memoryFileDetection.ts +289 -0
  1738. package/src/utils/messagePredicates.ts +8 -0
  1739. package/src/utils/messageQueueManager.ts +547 -0
  1740. package/src/utils/messages/mappers.ts +290 -0
  1741. package/src/utils/messages/systemInit.ts +96 -0
  1742. package/src/utils/messages.ts +5512 -0
  1743. package/src/utils/model/agent.ts +187 -0
  1744. package/src/utils/model/aliases.ts +25 -0
  1745. package/src/utils/model/antModels.ts +64 -0
  1746. package/src/utils/model/bedrock.ts +265 -0
  1747. package/src/utils/model/check1mAccess.ts +72 -0
  1748. package/src/utils/model/configs.ts +118 -0
  1749. package/src/utils/model/contextWindowUpgradeCheck.ts +47 -0
  1750. package/src/utils/model/deprecation.ts +101 -0
  1751. package/src/utils/model/model.ts +645 -0
  1752. package/src/utils/model/modelAllowlist.ts +192 -0
  1753. package/src/utils/model/modelCapabilities.ts +118 -0
  1754. package/src/utils/model/modelOptions.ts +554 -0
  1755. package/src/utils/model/modelStrings.ts +166 -0
  1756. package/src/utils/model/modelSupportOverrides.ts +50 -0
  1757. package/src/utils/model/providers.ts +42 -0
  1758. package/src/utils/model/thirdPartyProviders.ts +577 -0
  1759. package/src/utils/model/validateModel.ts +159 -0
  1760. package/src/utils/modelCost.ts +231 -0
  1761. package/src/utils/modifiers.ts +36 -0
  1762. package/src/utils/mtls.ts +179 -0
  1763. package/src/utils/nativeInstaller/download.ts +523 -0
  1764. package/src/utils/nativeInstaller/index.ts +18 -0
  1765. package/src/utils/nativeInstaller/installer.ts +1708 -0
  1766. package/src/utils/nativeInstaller/packageManagers.ts +336 -0
  1767. package/src/utils/nativeInstaller/pidLock.ts +433 -0
  1768. package/src/utils/notebook.ts +224 -0
  1769. package/src/utils/objectGroupBy.ts +18 -0
  1770. package/src/utils/pasteStore.ts +104 -0
  1771. package/src/utils/path.ts +155 -0
  1772. package/src/utils/pdf.ts +300 -0
  1773. package/src/utils/pdfUtils.ts +70 -0
  1774. package/src/utils/peerAddress.ts +21 -0
  1775. package/src/utils/permissions/PermissionMode.ts +141 -0
  1776. package/src/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
  1777. package/src/utils/permissions/PermissionResult.ts +35 -0
  1778. package/src/utils/permissions/PermissionRule.ts +40 -0
  1779. package/src/utils/permissions/PermissionUpdate.ts +389 -0
  1780. package/src/utils/permissions/PermissionUpdateSchema.ts +78 -0
  1781. package/src/utils/permissions/autoModeState.ts +39 -0
  1782. package/src/utils/permissions/bashClassifier.ts +61 -0
  1783. package/src/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
  1784. package/src/utils/permissions/classifierDecision.ts +98 -0
  1785. package/src/utils/permissions/classifierShared.ts +39 -0
  1786. package/src/utils/permissions/dangerousPatterns.ts +80 -0
  1787. package/src/utils/permissions/denialTracking.ts +45 -0
  1788. package/src/utils/permissions/filesystem.ts +1777 -0
  1789. package/src/utils/permissions/getNextPermissionMode.ts +101 -0
  1790. package/src/utils/permissions/pathValidation.ts +485 -0
  1791. package/src/utils/permissions/permissionExplainer.ts +250 -0
  1792. package/src/utils/permissions/permissionRuleParser.ts +198 -0
  1793. package/src/utils/permissions/permissionSetup.ts +1532 -0
  1794. package/src/utils/permissions/permissions.ts +1486 -0
  1795. package/src/utils/permissions/permissionsLoader.ts +296 -0
  1796. package/src/utils/permissions/shadowedRuleDetection.ts +234 -0
  1797. package/src/utils/permissions/shellRuleMatching.ts +228 -0
  1798. package/src/utils/permissions/yolo-classifier-prompts/auto_mode_system_prompt.txt +1 -0
  1799. package/src/utils/permissions/yolo-classifier-prompts/permissions_anthropic.txt +1 -0
  1800. package/src/utils/permissions/yolo-classifier-prompts/permissions_external.txt +1 -0
  1801. package/src/utils/permissions/yoloClassifier.ts +1495 -0
  1802. package/src/utils/planModeV2.ts +95 -0
  1803. package/src/utils/plans.ts +397 -0
  1804. package/src/utils/platform.ts +150 -0
  1805. package/src/utils/plugins/addDirPluginSettings.ts +71 -0
  1806. package/src/utils/plugins/cacheUtils.ts +196 -0
  1807. package/src/utils/plugins/dependencyResolver.ts +305 -0
  1808. package/src/utils/plugins/fetchTelemetry.ts +135 -0
  1809. package/src/utils/plugins/gitAvailability.ts +69 -0
  1810. package/src/utils/plugins/headlessPluginInstall.ts +174 -0
  1811. package/src/utils/plugins/hintRecommendation.ts +164 -0
  1812. package/src/utils/plugins/installCounts.ts +292 -0
  1813. package/src/utils/plugins/installedPluginsManager.ts +1268 -0
  1814. package/src/utils/plugins/loadPluginAgents.ts +348 -0
  1815. package/src/utils/plugins/loadPluginCommands.ts +946 -0
  1816. package/src/utils/plugins/loadPluginHooks.ts +287 -0
  1817. package/src/utils/plugins/loadPluginOutputStyles.ts +178 -0
  1818. package/src/utils/plugins/lspPluginIntegration.ts +387 -0
  1819. package/src/utils/plugins/lspRecommendation.ts +374 -0
  1820. package/src/utils/plugins/managedPlugins.ts +27 -0
  1821. package/src/utils/plugins/marketplaceHelpers.ts +592 -0
  1822. package/src/utils/plugins/marketplaceManager.ts +2643 -0
  1823. package/src/utils/plugins/mcpPluginIntegration.ts +634 -0
  1824. package/src/utils/plugins/mcpbHandler.ts +968 -0
  1825. package/src/utils/plugins/officialMarketplace.ts +25 -0
  1826. package/src/utils/plugins/officialMarketplaceGcs.ts +216 -0
  1827. package/src/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
  1828. package/src/utils/plugins/orphanedPluginFilter.ts +114 -0
  1829. package/src/utils/plugins/parseMarketplaceInput.ts +162 -0
  1830. package/src/utils/plugins/performStartupChecks.tsx +70 -0
  1831. package/src/utils/plugins/pluginAutoupdate.ts +284 -0
  1832. package/src/utils/plugins/pluginBlocklist.ts +127 -0
  1833. package/src/utils/plugins/pluginDirectories.ts +178 -0
  1834. package/src/utils/plugins/pluginFlagging.ts +208 -0
  1835. package/src/utils/plugins/pluginIdentifier.ts +123 -0
  1836. package/src/utils/plugins/pluginInstallationHelpers.ts +595 -0
  1837. package/src/utils/plugins/pluginLoader.ts +3302 -0
  1838. package/src/utils/plugins/pluginOptionsStorage.ts +400 -0
  1839. package/src/utils/plugins/pluginPolicy.ts +20 -0
  1840. package/src/utils/plugins/pluginStartupCheck.ts +341 -0
  1841. package/src/utils/plugins/pluginVersioning.ts +157 -0
  1842. package/src/utils/plugins/reconciler.ts +265 -0
  1843. package/src/utils/plugins/refresh.ts +215 -0
  1844. package/src/utils/plugins/schemas.ts +1681 -0
  1845. package/src/utils/plugins/validatePlugin.ts +903 -0
  1846. package/src/utils/plugins/walkPluginMarkdown.ts +69 -0
  1847. package/src/utils/plugins/zipCache.ts +406 -0
  1848. package/src/utils/plugins/zipCacheAdapters.ts +164 -0
  1849. package/src/utils/powershell/dangerousCmdlets.ts +185 -0
  1850. package/src/utils/powershell/parser.ts +1804 -0
  1851. package/src/utils/powershell/staticPrefix.ts +316 -0
  1852. package/src/utils/preflightChecks.tsx +151 -0
  1853. package/src/utils/privacyLevel.ts +55 -0
  1854. package/src/utils/process.ts +68 -0
  1855. package/src/utils/processUserInput/processBashCommand.tsx +140 -0
  1856. package/src/utils/processUserInput/processSlashCommand.tsx +922 -0
  1857. package/src/utils/processUserInput/processTextPrompt.ts +100 -0
  1858. package/src/utils/processUserInput/processUserInput.ts +605 -0
  1859. package/src/utils/profilerBase.ts +46 -0
  1860. package/src/utils/promptCategory.ts +49 -0
  1861. package/src/utils/promptEditor.ts +188 -0
  1862. package/src/utils/promptShellExecution.ts +183 -0
  1863. package/src/utils/protectedNamespace.ts +2 -0
  1864. package/src/utils/proxy.ts +426 -0
  1865. package/src/utils/queryContext.ts +179 -0
  1866. package/src/utils/queryHelpers.ts +552 -0
  1867. package/src/utils/queryProfiler.ts +301 -0
  1868. package/src/utils/queueProcessor.ts +95 -0
  1869. package/src/utils/readEditContext.ts +227 -0
  1870. package/src/utils/readFileInRange.ts +383 -0
  1871. package/src/utils/releaseNotes.ts +360 -0
  1872. package/src/utils/renderOptions.ts +77 -0
  1873. package/src/utils/ripgrep.ts +679 -0
  1874. package/src/utils/sandbox/sandbox-adapter.ts +985 -0
  1875. package/src/utils/sandbox/sandbox-ui-utils.ts +12 -0
  1876. package/src/utils/sanitization.ts +91 -0
  1877. package/src/utils/screenshotClipboard.ts +121 -0
  1878. package/src/utils/sdkEventQueue.ts +134 -0
  1879. package/src/utils/secureStorage/fallbackStorage.ts +70 -0
  1880. package/src/utils/secureStorage/index.ts +17 -0
  1881. package/src/utils/secureStorage/keychainPrefetch.ts +116 -0
  1882. package/src/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
  1883. package/src/utils/secureStorage/macOsKeychainStorage.ts +231 -0
  1884. package/src/utils/secureStorage/plainTextStorage.ts +84 -0
  1885. package/src/utils/secureStorage/types.ts +5 -0
  1886. package/src/utils/semanticBoolean.ts +29 -0
  1887. package/src/utils/semanticNumber.ts +36 -0
  1888. package/src/utils/semver.ts +59 -0
  1889. package/src/utils/sequential.ts +56 -0
  1890. package/src/utils/sessionActivity.ts +133 -0
  1891. package/src/utils/sessionEnvVars.ts +22 -0
  1892. package/src/utils/sessionEnvironment.ts +166 -0
  1893. package/src/utils/sessionFileAccessHooks.ts +250 -0
  1894. package/src/utils/sessionIngressAuth.ts +140 -0
  1895. package/src/utils/sessionRestore.ts +551 -0
  1896. package/src/utils/sessionStart.ts +232 -0
  1897. package/src/utils/sessionState.ts +150 -0
  1898. package/src/utils/sessionStorage.ts +5105 -0
  1899. package/src/utils/sessionStoragePortable.ts +793 -0
  1900. package/src/utils/sessionTitle.ts +129 -0
  1901. package/src/utils/sessionUrl.ts +64 -0
  1902. package/src/utils/set.ts +53 -0
  1903. package/src/utils/settings/allErrors.ts +32 -0
  1904. package/src/utils/settings/applySettingsChange.ts +92 -0
  1905. package/src/utils/settings/changeDetector.ts +488 -0
  1906. package/src/utils/settings/constants.ts +202 -0
  1907. package/src/utils/settings/internalWrites.ts +37 -0
  1908. package/src/utils/settings/managedPath.ts +34 -0
  1909. package/src/utils/settings/mdm/constants.ts +81 -0
  1910. package/src/utils/settings/mdm/rawRead.ts +130 -0
  1911. package/src/utils/settings/mdm/settings.ts +316 -0
  1912. package/src/utils/settings/permissionValidation.ts +262 -0
  1913. package/src/utils/settings/pluginOnlyPolicy.ts +60 -0
  1914. package/src/utils/settings/schemaOutput.ts +8 -0
  1915. package/src/utils/settings/settings.ts +1015 -0
  1916. package/src/utils/settings/settingsCache.ts +80 -0
  1917. package/src/utils/settings/toolValidationConfig.ts +103 -0
  1918. package/src/utils/settings/types.ts +1150 -0
  1919. package/src/utils/settings/validateEditTool.ts +45 -0
  1920. package/src/utils/settings/validation.ts +265 -0
  1921. package/src/utils/settings/validationTips.ts +164 -0
  1922. package/src/utils/shell/bashProvider.ts +255 -0
  1923. package/src/utils/shell/outputLimits.ts +14 -0
  1924. package/src/utils/shell/powershellDetection.ts +107 -0
  1925. package/src/utils/shell/powershellProvider.ts +123 -0
  1926. package/src/utils/shell/prefix.ts +367 -0
  1927. package/src/utils/shell/readOnlyCommandValidation.ts +1893 -0
  1928. package/src/utils/shell/resolveDefaultShell.ts +14 -0
  1929. package/src/utils/shell/shellProvider.ts +33 -0
  1930. package/src/utils/shell/shellToolUtils.ts +22 -0
  1931. package/src/utils/shell/specPrefix.ts +241 -0
  1932. package/src/utils/shellConfig.ts +167 -0
  1933. package/src/utils/sideQuery.ts +222 -0
  1934. package/src/utils/sideQuestion.ts +155 -0
  1935. package/src/utils/signal.ts +43 -0
  1936. package/src/utils/sinks.ts +16 -0
  1937. package/src/utils/skills/skillChangeDetector.ts +311 -0
  1938. package/src/utils/slashCommandParsing.ts +60 -0
  1939. package/src/utils/sleep.ts +84 -0
  1940. package/src/utils/sliceAnsi.ts +91 -0
  1941. package/src/utils/slowOperations.ts +286 -0
  1942. package/src/utils/standaloneAgent.ts +23 -0
  1943. package/src/utils/startupProfiler.ts +194 -0
  1944. package/src/utils/staticRender.tsx +116 -0
  1945. package/src/utils/stats.ts +1061 -0
  1946. package/src/utils/statsCache.ts +434 -0
  1947. package/src/utils/status.tsx +362 -0
  1948. package/src/utils/statusNoticeDefinitions.tsx +198 -0
  1949. package/src/utils/statusNoticeHelpers.ts +20 -0
  1950. package/src/utils/stream.ts +76 -0
  1951. package/src/utils/streamJsonStdoutGuard.ts +123 -0
  1952. package/src/utils/streamlinedTransform.ts +201 -0
  1953. package/src/utils/stringUtils.ts +235 -0
  1954. package/src/utils/subprocessEnv.ts +99 -0
  1955. package/src/utils/suggestions/commandSuggestions.ts +567 -0
  1956. package/src/utils/suggestions/directoryCompletion.ts +263 -0
  1957. package/src/utils/suggestions/shellHistoryCompletion.ts +119 -0
  1958. package/src/utils/suggestions/skillUsageTracking.ts +55 -0
  1959. package/src/utils/suggestions/slackChannelSuggestions.ts +209 -0
  1960. package/src/utils/swarm/It2SetupPrompt.tsx +380 -0
  1961. package/src/utils/swarm/backends/ITermBackend.ts +370 -0
  1962. package/src/utils/swarm/backends/InProcessBackend.ts +339 -0
  1963. package/src/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
  1964. package/src/utils/swarm/backends/TmuxBackend.ts +764 -0
  1965. package/src/utils/swarm/backends/detection.ts +128 -0
  1966. package/src/utils/swarm/backends/it2Setup.ts +245 -0
  1967. package/src/utils/swarm/backends/registry.ts +464 -0
  1968. package/src/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
  1969. package/src/utils/swarm/backends/types.ts +311 -0
  1970. package/src/utils/swarm/constants.ts +33 -0
  1971. package/src/utils/swarm/inProcessRunner.ts +1571 -0
  1972. package/src/utils/swarm/leaderPermissionBridge.ts +54 -0
  1973. package/src/utils/swarm/permissionSync.ts +928 -0
  1974. package/src/utils/swarm/reconnection.ts +119 -0
  1975. package/src/utils/swarm/spawnInProcess.ts +328 -0
  1976. package/src/utils/swarm/spawnUtils.ts +146 -0
  1977. package/src/utils/swarm/teamHelpers.ts +683 -0
  1978. package/src/utils/swarm/teammateInit.ts +129 -0
  1979. package/src/utils/swarm/teammateLayoutManager.ts +107 -0
  1980. package/src/utils/swarm/teammateModel.ts +10 -0
  1981. package/src/utils/swarm/teammatePromptAddendum.ts +18 -0
  1982. package/src/utils/systemDirectories.ts +74 -0
  1983. package/src/utils/systemPrompt.ts +123 -0
  1984. package/src/utils/systemPromptType.ts +14 -0
  1985. package/src/utils/systemTheme.ts +119 -0
  1986. package/src/utils/taggedId.ts +54 -0
  1987. package/src/utils/task/TaskOutput.ts +390 -0
  1988. package/src/utils/task/diskOutput.ts +451 -0
  1989. package/src/utils/task/framework.ts +308 -0
  1990. package/src/utils/task/outputFormatting.ts +38 -0
  1991. package/src/utils/task/sdkProgress.ts +36 -0
  1992. package/src/utils/taskSummary.ts +2 -0
  1993. package/src/utils/tasks.ts +862 -0
  1994. package/src/utils/teamDiscovery.ts +81 -0
  1995. package/src/utils/teamMemoryOps.ts +88 -0
  1996. package/src/utils/teammate.ts +292 -0
  1997. package/src/utils/teammateContext.ts +96 -0
  1998. package/src/utils/teammateMailbox.ts +1183 -0
  1999. package/src/utils/telemetry/betaSessionTracing.ts +491 -0
  2000. package/src/utils/telemetry/bigqueryExporter.ts +252 -0
  2001. package/src/utils/telemetry/events.ts +75 -0
  2002. package/src/utils/telemetry/instrumentation.ts +825 -0
  2003. package/src/utils/telemetry/logger.ts +26 -0
  2004. package/src/utils/telemetry/perfettoTracing.ts +1120 -0
  2005. package/src/utils/telemetry/pluginTelemetry.ts +289 -0
  2006. package/src/utils/telemetry/sessionTracing.ts +927 -0
  2007. package/src/utils/telemetry/skillLoadedEvent.ts +39 -0
  2008. package/src/utils/telemetryAttributes.ts +71 -0
  2009. package/src/utils/teleport/api.ts +466 -0
  2010. package/src/utils/teleport/environmentSelection.ts +77 -0
  2011. package/src/utils/teleport/environments.ts +120 -0
  2012. package/src/utils/teleport/gitBundle.ts +292 -0
  2013. package/src/utils/teleport.tsx +1226 -0
  2014. package/src/utils/tempfile.ts +31 -0
  2015. package/src/utils/terminal.ts +131 -0
  2016. package/src/utils/terminalPanel.ts +191 -0
  2017. package/src/utils/textHighlighting.ts +166 -0
  2018. package/src/utils/theme.ts +639 -0
  2019. package/src/utils/thinking.ts +171 -0
  2020. package/src/utils/timeouts.ts +39 -0
  2021. package/src/utils/tmuxSocket.ts +427 -0
  2022. package/src/utils/todo/types.ts +18 -0
  2023. package/src/utils/tokenBudget.ts +73 -0
  2024. package/src/utils/tokens.ts +261 -0
  2025. package/src/utils/toolErrors.ts +132 -0
  2026. package/src/utils/toolPool.ts +79 -0
  2027. package/src/utils/toolResultStorage.ts +1040 -0
  2028. package/src/utils/toolSchemaCache.ts +26 -0
  2029. package/src/utils/toolSearch.ts +756 -0
  2030. package/src/utils/transcriptSearch.ts +202 -0
  2031. package/src/utils/treeify.ts +170 -0
  2032. package/src/utils/truncate.ts +179 -0
  2033. package/src/utils/udsClient.ts +2 -0
  2034. package/src/utils/udsMessaging.ts +2 -0
  2035. package/src/utils/ultraplan/ccrSession.ts +349 -0
  2036. package/src/utils/ultraplan/keyword.ts +127 -0
  2037. package/src/utils/ultraplan/prompt.txt +1 -0
  2038. package/src/utils/unaryLogging.ts +39 -0
  2039. package/src/utils/undercover.ts +89 -0
  2040. package/src/utils/user.ts +194 -0
  2041. package/src/utils/userAgent.ts +10 -0
  2042. package/src/utils/userPromptKeywords.ts +27 -0
  2043. package/src/utils/uuid.ts +27 -0
  2044. package/src/utils/warningHandler.ts +121 -0
  2045. package/src/utils/which.ts +82 -0
  2046. package/src/utils/windowsPaths.ts +173 -0
  2047. package/src/utils/withResolvers.ts +13 -0
  2048. package/src/utils/words.ts +800 -0
  2049. package/src/utils/workloadContext.ts +57 -0
  2050. package/src/utils/worktree.ts +1519 -0
  2051. package/src/utils/worktreeModeEnabled.ts +11 -0
  2052. package/src/utils/xdg.ts +65 -0
  2053. package/src/utils/xml.ts +16 -0
  2054. package/src/utils/yaml.ts +15 -0
  2055. package/src/utils/zodToJsonSchema.ts +23 -0
  2056. package/src/vim/motions.ts +82 -0
  2057. package/src/vim/operators.ts +556 -0
  2058. package/src/vim/textObjects.ts +186 -0
  2059. package/src/vim/transitions.ts +490 -0
  2060. package/src/vim/types.ts +199 -0
  2061. package/src/voice/voiceModeEnabled.ts +54 -0
  2062. package/start.sh +19 -0
  2063. package/stubs/@ant/claude-for-chrome-mcp/index.d.ts +1 -0
  2064. package/stubs/@ant/claude-for-chrome-mcp/index.js +1 -0
  2065. package/stubs/@ant/claude-for-chrome-mcp/package.json +1 -0
  2066. package/stubs/@ant/computer-use-input/index.d.ts +1 -0
  2067. package/stubs/@ant/computer-use-input/index.js +1 -0
  2068. package/stubs/@ant/computer-use-input/package.json +1 -0
  2069. package/stubs/@ant/computer-use-mcp/index.d.ts +1 -0
  2070. package/stubs/@ant/computer-use-mcp/index.js +1 -0
  2071. package/stubs/@ant/computer-use-mcp/package.json +1 -0
  2072. package/stubs/@ant/computer-use-mcp/sentinelApps/index.d.ts +1 -0
  2073. package/stubs/@ant/computer-use-mcp/sentinelApps/index.js +1 -0
  2074. package/stubs/@ant/computer-use-mcp/types.d.ts +1 -0
  2075. package/stubs/@ant/computer-use-mcp/types.js +1 -0
  2076. package/stubs/@ant/computer-use-swift/index.d.ts +1 -0
  2077. package/stubs/@ant/computer-use-swift/index.js +1 -0
  2078. package/stubs/@ant/computer-use-swift/package.json +1 -0
  2079. package/stubs/@anthropic-ai/mcpb/index.d.ts +1 -0
  2080. package/stubs/@anthropic-ai/mcpb/index.js +7 -0
  2081. package/stubs/@anthropic-ai/mcpb/package.json +1 -0
  2082. package/stubs/@anthropic-ai/sandbox-runtime/index.d.ts +1 -0
  2083. package/stubs/@anthropic-ai/sandbox-runtime/index.js +49 -0
  2084. package/stubs/@anthropic-ai/sandbox-runtime/package.json +1 -0
  2085. package/stubs/color-diff-napi/index.d.ts +2 -0
  2086. package/stubs/color-diff-napi/index.js +1 -0
  2087. package/stubs/color-diff-napi/package.json +1 -0
  2088. package/stubs/modifiers-napi/index.js +10 -0
  2089. package/stubs/modifiers-napi/package.json +5 -0
  2090. package/tsconfig.json +23 -0
@@ -0,0 +1,3450 @@
1
+ import type {
2
+ BetaContentBlock,
3
+ BetaContentBlockParam,
4
+ BetaImageBlockParam,
5
+ BetaJSONOutputFormat,
6
+ BetaMessage,
7
+ BetaMessageDeltaUsage,
8
+ BetaMessageStreamParams,
9
+ BetaOutputConfig,
10
+ BetaRawMessageStreamEvent,
11
+ BetaRequestDocumentBlock,
12
+ BetaStopReason,
13
+ BetaToolChoiceAuto,
14
+ BetaToolChoiceTool,
15
+ BetaToolResultBlockParam,
16
+ BetaToolUnion,
17
+ BetaUsage,
18
+ BetaMessageParam as MessageParam,
19
+ } from '@anthropic-ai/sdk/resources/beta/messages/messages.mjs'
20
+ import type { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
21
+ import type { Stream } from '@anthropic-ai/sdk/streaming.mjs'
22
+ import { randomUUID } from 'crypto'
23
+ import {
24
+ getAPIProvider,
25
+ isFirstPartyAnthropicBaseUrl,
26
+ } from 'src/utils/model/providers.js'
27
+ import {
28
+ getAttributionHeader,
29
+ getCLISyspromptPrefix,
30
+ } from '../../constants/system.js'
31
+ import {
32
+ getEmptyToolPermissionContext,
33
+ type QueryChainTracking,
34
+ type Tool,
35
+ type ToolPermissionContext,
36
+ type Tools,
37
+ toolMatchesName,
38
+ } from '../../Tool.js'
39
+ import type { AgentDefinition } from '../../tools/AgentTool/loadAgentsDir.js'
40
+ import {
41
+ type ConnectorTextBlock,
42
+ type ConnectorTextDelta,
43
+ isConnectorTextBlock,
44
+ } from '../../types/connectorText.js'
45
+ import type {
46
+ AssistantMessage,
47
+ Message,
48
+ StreamEvent,
49
+ SystemAPIErrorMessage,
50
+ UserMessage,
51
+ } from '../../types/message.js'
52
+ import {
53
+ type CacheScope,
54
+ logAPIPrefix,
55
+ splitSysPromptPrefix,
56
+ toolToAPISchema,
57
+ } from '../../utils/api.js'
58
+ import { getOauthAccountInfo } from '../../utils/auth.js'
59
+ import {
60
+ getBedrockExtraBodyParamsBetas,
61
+ getMergedBetas,
62
+ getModelBetas,
63
+ } from '../../utils/betas.js'
64
+ import { getOrCreateUserID } from '../../utils/config.js'
65
+ import {
66
+ CAPPED_DEFAULT_MAX_TOKENS,
67
+ getModelMaxOutputTokens,
68
+ getSonnet1mExpTreatmentEnabled,
69
+ } from '../../utils/context.js'
70
+ import { resolveAppliedEffort } from '../../utils/effort.js'
71
+ import { isEnvTruthy } from '../../utils/envUtils.js'
72
+ import { errorMessage } from '../../utils/errors.js'
73
+ import { computeFingerprintFromMessages } from '../../utils/fingerprint.js'
74
+ import { captureAPIRequest, logError } from '../../utils/log.js'
75
+ import {
76
+ createAssistantAPIErrorMessage,
77
+ createUserMessage,
78
+ ensureToolResultPairing,
79
+ normalizeContentFromAPI,
80
+ normalizeMessagesForAPI,
81
+ stripAdvisorBlocks,
82
+ stripCallerFieldFromAssistantMessage,
83
+ stripToolReferenceBlocksFromUserMessage,
84
+ } from '../../utils/messages.js'
85
+ import {
86
+ getDefaultOpusModel,
87
+ getDefaultSonnetModel,
88
+ getSmallFastModel,
89
+ isNonCustomOpusModel,
90
+ } from '../../utils/model/model.js'
91
+ import {
92
+ asSystemPrompt,
93
+ type SystemPrompt,
94
+ } from '../../utils/systemPromptType.js'
95
+ import { tokenCountFromLastAPIResponse } from '../../utils/tokens.js'
96
+ import { getDynamicConfig_BLOCKS_ON_INIT } from '../analytics/growthbook.js'
97
+ import {
98
+ currentLimits,
99
+ extractQuotaStatusFromError,
100
+ extractQuotaStatusFromHeaders,
101
+ } from '../claudeAiLimits.js'
102
+ import { getAPIContextManagement } from '../compact/apiMicrocompact.js'
103
+
104
+ /* eslint-disable @typescript-eslint/no-require-imports */
105
+ const autoModeStateModule = feature('TRANSCRIPT_CLASSIFIER')
106
+ ? (require('../../utils/permissions/autoModeState.js') as typeof import('../../utils/permissions/autoModeState.js'))
107
+ : null
108
+
109
+ import { feature } from 'bun:bundle'
110
+ import type { ClientOptions } from '@anthropic-ai/sdk'
111
+ import {
112
+ APIConnectionTimeoutError,
113
+ APIError,
114
+ APIUserAbortError,
115
+ } from '@anthropic-ai/sdk/error'
116
+ import {
117
+ getAfkModeHeaderLatched,
118
+ getCacheEditingHeaderLatched,
119
+ getFastModeHeaderLatched,
120
+ getLastApiCompletionTimestamp,
121
+ getPromptCache1hAllowlist,
122
+ getPromptCache1hEligible,
123
+ getSessionId,
124
+ getThinkingClearLatched,
125
+ setAfkModeHeaderLatched,
126
+ setCacheEditingHeaderLatched,
127
+ setFastModeHeaderLatched,
128
+ setLastMainRequestId,
129
+ setPromptCache1hAllowlist,
130
+ setPromptCache1hEligible,
131
+ setThinkingClearLatched,
132
+ } from 'src/bootstrap/state.js'
133
+ import {
134
+ AFK_MODE_BETA_HEADER,
135
+ CONTEXT_1M_BETA_HEADER,
136
+ CONTEXT_MANAGEMENT_BETA_HEADER,
137
+ EFFORT_BETA_HEADER,
138
+ FAST_MODE_BETA_HEADER,
139
+ PROMPT_CACHING_SCOPE_BETA_HEADER,
140
+ REDACT_THINKING_BETA_HEADER,
141
+ STRUCTURED_OUTPUTS_BETA_HEADER,
142
+ TASK_BUDGETS_BETA_HEADER,
143
+ } from 'src/constants/betas.js'
144
+ import type { QuerySource } from 'src/constants/querySource.js'
145
+ import type { Notification } from 'src/context/notifications.js'
146
+ import { addToTotalSessionCost } from 'src/cost-tracker.js'
147
+ import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'
148
+ import type { AgentId } from 'src/types/ids.js'
149
+ import {
150
+ ADVISOR_TOOL_INSTRUCTIONS,
151
+ getExperimentAdvisorModels,
152
+ isAdvisorEnabled,
153
+ isValidAdvisorModel,
154
+ modelSupportsAdvisor,
155
+ } from 'src/utils/advisor.js'
156
+ import { getAgentContext } from 'src/utils/agentContext.js'
157
+ import { isClaudeAISubscriber } from 'src/utils/auth.js'
158
+ import {
159
+ getToolSearchBetaHeader,
160
+ modelSupportsStructuredOutputs,
161
+ shouldIncludeFirstPartyOnlyBetas,
162
+ shouldUseGlobalCacheScope,
163
+ } from 'src/utils/betas.js'
164
+ import { CLAUDE_IN_CHROME_MCP_SERVER_NAME } from 'src/utils/claudeInChrome/common.js'
165
+ import { CHROME_TOOL_SEARCH_INSTRUCTIONS } from 'src/utils/claudeInChrome/prompt.js'
166
+ import { getMaxThinkingTokensForModel } from 'src/utils/context.js'
167
+ import { logForDebugging } from 'src/utils/debug.js'
168
+ import { logForDiagnosticsNoPII } from 'src/utils/diagLogs.js'
169
+ import { type EffortValue, modelSupportsEffort } from 'src/utils/effort.js'
170
+ import {
171
+ isFastModeAvailable,
172
+ isFastModeCooldown,
173
+ isFastModeEnabled,
174
+ isFastModeSupportedByModel,
175
+ } from 'src/utils/fastMode.js'
176
+ import { returnValue } from 'src/utils/generators.js'
177
+ import { headlessProfilerCheckpoint } from 'src/utils/headlessProfiler.js'
178
+ import { isMcpInstructionsDeltaEnabled } from 'src/utils/mcpInstructionsDelta.js'
179
+ import { calculateUSDCost } from 'src/utils/modelCost.js'
180
+ import { endQueryProfile, queryCheckpoint } from 'src/utils/queryProfiler.js'
181
+ import {
182
+ modelSupportsAdaptiveThinking,
183
+ modelSupportsThinking,
184
+ type ThinkingConfig,
185
+ } from 'src/utils/thinking.js'
186
+ import {
187
+ extractDiscoveredToolNames,
188
+ isDeferredToolsDeltaEnabled,
189
+ isToolSearchEnabled,
190
+ } from 'src/utils/toolSearch.js'
191
+ import { API_MAX_MEDIA_PER_REQUEST } from '../../constants/apiLimits.js'
192
+ import { ADVISOR_BETA_HEADER } from '../../constants/betas.js'
193
+ import {
194
+ formatDeferredToolLine,
195
+ isDeferredTool,
196
+ TOOL_SEARCH_TOOL_NAME,
197
+ } from '../../tools/ToolSearchTool/prompt.js'
198
+ import { count } from '../../utils/array.js'
199
+ import { insertBlockAfterToolResults } from '../../utils/contentArray.js'
200
+ import { validateBoundedIntEnvVar } from '../../utils/envValidation.js'
201
+ import { safeParseJSON } from '../../utils/json.js'
202
+ import { getInferenceProfileBackingModel } from '../../utils/model/bedrock.js'
203
+ import {
204
+ normalizeModelStringForAPI,
205
+ parseUserSpecifiedModel,
206
+ } from '../../utils/model/model.js'
207
+ import {
208
+ startSessionActivity,
209
+ stopSessionActivity,
210
+ } from '../../utils/sessionActivity.js'
211
+ import { jsonStringify } from '../../utils/slowOperations.js'
212
+ import {
213
+ isBetaTracingEnabled,
214
+ type LLMRequestNewContext,
215
+ startLLMRequestSpan,
216
+ } from '../../utils/telemetry/sessionTracing.js'
217
+ /* eslint-enable @typescript-eslint/no-require-imports */
218
+ import {
219
+ type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
220
+ logEvent,
221
+ } from '../analytics/index.js'
222
+ import {
223
+ consumePendingCacheEdits,
224
+ getPinnedCacheEdits,
225
+ markToolsSentToAPIState,
226
+ pinCacheEdits,
227
+ } from '../compact/microCompact.js'
228
+ import { getInitializationStatus } from '../lsp/manager.js'
229
+ import { isToolFromMcpServer } from '../mcp/utils.js'
230
+ import { withStreamingVCR, withVCR } from '../vcr.js'
231
+ import {
232
+ CLIENT_REQUEST_ID_HEADER,
233
+ getAnthropicClient,
234
+ getAnthropicClientForModel,
235
+ isThirdPartyModel,
236
+ getActualModelId,
237
+ } from './client.js'
238
+ import {
239
+ API_ERROR_MESSAGE_PREFIX,
240
+ CUSTOM_OFF_SWITCH_MESSAGE,
241
+ getAssistantMessageFromError,
242
+ getErrorMessageIfRefusal,
243
+ } from './errors.js'
244
+ import {
245
+ EMPTY_USAGE,
246
+ type GlobalCacheStrategy,
247
+ logAPIError,
248
+ logAPIQuery,
249
+ logAPISuccessAndDuration,
250
+ type NonNullableUsage,
251
+ } from './logging.js'
252
+ import {
253
+ CACHE_TTL_1HOUR_MS,
254
+ checkResponseForCacheBreak,
255
+ recordPromptState,
256
+ } from './promptCacheBreakDetection.js'
257
+ import {
258
+ CannotRetryError,
259
+ FallbackTriggeredError,
260
+ is529Error,
261
+ type RetryContext,
262
+ withRetry,
263
+ } from './withRetry.js'
264
+
265
+ // Define a type that represents valid JSON values
266
+ type JsonValue = string | number | boolean | null | JsonObject | JsonArray
267
+ type JsonObject = { [key: string]: JsonValue }
268
+ type JsonArray = JsonValue[]
269
+
270
+ /**
271
+ * Assemble the extra body parameters for the API request, based on the
272
+ * CLAUDE_CODE_EXTRA_BODY environment variable if present and on any beta
273
+ * headers (primarily for Bedrock requests).
274
+ *
275
+ * @param betaHeaders - An array of beta headers to include in the request.
276
+ * @returns A JSON object representing the extra body parameters.
277
+ */
278
+ export function getExtraBodyParams(betaHeaders?: string[]): JsonObject {
279
+ // Parse user's extra body parameters first
280
+ const extraBodyStr = process.env.CLAUDE_CODE_EXTRA_BODY
281
+ let result: JsonObject = {}
282
+
283
+ if (extraBodyStr) {
284
+ try {
285
+ // Parse as JSON, which can be null, boolean, number, string, array or object
286
+ const parsed = safeParseJSON(extraBodyStr)
287
+ // We expect an object with key-value pairs to spread into API parameters
288
+ if (parsed && typeof parsed === 'object' && !Array.isArray(parsed)) {
289
+ // Shallow clone — safeParseJSON is LRU-cached and returns the same
290
+ // object reference for the same string. Mutating `result` below
291
+ // would poison the cache, causing stale values to persist.
292
+ result = { ...(parsed as JsonObject) }
293
+ } else {
294
+ logForDebugging(
295
+ `CLAUDE_CODE_EXTRA_BODY env var must be a JSON object, but was given ${extraBodyStr}`,
296
+ { level: 'error' },
297
+ )
298
+ }
299
+ } catch (error) {
300
+ logForDebugging(
301
+ `Error parsing CLAUDE_CODE_EXTRA_BODY: ${errorMessage(error)}`,
302
+ { level: 'error' },
303
+ )
304
+ }
305
+ }
306
+
307
+ // Anti-distillation: send fake_tools opt-in for 1P CLI only
308
+ if (
309
+ feature('ANTI_DISTILLATION_CC')
310
+ ? process.env.CLAUDE_CODE_ENTRYPOINT === 'cli' &&
311
+ shouldIncludeFirstPartyOnlyBetas() &&
312
+ getFeatureValue_CACHED_MAY_BE_STALE(
313
+ 'tengu_anti_distill_fake_tool_injection',
314
+ false,
315
+ )
316
+ : false
317
+ ) {
318
+ result.anti_distillation = ['fake_tools']
319
+ }
320
+
321
+ // Handle beta headers if provided
322
+ if (betaHeaders && betaHeaders.length > 0) {
323
+ if (result.anthropic_beta && Array.isArray(result.anthropic_beta)) {
324
+ // Add to existing array, avoiding duplicates
325
+ const existingHeaders = result.anthropic_beta as string[]
326
+ const newHeaders = betaHeaders.filter(
327
+ header => !existingHeaders.includes(header),
328
+ )
329
+ result.anthropic_beta = [...existingHeaders, ...newHeaders]
330
+ } else {
331
+ // Create new array with the beta headers
332
+ result.anthropic_beta = betaHeaders
333
+ }
334
+ }
335
+
336
+ return result
337
+ }
338
+
339
+ export function getPromptCachingEnabled(model: string): boolean {
340
+ // Third-party provider models (e.g. minimax:..., glm:...) do not support
341
+ // the cache_control field in message content blocks.
342
+ if (isThirdPartyModel(model)) return false
343
+
344
+ // Global disable takes precedence
345
+ if (isEnvTruthy(process.env.DISABLE_PROMPT_CACHING)) return false
346
+
347
+ // Check if we should disable for small/fast model
348
+ if (isEnvTruthy(process.env.DISABLE_PROMPT_CACHING_HAIKU)) {
349
+ const smallFastModel = getSmallFastModel()
350
+ if (model === smallFastModel) return false
351
+ }
352
+
353
+ // Check if we should disable for default Sonnet
354
+ if (isEnvTruthy(process.env.DISABLE_PROMPT_CACHING_SONNET)) {
355
+ const defaultSonnet = getDefaultSonnetModel()
356
+ if (model === defaultSonnet) return false
357
+ }
358
+
359
+ // Check if we should disable for default Opus
360
+ if (isEnvTruthy(process.env.DISABLE_PROMPT_CACHING_OPUS)) {
361
+ const defaultOpus = getDefaultOpusModel()
362
+ if (model === defaultOpus) return false
363
+ }
364
+
365
+ return true
366
+ }
367
+
368
+ export function getCacheControl({
369
+ scope,
370
+ querySource,
371
+ }: {
372
+ scope?: CacheScope
373
+ querySource?: QuerySource
374
+ } = {}): {
375
+ type: 'ephemeral'
376
+ ttl?: '1h'
377
+ scope?: CacheScope
378
+ } {
379
+ return {
380
+ type: 'ephemeral',
381
+ ...(should1hCacheTTL(querySource) && { ttl: '1h' }),
382
+ ...(scope === 'global' && { scope }),
383
+ }
384
+ }
385
+
386
+ /**
387
+ * Determines if 1h TTL should be used for prompt caching.
388
+ *
389
+ * Only applied when:
390
+ * 1. User is eligible (ant or subscriber within rate limits)
391
+ * 2. The query source matches a pattern in the GrowthBook allowlist
392
+ *
393
+ * GrowthBook config shape: { allowlist: string[] }
394
+ * Patterns support trailing '*' for prefix matching.
395
+ * Examples:
396
+ * - { allowlist: ["repl_main_thread*", "sdk"] } — main thread + SDK only
397
+ * - { allowlist: ["repl_main_thread*", "sdk", "agent:*"] } — also subagents
398
+ * - { allowlist: ["*"] } — all sources
399
+ *
400
+ * The allowlist is cached in STATE for session stability — prevents mixed
401
+ * TTLs when GrowthBook's disk cache updates mid-request.
402
+ */
403
+ function should1hCacheTTL(querySource?: QuerySource): boolean {
404
+ // 3P Bedrock users get 1h TTL when opted in via env var — they manage their own billing
405
+ // No GrowthBook gating needed since 3P users don't have GrowthBook configured
406
+ if (
407
+ getAPIProvider() === 'bedrock' &&
408
+ isEnvTruthy(process.env.ENABLE_PROMPT_CACHING_1H_BEDROCK)
409
+ ) {
410
+ return true
411
+ }
412
+
413
+ // Latch eligibility in bootstrap state for session stability — prevents
414
+ // mid-session overage flips from changing the cache_control TTL, which
415
+ // would bust the server-side prompt cache (~20K tokens per flip).
416
+ let userEligible = getPromptCache1hEligible()
417
+ if (userEligible === null) {
418
+ userEligible =
419
+ process.env.USER_TYPE === 'ant' ||
420
+ (isClaudeAISubscriber() && !currentLimits.isUsingOverage)
421
+ setPromptCache1hEligible(userEligible)
422
+ }
423
+ if (!userEligible) return false
424
+
425
+ // Cache allowlist in bootstrap state for session stability — prevents mixed
426
+ // TTLs when GrowthBook's disk cache updates mid-request
427
+ let allowlist = getPromptCache1hAllowlist()
428
+ if (allowlist === null) {
429
+ const config = getFeatureValue_CACHED_MAY_BE_STALE<{
430
+ allowlist?: string[]
431
+ }>('tengu_prompt_cache_1h_config', {})
432
+ allowlist = config.allowlist ?? []
433
+ setPromptCache1hAllowlist(allowlist)
434
+ }
435
+
436
+ return (
437
+ querySource !== undefined &&
438
+ allowlist.some(pattern =>
439
+ pattern.endsWith('*')
440
+ ? querySource.startsWith(pattern.slice(0, -1))
441
+ : querySource === pattern,
442
+ )
443
+ )
444
+ }
445
+
446
+ /**
447
+ * Configure effort parameters for API request.
448
+ *
449
+ */
450
+ function configureEffortParams(
451
+ effortValue: EffortValue | undefined,
452
+ outputConfig: BetaOutputConfig,
453
+ extraBodyParams: Record<string, unknown>,
454
+ betas: string[],
455
+ model: string,
456
+ ): void {
457
+ if (!modelSupportsEffort(model) || 'effort' in outputConfig) {
458
+ return
459
+ }
460
+
461
+ if (effortValue === undefined) {
462
+ betas.push(EFFORT_BETA_HEADER)
463
+ } else if (typeof effortValue === 'string') {
464
+ // Send string effort level as is
465
+ outputConfig.effort = effortValue
466
+ betas.push(EFFORT_BETA_HEADER)
467
+ } else if (process.env.USER_TYPE === 'ant') {
468
+ // Numeric effort override - ant-only (uses anthropic_internal)
469
+ const existingInternal =
470
+ (extraBodyParams.anthropic_internal as Record<string, unknown>) || {}
471
+ extraBodyParams.anthropic_internal = {
472
+ ...existingInternal,
473
+ effort_override: effortValue,
474
+ }
475
+ }
476
+ }
477
+
478
+ // output_config.task_budget — API-side token budget awareness for the model.
479
+ // Stainless SDK types don't yet include task_budget on BetaOutputConfig, so we
480
+ // define the wire shape locally and cast. The API validates on receipt; see
481
+ // api/api/schemas/messages/request/output_config.py:12-39 in the monorepo.
482
+ // Beta: task-budgets-2026-03-13 (EAP, claude-strudel-eap only as of Mar 2026).
483
+ type TaskBudgetParam = {
484
+ type: 'tokens'
485
+ total: number
486
+ remaining?: number
487
+ }
488
+
489
+ export function configureTaskBudgetParams(
490
+ taskBudget: Options['taskBudget'],
491
+ outputConfig: BetaOutputConfig & { task_budget?: TaskBudgetParam },
492
+ betas: string[],
493
+ ): void {
494
+ if (
495
+ !taskBudget ||
496
+ 'task_budget' in outputConfig ||
497
+ !shouldIncludeFirstPartyOnlyBetas()
498
+ ) {
499
+ return
500
+ }
501
+ outputConfig.task_budget = {
502
+ type: 'tokens',
503
+ total: taskBudget.total,
504
+ ...(taskBudget.remaining !== undefined && {
505
+ remaining: taskBudget.remaining,
506
+ }),
507
+ }
508
+ if (!betas.includes(TASK_BUDGETS_BETA_HEADER)) {
509
+ betas.push(TASK_BUDGETS_BETA_HEADER)
510
+ }
511
+ }
512
+
513
+ export function getAPIMetadata() {
514
+ // https://docs.google.com/document/d/1dURO9ycXXQCBS0V4Vhl4poDBRgkelFc5t2BNPoEgH5Q/edit?tab=t.0#heading=h.5g7nec5b09w5
515
+ let extra: JsonObject = {}
516
+ const extraStr = process.env.CLAUDE_CODE_EXTRA_METADATA
517
+ if (extraStr) {
518
+ const parsed = safeParseJSON(extraStr, false)
519
+ if (parsed && typeof parsed === 'object' && !Array.isArray(parsed)) {
520
+ extra = parsed as JsonObject
521
+ } else {
522
+ logForDebugging(
523
+ `CLAUDE_CODE_EXTRA_METADATA env var must be a JSON object, but was given ${extraStr}`,
524
+ { level: 'error' },
525
+ )
526
+ }
527
+ }
528
+
529
+ return {
530
+ user_id: jsonStringify({
531
+ ...extra,
532
+ device_id: getOrCreateUserID(),
533
+ // Only include OAuth account UUID when actively using OAuth authentication
534
+ account_uuid: getOauthAccountInfo()?.accountUuid ?? '',
535
+ session_id: getSessionId(),
536
+ }),
537
+ }
538
+ }
539
+
540
+ export async function verifyApiKey(
541
+ apiKey: string,
542
+ isNonInteractiveSession: boolean,
543
+ ): Promise<boolean> {
544
+ // Skip API verification if running in print mode (isNonInteractiveSession)
545
+ if (isNonInteractiveSession) {
546
+ return true
547
+ }
548
+
549
+ try {
550
+ // WARNING: if you change this to use a non-Haiku model, this request will fail in 1P unless it uses getCLISyspromptPrefix.
551
+ const model = getSmallFastModel()
552
+ const betas = getModelBetas(model)
553
+ return await returnValue(
554
+ withRetry(
555
+ () =>
556
+ getAnthropicClient({
557
+ apiKey,
558
+ maxRetries: 3,
559
+ model,
560
+ source: 'verify_api_key',
561
+ }),
562
+ async anthropic => {
563
+ const messages: MessageParam[] = [{ role: 'user', content: 'test' }]
564
+ // biome-ignore lint/plugin: API key verification is intentionally a minimal direct call
565
+ await anthropic.beta.messages.create({
566
+ model,
567
+ max_tokens: 1,
568
+ messages,
569
+ temperature: 1,
570
+ ...(betas.length > 0 && { betas }),
571
+ metadata: getAPIMetadata(),
572
+ ...getExtraBodyParams(),
573
+ })
574
+ return true
575
+ },
576
+ { maxRetries: 2, model, thinkingConfig: { type: 'disabled' } }, // Use fewer retries for API key verification
577
+ ),
578
+ )
579
+ } catch (errorFromRetry) {
580
+ let error = errorFromRetry
581
+ if (errorFromRetry instanceof CannotRetryError) {
582
+ error = errorFromRetry.originalError
583
+ }
584
+ logError(error)
585
+ // Check for authentication error
586
+ if (
587
+ error instanceof Error &&
588
+ error.message.includes(
589
+ '{"type":"error","error":{"type":"authentication_error","message":"invalid x-api-key"}}',
590
+ )
591
+ ) {
592
+ return false
593
+ }
594
+ throw error
595
+ }
596
+ }
597
+
598
+ export function userMessageToMessageParam(
599
+ message: UserMessage,
600
+ addCache = false,
601
+ enablePromptCaching: boolean,
602
+ querySource?: QuerySource,
603
+ ): MessageParam {
604
+ if (addCache) {
605
+ if (typeof message.message.content === 'string') {
606
+ return {
607
+ role: 'user',
608
+ content: [
609
+ {
610
+ type: 'text',
611
+ text: message.message.content,
612
+ ...(enablePromptCaching && {
613
+ cache_control: getCacheControl({ querySource }),
614
+ }),
615
+ },
616
+ ],
617
+ }
618
+ } else {
619
+ return {
620
+ role: 'user',
621
+ content: message.message.content.map((_, i) => ({
622
+ ..._,
623
+ ...(i === message.message.content.length - 1
624
+ ? enablePromptCaching
625
+ ? { cache_control: getCacheControl({ querySource }) }
626
+ : {}
627
+ : {}),
628
+ })),
629
+ }
630
+ }
631
+ }
632
+ // Clone array content to prevent in-place mutations (e.g., insertCacheEditsBlock's
633
+ // splice) from contaminating the original message. Without cloning, multiple calls
634
+ // to addCacheBreakpoints share the same array and each splices in duplicate cache_edits.
635
+ return {
636
+ role: 'user',
637
+ content: Array.isArray(message.message.content)
638
+ ? [...message.message.content]
639
+ : message.message.content,
640
+ }
641
+ }
642
+
643
+ export function assistantMessageToMessageParam(
644
+ message: AssistantMessage,
645
+ addCache = false,
646
+ enablePromptCaching: boolean,
647
+ querySource?: QuerySource,
648
+ ): MessageParam {
649
+ if (addCache) {
650
+ if (typeof message.message.content === 'string') {
651
+ return {
652
+ role: 'assistant',
653
+ content: [
654
+ {
655
+ type: 'text',
656
+ text: message.message.content,
657
+ ...(enablePromptCaching && {
658
+ cache_control: getCacheControl({ querySource }),
659
+ }),
660
+ },
661
+ ],
662
+ }
663
+ } else {
664
+ return {
665
+ role: 'assistant',
666
+ content: message.message.content.map((_, i) => ({
667
+ ..._,
668
+ ...(i === message.message.content.length - 1 &&
669
+ _.type !== 'thinking' &&
670
+ _.type !== 'redacted_thinking' &&
671
+ (feature('CONNECTOR_TEXT') ? !isConnectorTextBlock(_) : true)
672
+ ? enablePromptCaching
673
+ ? { cache_control: getCacheControl({ querySource }) }
674
+ : {}
675
+ : {}),
676
+ })),
677
+ }
678
+ }
679
+ }
680
+ return {
681
+ role: 'assistant',
682
+ content: message.message.content,
683
+ }
684
+ }
685
+
686
+ export type Options = {
687
+ getToolPermissionContext: () => Promise<ToolPermissionContext>
688
+ model: string
689
+ toolChoice?: BetaToolChoiceTool | BetaToolChoiceAuto | undefined
690
+ isNonInteractiveSession: boolean
691
+ extraToolSchemas?: BetaToolUnion[]
692
+ maxOutputTokensOverride?: number
693
+ fallbackModel?: string
694
+ onStreamingFallback?: () => void
695
+ querySource: QuerySource
696
+ agents: AgentDefinition[]
697
+ allowedAgentTypes?: string[]
698
+ hasAppendSystemPrompt: boolean
699
+ fetchOverride?: ClientOptions['fetch']
700
+ enablePromptCaching?: boolean
701
+ skipCacheWrite?: boolean
702
+ temperatureOverride?: number
703
+ effortValue?: EffortValue
704
+ mcpTools: Tools
705
+ hasPendingMcpServers?: boolean
706
+ queryTracking?: QueryChainTracking
707
+ agentId?: AgentId // Only set for subagents
708
+ outputFormat?: BetaJSONOutputFormat
709
+ fastMode?: boolean
710
+ advisorModel?: string
711
+ addNotification?: (notif: Notification) => void
712
+ // API-side task budget (output_config.task_budget). Distinct from the
713
+ // tokenBudget.ts +500k auto-continue feature — this one is sent to the API
714
+ // so the model can pace itself. `remaining` is computed by the caller
715
+ // (query.ts decrements across the agentic loop).
716
+ taskBudget?: { total: number; remaining?: number }
717
+ }
718
+
719
+ export async function queryModelWithoutStreaming({
720
+ messages,
721
+ systemPrompt,
722
+ thinkingConfig,
723
+ tools,
724
+ signal,
725
+ options,
726
+ }: {
727
+ messages: Message[]
728
+ systemPrompt: SystemPrompt
729
+ thinkingConfig: ThinkingConfig
730
+ tools: Tools
731
+ signal: AbortSignal
732
+ options: Options
733
+ }): Promise<AssistantMessage> {
734
+ // Store the assistant message but continue consuming the generator to ensure
735
+ // logAPISuccessAndDuration gets called (which happens after all yields)
736
+ let assistantMessage: AssistantMessage | undefined
737
+ for await (const message of withStreamingVCR(messages, async function* () {
738
+ yield* queryModel(
739
+ messages,
740
+ systemPrompt,
741
+ thinkingConfig,
742
+ tools,
743
+ signal,
744
+ options,
745
+ )
746
+ })) {
747
+ if (message.type === 'assistant') {
748
+ assistantMessage = message
749
+ }
750
+ }
751
+ if (!assistantMessage) {
752
+ // If the signal was aborted, throw APIUserAbortError instead of a generic error
753
+ // This allows callers to handle abort scenarios gracefully
754
+ if (signal.aborted) {
755
+ throw new APIUserAbortError()
756
+ }
757
+ throw new Error('No assistant message found')
758
+ }
759
+ return assistantMessage
760
+ }
761
+
762
+ export async function* queryModelWithStreaming({
763
+ messages,
764
+ systemPrompt,
765
+ thinkingConfig,
766
+ tools,
767
+ signal,
768
+ options,
769
+ }: {
770
+ messages: Message[]
771
+ systemPrompt: SystemPrompt
772
+ thinkingConfig: ThinkingConfig
773
+ tools: Tools
774
+ signal: AbortSignal
775
+ options: Options
776
+ }): AsyncGenerator<
777
+ StreamEvent | AssistantMessage | SystemAPIErrorMessage,
778
+ void
779
+ > {
780
+ return yield* withStreamingVCR(messages, async function* () {
781
+ yield* queryModel(
782
+ messages,
783
+ systemPrompt,
784
+ thinkingConfig,
785
+ tools,
786
+ signal,
787
+ options,
788
+ )
789
+ })
790
+ }
791
+
792
+ /**
793
+ * Determines if an LSP tool should be deferred (tool appears with defer_loading: true)
794
+ * because LSP initialization is not yet complete.
795
+ */
796
+ function shouldDeferLspTool(tool: Tool): boolean {
797
+ if (!('isLsp' in tool) || !tool.isLsp) {
798
+ return false
799
+ }
800
+ const status = getInitializationStatus()
801
+ // Defer when pending or not started
802
+ return status.status === 'pending' || status.status === 'not-started'
803
+ }
804
+
805
+ /**
806
+ * Per-attempt timeout for non-streaming fallback requests, in milliseconds.
807
+ * Reads API_TIMEOUT_MS when set so slow backends and the streaming path
808
+ * share the same ceiling.
809
+ *
810
+ * Remote sessions default to 120s to stay under CCR's container idle-kill
811
+ * (~5min) so a hung fallback to a wedged backend surfaces a clean
812
+ * APIConnectionTimeoutError instead of stalling past SIGKILL.
813
+ *
814
+ * Otherwise defaults to 300s — long enough for slow backends without
815
+ * approaching the API's 10-minute non-streaming boundary.
816
+ */
817
+ function getNonstreamingFallbackTimeoutMs(): number {
818
+ const override = parseInt(process.env.API_TIMEOUT_MS || '', 10)
819
+ if (override) return override
820
+ return isEnvTruthy(process.env.CLAUDE_CODE_REMOTE) ? 120_000 : 300_000
821
+ }
822
+
823
+ /**
824
+ * Helper generator for non-streaming API requests.
825
+ * Encapsulates the common pattern of creating a withRetry generator,
826
+ * iterating to yield system messages, and returning the final BetaMessage.
827
+ */
828
+ export async function* executeNonStreamingRequest(
829
+ clientOptions: {
830
+ model: string
831
+ fetchOverride?: Options['fetchOverride']
832
+ source: string
833
+ },
834
+ retryOptions: {
835
+ model: string
836
+ fallbackModel?: string
837
+ thinkingConfig: ThinkingConfig
838
+ fastMode?: boolean
839
+ signal: AbortSignal
840
+ initialConsecutive529Errors?: number
841
+ querySource?: QuerySource
842
+ },
843
+ paramsFromContext: (context: RetryContext) => BetaMessageStreamParams,
844
+ onAttempt: (attempt: number, start: number, maxOutputTokens: number) => void,
845
+ captureRequest: (params: BetaMessageStreamParams) => void,
846
+ /**
847
+ * Request ID of the failed streaming attempt this fallback is recovering
848
+ * from. Emitted in tengu_nonstreaming_fallback_error for funnel correlation.
849
+ */
850
+ originatingRequestId?: string | null,
851
+ ): AsyncGenerator<SystemAPIErrorMessage, BetaMessage> {
852
+ const fallbackTimeoutMs = getNonstreamingFallbackTimeoutMs()
853
+ const generator = withRetry(
854
+ async () => {
855
+ if (isThirdPartyModel(clientOptions.model)) {
856
+ const result = await getAnthropicClientForModel({
857
+ model: clientOptions.model,
858
+ maxRetries: 0,
859
+ fetchOverride: clientOptions.fetchOverride,
860
+ source: clientOptions.source,
861
+ })
862
+ return result.client
863
+ }
864
+ return getAnthropicClient({
865
+ maxRetries: 0,
866
+ model: clientOptions.model,
867
+ fetchOverride: clientOptions.fetchOverride,
868
+ source: clientOptions.source,
869
+ })
870
+ },
871
+ async (anthropic, attempt, context) => {
872
+ const start = Date.now()
873
+ const retryParams = paramsFromContext(context)
874
+ captureRequest(retryParams)
875
+ onAttempt(attempt, start, retryParams.max_tokens)
876
+
877
+ const adjustedParams = adjustParamsForNonStreaming(
878
+ retryParams,
879
+ MAX_NON_STREAMING_TOKENS,
880
+ )
881
+
882
+ try {
883
+ // biome-ignore lint/plugin: non-streaming API call
884
+ return await anthropic.beta.messages.create(
885
+ {
886
+ ...adjustedParams,
887
+ model: normalizeModelStringForAPI(adjustedParams.model),
888
+ },
889
+ {
890
+ signal: retryOptions.signal,
891
+ timeout: fallbackTimeoutMs,
892
+ },
893
+ )
894
+ } catch (err) {
895
+ // User aborts are not errors — re-throw immediately without logging
896
+ if (err instanceof APIUserAbortError) throw err
897
+
898
+ // Instrumentation: record when the non-streaming request errors (including
899
+ // timeouts). Lets us distinguish "fallback hung past container kill"
900
+ // (no event) from "fallback hit the bounded timeout" (this event).
901
+ logForDiagnosticsNoPII('error', 'cli_nonstreaming_fallback_error')
902
+ logEvent('tengu_nonstreaming_fallback_error', {
903
+ model:
904
+ clientOptions.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
905
+ error:
906
+ err instanceof Error
907
+ ? (err.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
908
+ : ('unknown' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS),
909
+ attempt,
910
+ timeout_ms: fallbackTimeoutMs,
911
+ request_id: (originatingRequestId ??
912
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
913
+ })
914
+ throw err
915
+ }
916
+ },
917
+ {
918
+ model: retryOptions.model,
919
+ fallbackModel: retryOptions.fallbackModel,
920
+ thinkingConfig: retryOptions.thinkingConfig,
921
+ ...(isFastModeEnabled() && { fastMode: retryOptions.fastMode }),
922
+ signal: retryOptions.signal,
923
+ initialConsecutive529Errors: retryOptions.initialConsecutive529Errors,
924
+ querySource: retryOptions.querySource,
925
+ },
926
+ )
927
+
928
+ let e
929
+ do {
930
+ e = await generator.next()
931
+ if (!e.done && e.value.type === 'system') {
932
+ yield e.value
933
+ }
934
+ } while (!e.done)
935
+
936
+ return e.value as BetaMessage
937
+ }
938
+
939
+ /**
940
+ * Extracts the request ID from the most recent assistant message in the
941
+ * conversation. Used to link consecutive API requests in analytics so we can
942
+ * join them for cache-hit-rate analysis and incremental token tracking.
943
+ *
944
+ * Deriving this from the message array (rather than global state) ensures each
945
+ * query chain (main thread, subagent, teammate) tracks its own request chain
946
+ * independently, and rollback/undo naturally updates the value.
947
+ */
948
+ function getPreviousRequestIdFromMessages(
949
+ messages: Message[],
950
+ ): string | undefined {
951
+ for (let i = messages.length - 1; i >= 0; i--) {
952
+ const msg = messages[i]!
953
+ if (msg.type === 'assistant' && msg.requestId) {
954
+ return msg.requestId
955
+ }
956
+ }
957
+ return undefined
958
+ }
959
+
960
+ function isMedia(
961
+ block: BetaContentBlockParam,
962
+ ): block is BetaImageBlockParam | BetaRequestDocumentBlock {
963
+ return block.type === 'image' || block.type === 'document'
964
+ }
965
+
966
+ function isToolResult(
967
+ block: BetaContentBlockParam,
968
+ ): block is BetaToolResultBlockParam {
969
+ return block.type === 'tool_result'
970
+ }
971
+
972
+ /**
973
+ * Ensures messages contain at most `limit` media items (images + documents).
974
+ * Strips oldest media first to preserve the most recent.
975
+ */
976
+ export function stripExcessMediaItems(
977
+ messages: (UserMessage | AssistantMessage)[],
978
+ limit: number,
979
+ ): (UserMessage | AssistantMessage)[] {
980
+ let toRemove = 0
981
+ for (const msg of messages) {
982
+ if (!Array.isArray(msg.message.content)) continue
983
+ for (const block of msg.message.content) {
984
+ if (isMedia(block)) toRemove++
985
+ if (isToolResult(block) && Array.isArray(block.content)) {
986
+ for (const nested of block.content) {
987
+ if (isMedia(nested)) toRemove++
988
+ }
989
+ }
990
+ }
991
+ }
992
+ toRemove -= limit
993
+ if (toRemove <= 0) return messages
994
+
995
+ return messages.map(msg => {
996
+ if (toRemove <= 0) return msg
997
+ const content = msg.message.content
998
+ if (!Array.isArray(content)) return msg
999
+
1000
+ const before = toRemove
1001
+ const stripped = content
1002
+ .map(block => {
1003
+ if (
1004
+ toRemove <= 0 ||
1005
+ !isToolResult(block) ||
1006
+ !Array.isArray(block.content)
1007
+ )
1008
+ return block
1009
+ const filtered = block.content.filter(n => {
1010
+ if (toRemove > 0 && isMedia(n)) {
1011
+ toRemove--
1012
+ return false
1013
+ }
1014
+ return true
1015
+ })
1016
+ return filtered.length === block.content.length
1017
+ ? block
1018
+ : { ...block, content: filtered }
1019
+ })
1020
+ .filter(block => {
1021
+ if (toRemove > 0 && isMedia(block)) {
1022
+ toRemove--
1023
+ return false
1024
+ }
1025
+ return true
1026
+ })
1027
+
1028
+ return before === toRemove
1029
+ ? msg
1030
+ : {
1031
+ ...msg,
1032
+ message: { ...msg.message, content: stripped },
1033
+ }
1034
+ }) as (UserMessage | AssistantMessage)[]
1035
+ }
1036
+
1037
+ async function* queryModel(
1038
+ messages: Message[],
1039
+ systemPrompt: SystemPrompt,
1040
+ thinkingConfig: ThinkingConfig,
1041
+ tools: Tools,
1042
+ signal: AbortSignal,
1043
+ options: Options,
1044
+ ): AsyncGenerator<
1045
+ StreamEvent | AssistantMessage | SystemAPIErrorMessage,
1046
+ void
1047
+ > {
1048
+ // Check cheap conditions first — the off-switch await blocks on GrowthBook
1049
+ // init (~10ms). For non-Opus models (haiku, sonnet) this skips the await
1050
+ // entirely. Subscribers don't hit this path at all.
1051
+ if (
1052
+ !isClaudeAISubscriber() &&
1053
+ isNonCustomOpusModel(options.model) &&
1054
+ (
1055
+ await getDynamicConfig_BLOCKS_ON_INIT<{ activated: boolean }>(
1056
+ 'tengu-off-switch',
1057
+ {
1058
+ activated: false,
1059
+ },
1060
+ )
1061
+ ).activated
1062
+ ) {
1063
+ logEvent('tengu_off_switch_query', {})
1064
+ yield getAssistantMessageFromError(
1065
+ new Error(CUSTOM_OFF_SWITCH_MESSAGE),
1066
+ options.model,
1067
+ )
1068
+ return
1069
+ }
1070
+
1071
+ // Derive previous request ID from the last assistant message in this query chain.
1072
+ // This is scoped per message array (main thread, subagent, teammate each have their own),
1073
+ // so concurrent agents don't clobber each other's request chain tracking.
1074
+ // Also naturally handles rollback/undo since removed messages won't be in the array.
1075
+ const previousRequestId = getPreviousRequestIdFromMessages(messages)
1076
+
1077
+ const resolvedModel =
1078
+ getAPIProvider() === 'bedrock' &&
1079
+ options.model.includes('application-inference-profile')
1080
+ ? ((await getInferenceProfileBackingModel(options.model)) ??
1081
+ options.model)
1082
+ : options.model
1083
+
1084
+ queryCheckpoint('query_tool_schema_build_start')
1085
+ const isAgenticQuery =
1086
+ options.querySource.startsWith('repl_main_thread') ||
1087
+ options.querySource.startsWith('agent:') ||
1088
+ options.querySource === 'sdk' ||
1089
+ options.querySource === 'hook_agent' ||
1090
+ options.querySource === 'verification_agent'
1091
+ const betas = getMergedBetas(options.model, { isAgenticQuery })
1092
+
1093
+ // Always send the advisor beta header when advisor is enabled, so
1094
+ // non-agentic queries (compact, side_question, extract_memories, etc.)
1095
+ // can parse advisor server_tool_use blocks already in the conversation history.
1096
+ if (isAdvisorEnabled()) {
1097
+ betas.push(ADVISOR_BETA_HEADER)
1098
+ }
1099
+
1100
+ let advisorModel: string | undefined
1101
+ if (isAgenticQuery && isAdvisorEnabled()) {
1102
+ let advisorOption = options.advisorModel
1103
+
1104
+ const advisorExperiment = getExperimentAdvisorModels()
1105
+ if (advisorExperiment !== undefined) {
1106
+ if (
1107
+ normalizeModelStringForAPI(advisorExperiment.baseModel) ===
1108
+ normalizeModelStringForAPI(options.model)
1109
+ ) {
1110
+ // Override the advisor model if the base model matches. We
1111
+ // should only have experiment models if the user cannot
1112
+ // configure it themselves.
1113
+ advisorOption = advisorExperiment.advisorModel
1114
+ }
1115
+ }
1116
+
1117
+ if (advisorOption) {
1118
+ const normalizedAdvisorModel = normalizeModelStringForAPI(
1119
+ parseUserSpecifiedModel(advisorOption),
1120
+ )
1121
+ if (!modelSupportsAdvisor(options.model)) {
1122
+ logForDebugging(
1123
+ `[AdvisorTool] Skipping advisor - base model ${options.model} does not support advisor`,
1124
+ )
1125
+ } else if (!isValidAdvisorModel(normalizedAdvisorModel)) {
1126
+ logForDebugging(
1127
+ `[AdvisorTool] Skipping advisor - ${normalizedAdvisorModel} is not a valid advisor model`,
1128
+ )
1129
+ } else {
1130
+ advisorModel = normalizedAdvisorModel
1131
+ logForDebugging(
1132
+ `[AdvisorTool] Server-side tool enabled with ${advisorModel} as the advisor model`,
1133
+ )
1134
+ }
1135
+ }
1136
+ }
1137
+
1138
+ // Check if tool search is enabled (checks mode, model support, and threshold for auto mode)
1139
+ // This is async because it may need to calculate MCP tool description sizes for TstAuto mode
1140
+ let useToolSearch = await isToolSearchEnabled(
1141
+ options.model,
1142
+ tools,
1143
+ options.getToolPermissionContext,
1144
+ options.agents,
1145
+ 'query',
1146
+ )
1147
+
1148
+ // Precompute once — isDeferredTool does 2 GrowthBook lookups per call
1149
+ const deferredToolNames = new Set<string>()
1150
+ if (useToolSearch) {
1151
+ for (const t of tools) {
1152
+ if (isDeferredTool(t)) deferredToolNames.add(t.name)
1153
+ }
1154
+ }
1155
+
1156
+ // Even if tool search mode is enabled, skip if there are no deferred tools
1157
+ // AND no MCP servers are still connecting. When servers are pending, keep
1158
+ // ToolSearch available so the model can discover tools after they connect.
1159
+ if (
1160
+ useToolSearch &&
1161
+ deferredToolNames.size === 0 &&
1162
+ !options.hasPendingMcpServers
1163
+ ) {
1164
+ logForDebugging(
1165
+ 'Tool search disabled: no deferred tools available to search',
1166
+ )
1167
+ useToolSearch = false
1168
+ }
1169
+
1170
+ // Filter out ToolSearchTool if tool search is not enabled for this model
1171
+ // ToolSearchTool returns tool_reference blocks which unsupported models can't handle
1172
+ let filteredTools: Tools
1173
+
1174
+ if (useToolSearch) {
1175
+ // Dynamic tool loading: Only include deferred tools that have been discovered
1176
+ // via tool_reference blocks in the message history. This eliminates the need
1177
+ // to predeclare all deferred tools upfront and removes limits on tool quantity.
1178
+ const discoveredToolNames = extractDiscoveredToolNames(messages)
1179
+
1180
+ filteredTools = tools.filter(tool => {
1181
+ // Always include non-deferred tools
1182
+ if (!deferredToolNames.has(tool.name)) return true
1183
+ // Always include ToolSearchTool (so it can discover more tools)
1184
+ if (toolMatchesName(tool, TOOL_SEARCH_TOOL_NAME)) return true
1185
+ // Only include deferred tools that have been discovered
1186
+ return discoveredToolNames.has(tool.name)
1187
+ })
1188
+ } else {
1189
+ filteredTools = tools.filter(
1190
+ t => !toolMatchesName(t, TOOL_SEARCH_TOOL_NAME),
1191
+ )
1192
+ }
1193
+
1194
+ // Add tool search beta header if enabled - required for defer_loading to be accepted
1195
+ // Header differs by provider: 1P/Foundry use advanced-tool-use, Vertex/Bedrock use tool-search-tool
1196
+ // For Bedrock, this header must go in extraBodyParams, not the betas array
1197
+ const toolSearchHeader = useToolSearch ? getToolSearchBetaHeader() : null
1198
+ if (toolSearchHeader && getAPIProvider() !== 'bedrock') {
1199
+ if (!betas.includes(toolSearchHeader)) {
1200
+ betas.push(toolSearchHeader)
1201
+ }
1202
+ }
1203
+
1204
+ // Determine if cached microcompact is enabled for this model.
1205
+ // Computed once here (in async context) and captured by paramsFromContext.
1206
+ // The beta header is also captured here to avoid a top-level import of the
1207
+ // ant-only CACHE_EDITING_BETA_HEADER constant.
1208
+ let cachedMCEnabled = false
1209
+ let cacheEditingBetaHeader = ''
1210
+ if (feature('CACHED_MICROCOMPACT')) {
1211
+ const {
1212
+ isCachedMicrocompactEnabled,
1213
+ isModelSupportedForCacheEditing,
1214
+ getCachedMCConfig,
1215
+ } = await import('../compact/cachedMicrocompact.js')
1216
+ const betas = await import('src/constants/betas.js')
1217
+ cacheEditingBetaHeader = betas.CACHE_EDITING_BETA_HEADER
1218
+ const featureEnabled = isCachedMicrocompactEnabled()
1219
+ const modelSupported = isModelSupportedForCacheEditing(options.model)
1220
+ cachedMCEnabled = featureEnabled && modelSupported
1221
+ const config = getCachedMCConfig()
1222
+ logForDebugging(
1223
+ `Cached MC gate: enabled=${featureEnabled} modelSupported=${modelSupported} model=${options.model} supportedModels=${jsonStringify(config.supportedModels)}`,
1224
+ )
1225
+ }
1226
+
1227
+ const useGlobalCacheFeature = shouldUseGlobalCacheScope()
1228
+ const willDefer = (t: Tool) =>
1229
+ useToolSearch && (deferredToolNames.has(t.name) || shouldDeferLspTool(t))
1230
+ // MCP tools are per-user → dynamic tool section → can't globally cache.
1231
+ // Only gate when an MCP tool will actually render (not defer_loading).
1232
+ const needsToolBasedCacheMarker =
1233
+ useGlobalCacheFeature &&
1234
+ filteredTools.some(t => t.isMcp === true && !willDefer(t))
1235
+
1236
+ // Ensure prompt_caching_scope beta header is present when global cache is enabled.
1237
+ if (
1238
+ useGlobalCacheFeature &&
1239
+ !betas.includes(PROMPT_CACHING_SCOPE_BETA_HEADER)
1240
+ ) {
1241
+ betas.push(PROMPT_CACHING_SCOPE_BETA_HEADER)
1242
+ }
1243
+
1244
+ // Determine global cache strategy for logging
1245
+ const globalCacheStrategy: GlobalCacheStrategy = useGlobalCacheFeature
1246
+ ? needsToolBasedCacheMarker
1247
+ ? 'none'
1248
+ : 'system_prompt'
1249
+ : 'none'
1250
+
1251
+ // Build tool schemas, adding defer_loading for MCP tools when tool search is enabled
1252
+ // Note: We pass the full `tools` list (not filteredTools) to toolToAPISchema so that
1253
+ // ToolSearchTool's prompt can list ALL available MCP tools. The filtering only affects
1254
+ // which tools are actually sent to the API, not what the model sees in tool descriptions.
1255
+ const toolSchemas = await Promise.all(
1256
+ filteredTools.map(tool =>
1257
+ toolToAPISchema(tool, {
1258
+ getToolPermissionContext: options.getToolPermissionContext,
1259
+ tools,
1260
+ agents: options.agents,
1261
+ allowedAgentTypes: options.allowedAgentTypes,
1262
+ model: options.model,
1263
+ deferLoading: willDefer(tool),
1264
+ }),
1265
+ ),
1266
+ )
1267
+
1268
+ if (useToolSearch) {
1269
+ const includedDeferredTools = count(filteredTools, t =>
1270
+ deferredToolNames.has(t.name),
1271
+ )
1272
+ logForDebugging(
1273
+ `Dynamic tool loading: ${includedDeferredTools}/${deferredToolNames.size} deferred tools included`,
1274
+ )
1275
+ }
1276
+
1277
+ queryCheckpoint('query_tool_schema_build_end')
1278
+
1279
+ // Normalize messages before building system prompt (needed for fingerprinting)
1280
+ // Instrumentation: Track message count before normalization
1281
+ logEvent('tengu_api_before_normalize', {
1282
+ preNormalizedMessageCount: messages.length,
1283
+ })
1284
+
1285
+ queryCheckpoint('query_message_normalization_start')
1286
+ let messagesForAPI = normalizeMessagesForAPI(messages, filteredTools)
1287
+ queryCheckpoint('query_message_normalization_end')
1288
+
1289
+ // Model-specific post-processing: strip tool-search-specific fields if the
1290
+ // selected model doesn't support tool search.
1291
+ //
1292
+ // Why is this needed in addition to normalizeMessagesForAPI?
1293
+ // - normalizeMessagesForAPI uses isToolSearchEnabledNoModelCheck() because it's
1294
+ // called from ~20 places (analytics, feedback, sharing, etc.), many of which
1295
+ // don't have model context. Adding model to its signature would be a large refactor.
1296
+ // - This post-processing uses the model-aware isToolSearchEnabled() check
1297
+ // - This handles mid-conversation model switching (e.g., Sonnet → Haiku) where
1298
+ // stale tool-search fields from the previous model would cause 400 errors
1299
+ //
1300
+ // Note: For assistant messages, normalizeMessagesForAPI already normalized the
1301
+ // tool inputs, so stripCallerFieldFromAssistantMessage only needs to remove the
1302
+ // 'caller' field (not re-normalize inputs).
1303
+ if (!useToolSearch) {
1304
+ messagesForAPI = messagesForAPI.map(msg => {
1305
+ switch (msg.type) {
1306
+ case 'user':
1307
+ // Strip tool_reference blocks from tool_result content
1308
+ return stripToolReferenceBlocksFromUserMessage(msg)
1309
+ case 'assistant':
1310
+ // Strip 'caller' field from tool_use blocks
1311
+ return stripCallerFieldFromAssistantMessage(msg)
1312
+ default:
1313
+ return msg
1314
+ }
1315
+ })
1316
+ }
1317
+
1318
+ // Repair tool_use/tool_result pairing mismatches that can occur when resuming
1319
+ // remote/teleport sessions. Inserts synthetic error tool_results for orphaned
1320
+ // tool_uses and strips orphaned tool_results referencing non-existent tool_uses.
1321
+ messagesForAPI = ensureToolResultPairing(messagesForAPI)
1322
+
1323
+ // Strip advisor blocks — the API rejects them without the beta header.
1324
+ if (!betas.includes(ADVISOR_BETA_HEADER)) {
1325
+ messagesForAPI = stripAdvisorBlocks(messagesForAPI)
1326
+ }
1327
+
1328
+ // Strip excess media items before making the API call.
1329
+ // The API rejects requests with >100 media items but returns a confusing error.
1330
+ // Rather than erroring (which is hard to recover from in Cowork/CCD), we
1331
+ // silently drop the oldest media items to stay within the limit.
1332
+ messagesForAPI = stripExcessMediaItems(
1333
+ messagesForAPI,
1334
+ API_MAX_MEDIA_PER_REQUEST,
1335
+ )
1336
+
1337
+ // Instrumentation: Track message count after normalization
1338
+ logEvent('tengu_api_after_normalize', {
1339
+ postNormalizedMessageCount: messagesForAPI.length,
1340
+ })
1341
+
1342
+ // Compute fingerprint from first user message for attribution.
1343
+ // Must run BEFORE injecting synthetic messages (e.g. deferred tool names)
1344
+ // so the fingerprint reflects the actual user input.
1345
+ const fingerprint = computeFingerprintFromMessages(messagesForAPI)
1346
+
1347
+ // When the delta attachment is enabled, deferred tools are announced
1348
+ // via persisted deferred_tools_delta attachments instead of this
1349
+ // ephemeral prepend (which busts cache whenever the pool changes).
1350
+ if (useToolSearch && !isDeferredToolsDeltaEnabled()) {
1351
+ const deferredToolList = tools
1352
+ .filter(t => deferredToolNames.has(t.name))
1353
+ .map(formatDeferredToolLine)
1354
+ .sort()
1355
+ .join('\n')
1356
+ if (deferredToolList) {
1357
+ messagesForAPI = [
1358
+ createUserMessage({
1359
+ content: `<available-deferred-tools>\n${deferredToolList}\n</available-deferred-tools>`,
1360
+ isMeta: true,
1361
+ }),
1362
+ ...messagesForAPI,
1363
+ ]
1364
+ }
1365
+ }
1366
+
1367
+ // Chrome tool-search instructions: when the delta attachment is enabled,
1368
+ // these are carried as a client-side block in mcp_instructions_delta
1369
+ // (attachments.ts) instead of here. This per-request sys-prompt append
1370
+ // busts the prompt cache when chrome connects late.
1371
+ const hasChromeTools = filteredTools.some(t =>
1372
+ isToolFromMcpServer(t.name, CLAUDE_IN_CHROME_MCP_SERVER_NAME),
1373
+ )
1374
+ const injectChromeHere =
1375
+ useToolSearch && hasChromeTools && !isMcpInstructionsDeltaEnabled()
1376
+
1377
+ // filter(Boolean) works by converting each element to a boolean - empty strings become false and are filtered out.
1378
+ systemPrompt = asSystemPrompt(
1379
+ [
1380
+ getAttributionHeader(fingerprint),
1381
+ getCLISyspromptPrefix({
1382
+ isNonInteractive: options.isNonInteractiveSession,
1383
+ hasAppendSystemPrompt: options.hasAppendSystemPrompt,
1384
+ }),
1385
+ ...systemPrompt,
1386
+ ...(advisorModel ? [ADVISOR_TOOL_INSTRUCTIONS] : []),
1387
+ ...(injectChromeHere ? [CHROME_TOOL_SEARCH_INSTRUCTIONS] : []),
1388
+ ].filter(Boolean),
1389
+ )
1390
+
1391
+ // Prepend system prompt block for easy API identification
1392
+ logAPIPrefix(systemPrompt)
1393
+
1394
+ const enablePromptCaching =
1395
+ options.enablePromptCaching ?? getPromptCachingEnabled(options.model)
1396
+ const system = buildSystemPromptBlocks(systemPrompt, enablePromptCaching, {
1397
+ skipGlobalCacheForSystemPrompt: needsToolBasedCacheMarker,
1398
+ querySource: options.querySource,
1399
+ })
1400
+ const useBetas = betas.length > 0
1401
+
1402
+ // Build minimal context for detailed tracing (when beta tracing is enabled)
1403
+ // Note: The actual new_context message extraction is done in sessionTracing.ts using
1404
+ // hash-based tracking per querySource (agent) from the messagesForAPI array
1405
+ const extraToolSchemas = [...(options.extraToolSchemas ?? [])]
1406
+ if (advisorModel) {
1407
+ // Server tools must be in the tools array by API contract. Appended after
1408
+ // toolSchemas (which carries the cache_control marker) so toggling /advisor
1409
+ // only churns the small suffix, not the cached prefix.
1410
+ extraToolSchemas.push({
1411
+ type: 'advisor_20260301',
1412
+ name: 'advisor',
1413
+ model: advisorModel,
1414
+ } as unknown as BetaToolUnion)
1415
+ }
1416
+ const allTools = [...toolSchemas, ...extraToolSchemas]
1417
+
1418
+ const isFastMode =
1419
+ isFastModeEnabled() &&
1420
+ isFastModeAvailable() &&
1421
+ !isFastModeCooldown() &&
1422
+ isFastModeSupportedByModel(options.model) &&
1423
+ !!options.fastMode
1424
+
1425
+ // Sticky-on latches for dynamic beta headers. Each header, once first
1426
+ // sent, keeps being sent for the rest of the session so mid-session
1427
+ // toggles don't change the server-side cache key and bust ~50-70K tokens.
1428
+ // Latches are cleared on /clear and /compact via clearBetaHeaderLatches().
1429
+ // Per-call gates (isAgenticQuery, querySource===repl_main_thread) stay
1430
+ // per-call so non-agentic queries keep their own stable header set.
1431
+
1432
+ let afkHeaderLatched = getAfkModeHeaderLatched() === true
1433
+ if (feature('TRANSCRIPT_CLASSIFIER')) {
1434
+ if (
1435
+ !afkHeaderLatched &&
1436
+ isAgenticQuery &&
1437
+ shouldIncludeFirstPartyOnlyBetas() &&
1438
+ (autoModeStateModule?.isAutoModeActive() ?? false)
1439
+ ) {
1440
+ afkHeaderLatched = true
1441
+ setAfkModeHeaderLatched(true)
1442
+ }
1443
+ }
1444
+
1445
+ let fastModeHeaderLatched = getFastModeHeaderLatched() === true
1446
+ if (!fastModeHeaderLatched && isFastMode) {
1447
+ fastModeHeaderLatched = true
1448
+ setFastModeHeaderLatched(true)
1449
+ }
1450
+
1451
+ let cacheEditingHeaderLatched = getCacheEditingHeaderLatched() === true
1452
+ if (feature('CACHED_MICROCOMPACT')) {
1453
+ if (
1454
+ !cacheEditingHeaderLatched &&
1455
+ cachedMCEnabled &&
1456
+ getAPIProvider() === 'firstParty' &&
1457
+ options.querySource === 'repl_main_thread'
1458
+ ) {
1459
+ cacheEditingHeaderLatched = true
1460
+ setCacheEditingHeaderLatched(true)
1461
+ }
1462
+ }
1463
+
1464
+ // Only latch from agentic queries so a classifier call doesn't flip the
1465
+ // main thread's context_management mid-turn.
1466
+ let thinkingClearLatched = getThinkingClearLatched() === true
1467
+ if (!thinkingClearLatched && isAgenticQuery) {
1468
+ const lastCompletion = getLastApiCompletionTimestamp()
1469
+ if (
1470
+ lastCompletion !== null &&
1471
+ Date.now() - lastCompletion > CACHE_TTL_1HOUR_MS
1472
+ ) {
1473
+ thinkingClearLatched = true
1474
+ setThinkingClearLatched(true)
1475
+ }
1476
+ }
1477
+
1478
+ const effort = resolveAppliedEffort(options.model, options.effortValue)
1479
+
1480
+ if (feature('PROMPT_CACHE_BREAK_DETECTION')) {
1481
+ // Exclude defer_loading tools from the hash -- the API strips them from the
1482
+ // prompt, so they never affect the actual cache key. Including them creates
1483
+ // false-positive "tool schemas changed" breaks when tools are discovered or
1484
+ // MCP servers reconnect.
1485
+ const toolsForCacheDetection = allTools.filter(
1486
+ t => !('defer_loading' in t && t.defer_loading),
1487
+ )
1488
+ // Capture everything that could affect the server-side cache key.
1489
+ // Pass latched header values (not live state) so break detection
1490
+ // reflects what we actually send, not what the user toggled.
1491
+ recordPromptState({
1492
+ system,
1493
+ toolSchemas: toolsForCacheDetection,
1494
+ querySource: options.querySource,
1495
+ model: options.model,
1496
+ agentId: options.agentId,
1497
+ fastMode: fastModeHeaderLatched,
1498
+ globalCacheStrategy,
1499
+ betas,
1500
+ autoModeActive: afkHeaderLatched,
1501
+ isUsingOverage: currentLimits.isUsingOverage ?? false,
1502
+ cachedMCEnabled: cacheEditingHeaderLatched,
1503
+ effortValue: effort,
1504
+ extraBodyParams: getExtraBodyParams(),
1505
+ })
1506
+ }
1507
+
1508
+ const newContext: LLMRequestNewContext | undefined = isBetaTracingEnabled()
1509
+ ? {
1510
+ systemPrompt: systemPrompt.join('\n\n'),
1511
+ querySource: options.querySource,
1512
+ tools: jsonStringify(allTools),
1513
+ }
1514
+ : undefined
1515
+
1516
+ // Capture the span so we can pass it to endLLMRequestSpan later
1517
+ // This ensures responses are matched to the correct request when multiple requests run in parallel
1518
+ const llmSpan = startLLMRequestSpan(
1519
+ options.model,
1520
+ newContext,
1521
+ messagesForAPI,
1522
+ isFastMode,
1523
+ )
1524
+
1525
+ const startIncludingRetries = Date.now()
1526
+ let start = Date.now()
1527
+ let attemptNumber = 0
1528
+ const attemptStartTimes: number[] = []
1529
+ let stream: Stream<BetaRawMessageStreamEvent> | undefined = undefined
1530
+ let streamRequestId: string | null | undefined = undefined
1531
+ let clientRequestId: string | undefined = undefined
1532
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins -- Response is available in Node 18+ and is used by the SDK
1533
+ let streamResponse: Response | undefined = undefined
1534
+
1535
+ // Release all stream resources to prevent native memory leaks.
1536
+ // The Response object holds native TLS/socket buffers that live outside the
1537
+ // V8 heap (observed on the Node.js/npm path; see GH #32920), so we must
1538
+ // explicitly cancel and release it regardless of how the generator exits.
1539
+ function releaseStreamResources(): void {
1540
+ cleanupStream(stream)
1541
+ stream = undefined
1542
+ if (streamResponse) {
1543
+ streamResponse.body?.cancel().catch(() => {})
1544
+ streamResponse = undefined
1545
+ }
1546
+ }
1547
+
1548
+ // Consume pending cache edits ONCE before paramsFromContext is defined.
1549
+ // paramsFromContext is called multiple times (logging, retries), so consuming
1550
+ // inside it would cause the first call to steal edits from subsequent calls.
1551
+ const consumedCacheEdits = cachedMCEnabled ? consumePendingCacheEdits() : null
1552
+ const consumedPinnedEdits = cachedMCEnabled ? getPinnedCacheEdits() : []
1553
+
1554
+ // Capture the betas sent in the last API request, including the ones that
1555
+ // were dynamically added, so we can log and send it to telemetry.
1556
+ let lastRequestBetas: string[] | undefined
1557
+
1558
+ const paramsFromContext = (retryContext: RetryContext) => {
1559
+ const betasParams = [...betas]
1560
+
1561
+ // Append 1M beta dynamically for the Sonnet 1M experiment.
1562
+ if (
1563
+ !betasParams.includes(CONTEXT_1M_BETA_HEADER) &&
1564
+ getSonnet1mExpTreatmentEnabled(retryContext.model)
1565
+ ) {
1566
+ betasParams.push(CONTEXT_1M_BETA_HEADER)
1567
+ }
1568
+
1569
+ // For Bedrock, include both model-based betas and dynamically-added tool search header
1570
+ const bedrockBetas =
1571
+ getAPIProvider() === 'bedrock'
1572
+ ? [
1573
+ ...getBedrockExtraBodyParamsBetas(retryContext.model),
1574
+ ...(toolSearchHeader ? [toolSearchHeader] : []),
1575
+ ]
1576
+ : []
1577
+ const extraBodyParams = getExtraBodyParams(bedrockBetas)
1578
+
1579
+ const outputConfig: BetaOutputConfig = {
1580
+ ...((extraBodyParams.output_config as BetaOutputConfig) ?? {}),
1581
+ }
1582
+
1583
+ configureEffortParams(
1584
+ effort,
1585
+ outputConfig,
1586
+ extraBodyParams,
1587
+ betasParams,
1588
+ options.model,
1589
+ )
1590
+
1591
+ configureTaskBudgetParams(
1592
+ options.taskBudget,
1593
+ outputConfig as BetaOutputConfig & { task_budget?: TaskBudgetParam },
1594
+ betasParams,
1595
+ )
1596
+
1597
+ // Merge outputFormat into extraBodyParams.output_config alongside effort
1598
+ // Requires structured-outputs beta header per SDK (see parse() in messages.mjs)
1599
+ if (options.outputFormat && !('format' in outputConfig)) {
1600
+ outputConfig.format = options.outputFormat as BetaJSONOutputFormat
1601
+ // Add beta header if not already present and provider supports it
1602
+ if (
1603
+ modelSupportsStructuredOutputs(options.model) &&
1604
+ !betasParams.includes(STRUCTURED_OUTPUTS_BETA_HEADER)
1605
+ ) {
1606
+ betasParams.push(STRUCTURED_OUTPUTS_BETA_HEADER)
1607
+ }
1608
+ }
1609
+
1610
+ // Retry context gets preference because it tries to course correct if we exceed the context window limit
1611
+ const maxOutputTokens =
1612
+ retryContext?.maxTokensOverride ||
1613
+ options.maxOutputTokensOverride ||
1614
+ getMaxOutputTokensForModel(options.model)
1615
+
1616
+ const hasThinking =
1617
+ thinkingConfig.type !== 'disabled' &&
1618
+ !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_THINKING)
1619
+ let thinking: BetaMessageStreamParams['thinking'] | undefined = undefined
1620
+
1621
+ // IMPORTANT: Do not change the adaptive-vs-budget thinking selection below
1622
+ // without notifying the model launch DRI and research. This is a sensitive
1623
+ // setting that can greatly affect model quality and bashing.
1624
+ if (hasThinking && modelSupportsThinking(options.model)) {
1625
+ if (
1626
+ !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_ADAPTIVE_THINKING) &&
1627
+ modelSupportsAdaptiveThinking(options.model)
1628
+ ) {
1629
+ // For models that support adaptive thinking, always use adaptive
1630
+ // thinking without a budget.
1631
+ thinking = {
1632
+ type: 'adaptive',
1633
+ } satisfies BetaMessageStreamParams['thinking']
1634
+ } else {
1635
+ // For models that do not support adaptive thinking, use the default
1636
+ // thinking budget unless explicitly specified.
1637
+ let thinkingBudget = getMaxThinkingTokensForModel(options.model)
1638
+ if (
1639
+ thinkingConfig.type === 'enabled' &&
1640
+ thinkingConfig.budgetTokens !== undefined
1641
+ ) {
1642
+ thinkingBudget = thinkingConfig.budgetTokens
1643
+ }
1644
+ thinkingBudget = Math.min(maxOutputTokens - 1, thinkingBudget)
1645
+ thinking = {
1646
+ budget_tokens: thinkingBudget,
1647
+ type: 'enabled',
1648
+ } satisfies BetaMessageStreamParams['thinking']
1649
+ }
1650
+ }
1651
+
1652
+ // Get API context management strategies if enabled
1653
+ const contextManagement = getAPIContextManagement({
1654
+ hasThinking,
1655
+ isRedactThinkingActive: betasParams.includes(REDACT_THINKING_BETA_HEADER),
1656
+ clearAllThinking: thinkingClearLatched,
1657
+ })
1658
+
1659
+ const enablePromptCaching =
1660
+ options.enablePromptCaching ?? getPromptCachingEnabled(retryContext.model)
1661
+
1662
+ // Fast mode: header is latched session-stable (cache-safe), but
1663
+ // `speed='fast'` stays dynamic so cooldown still suppresses the actual
1664
+ // fast-mode request without changing the cache key.
1665
+ let speed: BetaMessageStreamParams['speed']
1666
+ const isFastModeForRetry =
1667
+ isFastModeEnabled() &&
1668
+ isFastModeAvailable() &&
1669
+ !isFastModeCooldown() &&
1670
+ isFastModeSupportedByModel(options.model) &&
1671
+ !!retryContext.fastMode
1672
+ if (isFastModeForRetry) {
1673
+ speed = 'fast'
1674
+ }
1675
+ if (fastModeHeaderLatched && !betasParams.includes(FAST_MODE_BETA_HEADER)) {
1676
+ betasParams.push(FAST_MODE_BETA_HEADER)
1677
+ }
1678
+
1679
+ // AFK mode beta: latched once auto mode is first activated. Still gated
1680
+ // by isAgenticQuery per-call so classifiers/compaction don't get it.
1681
+ if (feature('TRANSCRIPT_CLASSIFIER')) {
1682
+ if (
1683
+ afkHeaderLatched &&
1684
+ shouldIncludeFirstPartyOnlyBetas() &&
1685
+ isAgenticQuery &&
1686
+ !betasParams.includes(AFK_MODE_BETA_HEADER)
1687
+ ) {
1688
+ betasParams.push(AFK_MODE_BETA_HEADER)
1689
+ }
1690
+ }
1691
+
1692
+ // Cache editing beta: header is latched session-stable; useCachedMC
1693
+ // (controls cache_edits body behavior) stays live so edits stop when
1694
+ // the feature disables but the header doesn't flip.
1695
+ const useCachedMC =
1696
+ cachedMCEnabled &&
1697
+ getAPIProvider() === 'firstParty' &&
1698
+ options.querySource === 'repl_main_thread'
1699
+ if (
1700
+ cacheEditingHeaderLatched &&
1701
+ getAPIProvider() === 'firstParty' &&
1702
+ options.querySource === 'repl_main_thread' &&
1703
+ !betasParams.includes(cacheEditingBetaHeader)
1704
+ ) {
1705
+ betasParams.push(cacheEditingBetaHeader)
1706
+ logForDebugging(
1707
+ 'Cache editing beta header enabled for cached microcompact',
1708
+ )
1709
+ }
1710
+
1711
+ // Only send temperature when thinking is disabled — the API requires
1712
+ // temperature: 1 when thinking is enabled, which is already the default.
1713
+ const temperature = !hasThinking
1714
+ ? (options.temperatureOverride ?? 1)
1715
+ : undefined
1716
+
1717
+ lastRequestBetas = betasParams
1718
+
1719
+ return {
1720
+ model: normalizeModelStringForAPI(options.model),
1721
+ messages: addCacheBreakpoints(
1722
+ messagesForAPI,
1723
+ enablePromptCaching,
1724
+ options.querySource,
1725
+ useCachedMC,
1726
+ consumedCacheEdits,
1727
+ consumedPinnedEdits,
1728
+ options.skipCacheWrite,
1729
+ ),
1730
+ system,
1731
+ tools: allTools,
1732
+ tool_choice: options.toolChoice,
1733
+ ...(useBetas && { betas: betasParams }),
1734
+ metadata: getAPIMetadata(),
1735
+ max_tokens: maxOutputTokens,
1736
+ thinking,
1737
+ ...(temperature !== undefined && { temperature }),
1738
+ ...(contextManagement &&
1739
+ useBetas &&
1740
+ betasParams.includes(CONTEXT_MANAGEMENT_BETA_HEADER) && {
1741
+ context_management: contextManagement,
1742
+ }),
1743
+ ...extraBodyParams,
1744
+ ...(Object.keys(outputConfig).length > 0 && {
1745
+ output_config: outputConfig,
1746
+ }),
1747
+ ...(speed !== undefined && { speed }),
1748
+ }
1749
+ }
1750
+
1751
+ // Compute log scalars synchronously so the fire-and-forget .then() closure
1752
+ // captures only primitives instead of paramsFromContext's full closure scope
1753
+ // (messagesForAPI, system, allTools, betas — the entire request-building
1754
+ // context), which would otherwise be pinned until the promise resolves.
1755
+ {
1756
+ const queryParams = paramsFromContext({
1757
+ model: options.model,
1758
+ thinkingConfig,
1759
+ })
1760
+ const logMessagesLength = queryParams.messages.length
1761
+ const logBetas = useBetas ? (queryParams.betas ?? []) : []
1762
+ const logThinkingType = queryParams.thinking?.type ?? 'disabled'
1763
+ const logEffortValue = queryParams.output_config?.effort
1764
+ void options.getToolPermissionContext().then(permissionContext => {
1765
+ logAPIQuery({
1766
+ model: options.model,
1767
+ messagesLength: logMessagesLength,
1768
+ temperature: options.temperatureOverride ?? 1,
1769
+ betas: logBetas,
1770
+ permissionMode: permissionContext.mode,
1771
+ querySource: options.querySource,
1772
+ queryTracking: options.queryTracking,
1773
+ thinkingType: logThinkingType,
1774
+ effortValue: logEffortValue,
1775
+ fastMode: isFastMode,
1776
+ previousRequestId,
1777
+ })
1778
+ })
1779
+ }
1780
+
1781
+ const newMessages: AssistantMessage[] = []
1782
+ let ttftMs = 0
1783
+ let partialMessage: BetaMessage | undefined = undefined
1784
+ const contentBlocks: (BetaContentBlock | ConnectorTextBlock)[] = []
1785
+ let usage: NonNullableUsage = EMPTY_USAGE
1786
+ let costUSD = 0
1787
+ let stopReason: BetaStopReason | null = null
1788
+ let didFallBackToNonStreaming = false
1789
+ let fallbackMessage: AssistantMessage | undefined
1790
+ let maxOutputTokens = 0
1791
+ let responseHeaders: globalThis.Headers | undefined = undefined
1792
+ let research: unknown = undefined
1793
+ let isFastModeRequest = isFastMode // Keep separate state as it may change if falling back
1794
+ let isAdvisorInProgress = false
1795
+
1796
+ try {
1797
+ queryCheckpoint('query_client_creation_start')
1798
+ const generator = withRetry(
1799
+ async () => {
1800
+ // For third-party models, create a client configured for that provider
1801
+ if (isThirdPartyModel(options.model)) {
1802
+ const result = await getAnthropicClientForModel({
1803
+ model: options.model,
1804
+ maxRetries: 0,
1805
+ fetchOverride: options.fetchOverride,
1806
+ source: options.querySource,
1807
+ })
1808
+ return result.client
1809
+ }
1810
+ return getAnthropicClient({
1811
+ maxRetries: 0, // Disabled auto-retry in favor of manual implementation
1812
+ model: options.model,
1813
+ fetchOverride: options.fetchOverride,
1814
+ source: options.querySource,
1815
+ })
1816
+ },
1817
+ async (anthropic, attempt, context) => {
1818
+ attemptNumber = attempt
1819
+ isFastModeRequest = context.fastMode ?? false
1820
+ start = Date.now()
1821
+ attemptStartTimes.push(start)
1822
+ // Client has been created by withRetry's getClient() call. This fires
1823
+ // once per attempt; on retries the client is usually cached (withRetry
1824
+ // only calls getClient() again after auth errors), so the delta from
1825
+ // client_creation_start is meaningful on attempt 1.
1826
+ queryCheckpoint('query_client_creation_end')
1827
+
1828
+ const params = paramsFromContext(context)
1829
+ captureAPIRequest(params, options.querySource) // Capture for bug reports
1830
+
1831
+ maxOutputTokens = params.max_tokens
1832
+
1833
+ // Fire immediately before the fetch is dispatched. .withResponse() below
1834
+ // awaits until response headers arrive, so this MUST be before the await
1835
+ // or the "Network TTFB" phase measurement is wrong.
1836
+ queryCheckpoint('query_api_request_sent')
1837
+ if (!options.agentId) {
1838
+ headlessProfilerCheckpoint('api_request_sent')
1839
+ }
1840
+
1841
+ // Generate and track client request ID so timeouts (which return no
1842
+ // server request ID) can still be correlated with server logs.
1843
+ // First-party only — 3P providers don't log it (inc-4029 class).
1844
+ clientRequestId =
1845
+ getAPIProvider() === 'firstParty' && isFirstPartyAnthropicBaseUrl()
1846
+ ? randomUUID()
1847
+ : undefined
1848
+
1849
+ // Use raw stream instead of BetaMessageStream to avoid O(n²) partial JSON parsing
1850
+ // BetaMessageStream calls partialParse() on every input_json_delta, which we don't need
1851
+ // since we handle tool input accumulation ourselves
1852
+ // biome-ignore lint/plugin: main conversation loop handles attribution separately
1853
+ const result = await anthropic.beta.messages
1854
+ .create(
1855
+ { ...params, stream: true },
1856
+ {
1857
+ signal,
1858
+ ...(clientRequestId && {
1859
+ headers: { [CLIENT_REQUEST_ID_HEADER]: clientRequestId },
1860
+ }),
1861
+ },
1862
+ )
1863
+ .withResponse()
1864
+ queryCheckpoint('query_response_headers_received')
1865
+ streamRequestId = result.request_id
1866
+ streamResponse = result.response
1867
+ return result.data
1868
+ },
1869
+ {
1870
+ model: options.model,
1871
+ fallbackModel: options.fallbackModel,
1872
+ thinkingConfig,
1873
+ ...(isFastModeEnabled() ? { fastMode: isFastMode } : false),
1874
+ signal,
1875
+ querySource: options.querySource,
1876
+ },
1877
+ )
1878
+
1879
+ let e
1880
+ do {
1881
+ e = await generator.next()
1882
+
1883
+ // yield API error messages (the stream has a 'controller' property, error messages don't)
1884
+ if (!('controller' in e.value)) {
1885
+ yield e.value
1886
+ }
1887
+ } while (!e.done)
1888
+ stream = e.value as Stream<BetaRawMessageStreamEvent>
1889
+
1890
+ // reset state
1891
+ newMessages.length = 0
1892
+ ttftMs = 0
1893
+ partialMessage = undefined
1894
+ contentBlocks.length = 0
1895
+ usage = EMPTY_USAGE
1896
+ stopReason = null
1897
+ isAdvisorInProgress = false
1898
+
1899
+ // Streaming idle timeout watchdog: abort the stream if no chunks arrive
1900
+ // for STREAM_IDLE_TIMEOUT_MS. Unlike the stall detection below (which only
1901
+ // fires when the *next* chunk arrives), this uses setTimeout to actively
1902
+ // kill hung streams. Without this, a silently dropped connection can hang
1903
+ // the session indefinitely since the SDK's request timeout only covers the
1904
+ // initial fetch(), not the streaming body.
1905
+ const streamWatchdogEnabled = isEnvTruthy(
1906
+ process.env.CLAUDE_ENABLE_STREAM_WATCHDOG,
1907
+ )
1908
+ const STREAM_IDLE_TIMEOUT_MS =
1909
+ parseInt(process.env.CLAUDE_STREAM_IDLE_TIMEOUT_MS || '', 10) || 90_000
1910
+ const STREAM_IDLE_WARNING_MS = STREAM_IDLE_TIMEOUT_MS / 2
1911
+ let streamIdleAborted = false
1912
+ // performance.now() snapshot when watchdog fires, for measuring abort propagation delay
1913
+ let streamWatchdogFiredAt: number | null = null
1914
+ let streamIdleWarningTimer: ReturnType<typeof setTimeout> | null = null
1915
+ let streamIdleTimer: ReturnType<typeof setTimeout> | null = null
1916
+ function clearStreamIdleTimers(): void {
1917
+ if (streamIdleWarningTimer !== null) {
1918
+ clearTimeout(streamIdleWarningTimer)
1919
+ streamIdleWarningTimer = null
1920
+ }
1921
+ if (streamIdleTimer !== null) {
1922
+ clearTimeout(streamIdleTimer)
1923
+ streamIdleTimer = null
1924
+ }
1925
+ }
1926
+ function resetStreamIdleTimer(): void {
1927
+ clearStreamIdleTimers()
1928
+ if (!streamWatchdogEnabled) {
1929
+ return
1930
+ }
1931
+ streamIdleWarningTimer = setTimeout(
1932
+ warnMs => {
1933
+ logForDebugging(
1934
+ `Streaming idle warning: no chunks received for ${warnMs / 1000}s`,
1935
+ { level: 'warn' },
1936
+ )
1937
+ logForDiagnosticsNoPII('warn', 'cli_streaming_idle_warning')
1938
+ },
1939
+ STREAM_IDLE_WARNING_MS,
1940
+ STREAM_IDLE_WARNING_MS,
1941
+ )
1942
+ streamIdleTimer = setTimeout(() => {
1943
+ streamIdleAborted = true
1944
+ streamWatchdogFiredAt = performance.now()
1945
+ logForDebugging(
1946
+ `Streaming idle timeout: no chunks received for ${STREAM_IDLE_TIMEOUT_MS / 1000}s, aborting stream`,
1947
+ { level: 'error' },
1948
+ )
1949
+ logForDiagnosticsNoPII('error', 'cli_streaming_idle_timeout')
1950
+ logEvent('tengu_streaming_idle_timeout', {
1951
+ model:
1952
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1953
+ request_id: (streamRequestId ??
1954
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1955
+ timeout_ms: STREAM_IDLE_TIMEOUT_MS,
1956
+ })
1957
+ releaseStreamResources()
1958
+ }, STREAM_IDLE_TIMEOUT_MS)
1959
+ }
1960
+ resetStreamIdleTimer()
1961
+
1962
+ startSessionActivity('api_call')
1963
+ try {
1964
+ // stream in and accumulate state
1965
+ let isFirstChunk = true
1966
+ let lastEventTime: number | null = null // Set after first chunk to avoid measuring TTFB as a stall
1967
+ const STALL_THRESHOLD_MS = 30_000 // 30 seconds
1968
+ let totalStallTime = 0
1969
+ let stallCount = 0
1970
+
1971
+ for await (const part of stream) {
1972
+ resetStreamIdleTimer()
1973
+ const now = Date.now()
1974
+
1975
+ // Detect and log streaming stalls (only after first event to avoid counting TTFB)
1976
+ if (lastEventTime !== null) {
1977
+ const timeSinceLastEvent = now - lastEventTime
1978
+ if (timeSinceLastEvent > STALL_THRESHOLD_MS) {
1979
+ stallCount++
1980
+ totalStallTime += timeSinceLastEvent
1981
+ logForDebugging(
1982
+ `Streaming stall detected: ${(timeSinceLastEvent / 1000).toFixed(1)}s gap between events (stall #${stallCount})`,
1983
+ { level: 'warn' },
1984
+ )
1985
+ logEvent('tengu_streaming_stall', {
1986
+ stall_duration_ms: timeSinceLastEvent,
1987
+ stall_count: stallCount,
1988
+ total_stall_time_ms: totalStallTime,
1989
+ event_type:
1990
+ part.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1991
+ model:
1992
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1993
+ request_id: (streamRequestId ??
1994
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1995
+ })
1996
+ }
1997
+ }
1998
+ lastEventTime = now
1999
+
2000
+ if (isFirstChunk) {
2001
+ logForDebugging('Stream started - received first chunk')
2002
+ queryCheckpoint('query_first_chunk_received')
2003
+ if (!options.agentId) {
2004
+ headlessProfilerCheckpoint('first_chunk')
2005
+ }
2006
+ endQueryProfile()
2007
+ isFirstChunk = false
2008
+ }
2009
+
2010
+ switch (part.type) {
2011
+ case 'message_start': {
2012
+ partialMessage = part.message
2013
+ ttftMs = Date.now() - start
2014
+ usage = updateUsage(usage, part.message?.usage)
2015
+ // Capture research from message_start if available (internal only).
2016
+ // Always overwrite with the latest value.
2017
+ if (
2018
+ process.env.USER_TYPE === 'ant' &&
2019
+ 'research' in (part.message as unknown as Record<string, unknown>)
2020
+ ) {
2021
+ research = (part.message as unknown as Record<string, unknown>)
2022
+ .research
2023
+ }
2024
+ break
2025
+ }
2026
+ case 'content_block_start':
2027
+ switch (part.content_block.type) {
2028
+ case 'tool_use':
2029
+ contentBlocks[part.index] = {
2030
+ ...part.content_block,
2031
+ input: '',
2032
+ }
2033
+ break
2034
+ case 'server_tool_use':
2035
+ contentBlocks[part.index] = {
2036
+ ...part.content_block,
2037
+ input: '' as unknown as { [key: string]: unknown },
2038
+ }
2039
+ if ((part.content_block.name as string) === 'advisor') {
2040
+ isAdvisorInProgress = true
2041
+ logForDebugging(`[AdvisorTool] Advisor tool called`)
2042
+ logEvent('tengu_advisor_tool_call', {
2043
+ model:
2044
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2045
+ advisor_model: (advisorModel ??
2046
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2047
+ })
2048
+ }
2049
+ break
2050
+ case 'text':
2051
+ contentBlocks[part.index] = {
2052
+ ...part.content_block,
2053
+ // awkwardly, the sdk sometimes returns text as part of a
2054
+ // content_block_start message, then returns the same text
2055
+ // again in a content_block_delta message. we ignore it here
2056
+ // since there doesn't seem to be a way to detect when a
2057
+ // content_block_delta message duplicates the text.
2058
+ text: '',
2059
+ }
2060
+ break
2061
+ case 'thinking':
2062
+ contentBlocks[part.index] = {
2063
+ ...part.content_block,
2064
+ // also awkward
2065
+ thinking: '',
2066
+ // initialize signature to ensure field exists even if signature_delta never arrives
2067
+ signature: '',
2068
+ }
2069
+ break
2070
+ default:
2071
+ // even more awkwardly, the sdk mutates the contents of text blocks
2072
+ // as it works. we want the blocks to be immutable, so that we can
2073
+ // accumulate state ourselves.
2074
+ contentBlocks[part.index] = { ...part.content_block }
2075
+ if (
2076
+ (part.content_block.type as string) === 'advisor_tool_result'
2077
+ ) {
2078
+ isAdvisorInProgress = false
2079
+ logForDebugging(`[AdvisorTool] Advisor tool result received`)
2080
+ }
2081
+ break
2082
+ }
2083
+ break
2084
+ case 'content_block_delta': {
2085
+ const contentBlock = contentBlocks[part.index]
2086
+ const delta = part.delta as typeof part.delta | ConnectorTextDelta
2087
+ if (!contentBlock) {
2088
+ logEvent('tengu_streaming_error', {
2089
+ error_type:
2090
+ 'content_block_not_found_delta' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2091
+ part_type:
2092
+ part.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2093
+ part_index: part.index,
2094
+ })
2095
+ throw new RangeError('Content block not found')
2096
+ }
2097
+ if (
2098
+ feature('CONNECTOR_TEXT') &&
2099
+ delta.type === 'connector_text_delta'
2100
+ ) {
2101
+ if (contentBlock.type !== 'connector_text') {
2102
+ logEvent('tengu_streaming_error', {
2103
+ error_type:
2104
+ 'content_block_type_mismatch_connector_text' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2105
+ expected_type:
2106
+ 'connector_text' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2107
+ actual_type:
2108
+ contentBlock.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2109
+ })
2110
+ throw new Error('Content block is not a connector_text block')
2111
+ }
2112
+ contentBlock.connector_text += delta.connector_text
2113
+ } else {
2114
+ switch (delta.type) {
2115
+ case 'citations_delta':
2116
+ // TODO: handle citations
2117
+ break
2118
+ case 'input_json_delta':
2119
+ if (
2120
+ contentBlock.type !== 'tool_use' &&
2121
+ contentBlock.type !== 'server_tool_use'
2122
+ ) {
2123
+ logEvent('tengu_streaming_error', {
2124
+ error_type:
2125
+ 'content_block_type_mismatch_input_json' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2126
+ expected_type:
2127
+ 'tool_use' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2128
+ actual_type:
2129
+ contentBlock.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2130
+ })
2131
+ throw new Error('Content block is not a input_json block')
2132
+ }
2133
+ if (typeof contentBlock.input !== 'string') {
2134
+ logEvent('tengu_streaming_error', {
2135
+ error_type:
2136
+ 'content_block_input_not_string' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2137
+ input_type:
2138
+ typeof contentBlock.input as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2139
+ })
2140
+ throw new Error('Content block input is not a string')
2141
+ }
2142
+ contentBlock.input += delta.partial_json
2143
+ break
2144
+ case 'text_delta':
2145
+ if (contentBlock.type !== 'text') {
2146
+ logEvent('tengu_streaming_error', {
2147
+ error_type:
2148
+ 'content_block_type_mismatch_text' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2149
+ expected_type:
2150
+ 'text' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2151
+ actual_type:
2152
+ contentBlock.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2153
+ })
2154
+ throw new Error('Content block is not a text block')
2155
+ }
2156
+ contentBlock.text += delta.text
2157
+ break
2158
+ case 'signature_delta':
2159
+ if (
2160
+ feature('CONNECTOR_TEXT') &&
2161
+ contentBlock.type === 'connector_text'
2162
+ ) {
2163
+ contentBlock.signature = delta.signature
2164
+ break
2165
+ }
2166
+ if (contentBlock.type !== 'thinking') {
2167
+ logEvent('tengu_streaming_error', {
2168
+ error_type:
2169
+ 'content_block_type_mismatch_thinking_signature' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2170
+ expected_type:
2171
+ 'thinking' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2172
+ actual_type:
2173
+ contentBlock.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2174
+ })
2175
+ throw new Error('Content block is not a thinking block')
2176
+ }
2177
+ contentBlock.signature = delta.signature
2178
+ break
2179
+ case 'thinking_delta':
2180
+ if (contentBlock.type !== 'thinking') {
2181
+ logEvent('tengu_streaming_error', {
2182
+ error_type:
2183
+ 'content_block_type_mismatch_thinking_delta' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2184
+ expected_type:
2185
+ 'thinking' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2186
+ actual_type:
2187
+ contentBlock.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2188
+ })
2189
+ throw new Error('Content block is not a thinking block')
2190
+ }
2191
+ contentBlock.thinking += delta.thinking
2192
+ break
2193
+ }
2194
+ }
2195
+ // Capture research from content_block_delta if available (internal only).
2196
+ // Always overwrite with the latest value.
2197
+ if (process.env.USER_TYPE === 'ant' && 'research' in part) {
2198
+ research = (part as { research: unknown }).research
2199
+ }
2200
+ break
2201
+ }
2202
+ case 'content_block_stop': {
2203
+ const contentBlock = contentBlocks[part.index]
2204
+ if (!contentBlock) {
2205
+ logEvent('tengu_streaming_error', {
2206
+ error_type:
2207
+ 'content_block_not_found_stop' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2208
+ part_type:
2209
+ part.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2210
+ part_index: part.index,
2211
+ })
2212
+ throw new RangeError('Content block not found')
2213
+ }
2214
+ if (!partialMessage) {
2215
+ logEvent('tengu_streaming_error', {
2216
+ error_type:
2217
+ 'partial_message_not_found' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2218
+ part_type:
2219
+ part.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2220
+ })
2221
+ throw new Error('Message not found')
2222
+ }
2223
+ const m: AssistantMessage = {
2224
+ message: {
2225
+ ...partialMessage,
2226
+ content: normalizeContentFromAPI(
2227
+ [contentBlock] as BetaContentBlock[],
2228
+ tools,
2229
+ options.agentId,
2230
+ ),
2231
+ },
2232
+ requestId: streamRequestId ?? undefined,
2233
+ type: 'assistant',
2234
+ uuid: randomUUID(),
2235
+ timestamp: new Date().toISOString(),
2236
+ ...(process.env.USER_TYPE === 'ant' &&
2237
+ research !== undefined && { research }),
2238
+ ...(advisorModel && { advisorModel }),
2239
+ }
2240
+ newMessages.push(m)
2241
+ yield m
2242
+ break
2243
+ }
2244
+ case 'message_delta': {
2245
+ usage = updateUsage(usage, part.usage)
2246
+ // Capture research from message_delta if available (internal only).
2247
+ // Always overwrite with the latest value. Also write back to
2248
+ // already-yielded messages since message_delta arrives after
2249
+ // content_block_stop.
2250
+ if (
2251
+ process.env.USER_TYPE === 'ant' &&
2252
+ 'research' in (part as unknown as Record<string, unknown>)
2253
+ ) {
2254
+ research = (part as unknown as Record<string, unknown>).research
2255
+ for (const msg of newMessages) {
2256
+ msg.research = research
2257
+ }
2258
+ }
2259
+
2260
+ // Write final usage and stop_reason back to the last yielded
2261
+ // message. Messages are created at content_block_stop from
2262
+ // partialMessage, which was set at message_start before any tokens
2263
+ // were generated (output_tokens: 0, stop_reason: null).
2264
+ // message_delta arrives after content_block_stop with the real
2265
+ // values.
2266
+ //
2267
+ // IMPORTANT: Use direct property mutation, not object replacement.
2268
+ // The transcript write queue holds a reference to message.message
2269
+ // and serializes it lazily (100ms flush interval). Object
2270
+ // replacement ({ ...lastMsg.message, usage }) would disconnect
2271
+ // the queued reference; direct mutation ensures the transcript
2272
+ // captures the final values.
2273
+ stopReason = part.delta.stop_reason
2274
+
2275
+ const lastMsg = newMessages.at(-1)
2276
+ if (lastMsg) {
2277
+ lastMsg.message.usage = usage
2278
+ lastMsg.message.stop_reason = stopReason
2279
+ }
2280
+
2281
+ // Update cost
2282
+ const costUSDForPart = calculateUSDCost(resolvedModel, usage)
2283
+ costUSD += addToTotalSessionCost(
2284
+ costUSDForPart,
2285
+ usage,
2286
+ options.model,
2287
+ )
2288
+
2289
+ const refusalMessage = getErrorMessageIfRefusal(
2290
+ part.delta.stop_reason,
2291
+ options.model,
2292
+ )
2293
+ if (refusalMessage) {
2294
+ yield refusalMessage
2295
+ }
2296
+
2297
+ if (stopReason === 'max_tokens') {
2298
+ logEvent('tengu_max_tokens_reached', {
2299
+ max_tokens: maxOutputTokens,
2300
+ })
2301
+ yield createAssistantAPIErrorMessage({
2302
+ content: `${API_ERROR_MESSAGE_PREFIX}: Claude's response exceeded the ${
2303
+ maxOutputTokens
2304
+ } output token maximum. To configure this behavior, set the CLAUDE_CODE_MAX_OUTPUT_TOKENS environment variable.`,
2305
+ apiError: 'max_output_tokens',
2306
+ error: 'max_output_tokens',
2307
+ })
2308
+ }
2309
+
2310
+ if (stopReason === 'model_context_window_exceeded') {
2311
+ logEvent('tengu_context_window_exceeded', {
2312
+ max_tokens: maxOutputTokens,
2313
+ output_tokens: usage.output_tokens,
2314
+ })
2315
+ // Reuse the max_output_tokens recovery path — from the model's
2316
+ // perspective, both mean "response was cut off, continue from
2317
+ // where you left off."
2318
+ yield createAssistantAPIErrorMessage({
2319
+ content: `${API_ERROR_MESSAGE_PREFIX}: The model has reached its context window limit.`,
2320
+ apiError: 'max_output_tokens',
2321
+ error: 'max_output_tokens',
2322
+ })
2323
+ }
2324
+ break
2325
+ }
2326
+ case 'message_stop':
2327
+ break
2328
+ }
2329
+
2330
+ yield {
2331
+ type: 'stream_event',
2332
+ event: part,
2333
+ ...(part.type === 'message_start' ? { ttftMs } : undefined),
2334
+ }
2335
+ }
2336
+ // Clear the idle timeout watchdog now that the stream loop has exited
2337
+ clearStreamIdleTimers()
2338
+
2339
+ // If the stream was aborted by our idle timeout watchdog, fall back to
2340
+ // non-streaming retry rather than treating it as a completed stream.
2341
+ if (streamIdleAborted) {
2342
+ // Instrumentation: proves the for-await exited after the watchdog fired
2343
+ // (vs. hung forever). exit_delay_ms measures abort propagation latency:
2344
+ // 0-10ms = abort worked; >>1000ms = something else woke the loop.
2345
+ const exitDelayMs =
2346
+ streamWatchdogFiredAt !== null
2347
+ ? Math.round(performance.now() - streamWatchdogFiredAt)
2348
+ : -1
2349
+ logForDiagnosticsNoPII(
2350
+ 'info',
2351
+ 'cli_stream_loop_exited_after_watchdog_clean',
2352
+ )
2353
+ logEvent('tengu_stream_loop_exited_after_watchdog', {
2354
+ request_id: (streamRequestId ??
2355
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2356
+ exit_delay_ms: exitDelayMs,
2357
+ exit_path:
2358
+ 'clean' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2359
+ model:
2360
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2361
+ })
2362
+ // Prevent double-emit: this throw lands in the catch block below,
2363
+ // whose exit_path='error' probe guards on streamWatchdogFiredAt.
2364
+ streamWatchdogFiredAt = null
2365
+ throw new Error('Stream idle timeout - no chunks received')
2366
+ }
2367
+
2368
+ // Detect when the stream completed without producing any assistant messages.
2369
+ // This covers two proxy failure modes:
2370
+ // 1. No events at all (!partialMessage): proxy returned 200 with non-SSE body
2371
+ // 2. Partial events (partialMessage set but no content blocks completed AND
2372
+ // no stop_reason received): proxy returned message_start but stream ended
2373
+ // before content_block_stop and before message_delta with stop_reason
2374
+ // BetaMessageStream had the first check in _endRequest() but the raw Stream
2375
+ // does not - without it the generator silently returns no assistant messages,
2376
+ // causing "Execution error" in -p mode.
2377
+ // Note: We must check stopReason to avoid false positives. For example, with
2378
+ // structured output (--json-schema), the model calls a StructuredOutput tool
2379
+ // on turn 1, then on turn 2 responds with end_turn and no content blocks.
2380
+ // That's a legitimate empty response, not an incomplete stream.
2381
+ if (!partialMessage || (newMessages.length === 0 && !stopReason)) {
2382
+ logForDebugging(
2383
+ !partialMessage
2384
+ ? 'Stream completed without receiving message_start event - triggering non-streaming fallback'
2385
+ : 'Stream completed with message_start but no content blocks completed - triggering non-streaming fallback',
2386
+ { level: 'error' },
2387
+ )
2388
+ logEvent('tengu_stream_no_events', {
2389
+ model:
2390
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2391
+ request_id: (streamRequestId ??
2392
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2393
+ })
2394
+ throw new Error('Stream ended without receiving any events')
2395
+ }
2396
+
2397
+ // Log summary if any stalls occurred during streaming
2398
+ if (stallCount > 0) {
2399
+ logForDebugging(
2400
+ `Streaming completed with ${stallCount} stall(s), total stall time: ${(totalStallTime / 1000).toFixed(1)}s`,
2401
+ { level: 'warn' },
2402
+ )
2403
+ logEvent('tengu_streaming_stall_summary', {
2404
+ stall_count: stallCount,
2405
+ total_stall_time_ms: totalStallTime,
2406
+ model:
2407
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2408
+ request_id: (streamRequestId ??
2409
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2410
+ })
2411
+ }
2412
+
2413
+ // Check if the cache actually broke based on response tokens
2414
+ if (feature('PROMPT_CACHE_BREAK_DETECTION')) {
2415
+ void checkResponseForCacheBreak(
2416
+ options.querySource,
2417
+ usage.cache_read_input_tokens,
2418
+ usage.cache_creation_input_tokens,
2419
+ messages,
2420
+ options.agentId,
2421
+ streamRequestId,
2422
+ )
2423
+ }
2424
+
2425
+ // Process fallback percentage header and quota status if available
2426
+ // streamResponse is set when the stream is created in the withRetry callback above
2427
+ // TypeScript's control flow analysis can't track that streamResponse is set in the callback
2428
+ // eslint-disable-next-line eslint-plugin-n/no-unsupported-features/node-builtins
2429
+ const resp = streamResponse as unknown as Response | undefined
2430
+ if (resp) {
2431
+ extractQuotaStatusFromHeaders(resp.headers)
2432
+ // Store headers for gateway detection
2433
+ responseHeaders = resp.headers
2434
+ }
2435
+ } catch (streamingError) {
2436
+ // Clear the idle timeout watchdog on error path too
2437
+ clearStreamIdleTimers()
2438
+
2439
+ // Instrumentation: if the watchdog had already fired and the for-await
2440
+ // threw (rather than exiting cleanly), record that the loop DID exit and
2441
+ // how long after the watchdog. Distinguishes true hangs from error exits.
2442
+ if (streamIdleAborted && streamWatchdogFiredAt !== null) {
2443
+ const exitDelayMs = Math.round(
2444
+ performance.now() - streamWatchdogFiredAt,
2445
+ )
2446
+ logForDiagnosticsNoPII(
2447
+ 'info',
2448
+ 'cli_stream_loop_exited_after_watchdog_error',
2449
+ )
2450
+ logEvent('tengu_stream_loop_exited_after_watchdog', {
2451
+ request_id: (streamRequestId ??
2452
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2453
+ exit_delay_ms: exitDelayMs,
2454
+ exit_path:
2455
+ 'error' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2456
+ error_name:
2457
+ streamingError instanceof Error
2458
+ ? (streamingError.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2459
+ : ('unknown' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS),
2460
+ model:
2461
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2462
+ })
2463
+ }
2464
+
2465
+ if (streamingError instanceof APIUserAbortError) {
2466
+ // Check if the abort signal was triggered by the user (ESC key)
2467
+ // If the signal is aborted, it's a user-initiated abort
2468
+ // If not, it's likely a timeout from the SDK
2469
+ if (signal.aborted) {
2470
+ // This is a real user abort (ESC key was pressed)
2471
+ logForDebugging(
2472
+ `Streaming aborted by user: ${errorMessage(streamingError)}`,
2473
+ )
2474
+ if (isAdvisorInProgress) {
2475
+ logEvent('tengu_advisor_tool_interrupted', {
2476
+ model:
2477
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2478
+ advisor_model: (advisorModel ??
2479
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2480
+ })
2481
+ }
2482
+ throw streamingError
2483
+ } else {
2484
+ // The SDK threw APIUserAbortError but our signal wasn't aborted
2485
+ // This means it's a timeout from the SDK's internal timeout
2486
+ logForDebugging(
2487
+ `Streaming timeout (SDK abort): ${streamingError.message}`,
2488
+ { level: 'error' },
2489
+ )
2490
+ // Throw a more specific error for timeout
2491
+ throw new APIConnectionTimeoutError({ message: 'Request timed out' })
2492
+ }
2493
+ }
2494
+
2495
+ // When the flag is enabled, skip the non-streaming fallback and let the
2496
+ // error propagate to withRetry. The mid-stream fallback causes double tool
2497
+ // execution when streaming tool execution is active: the partial stream
2498
+ // starts a tool, then the non-streaming retry produces the same tool_use
2499
+ // and runs it again. See inc-4258.
2500
+ const disableFallback =
2501
+ isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_NONSTREAMING_FALLBACK) ||
2502
+ getFeatureValue_CACHED_MAY_BE_STALE(
2503
+ 'tengu_disable_streaming_to_non_streaming_fallback',
2504
+ false,
2505
+ )
2506
+
2507
+ if (disableFallback) {
2508
+ logForDebugging(
2509
+ `Error streaming (non-streaming fallback disabled): ${errorMessage(streamingError)}`,
2510
+ { level: 'error' },
2511
+ )
2512
+ logEvent('tengu_streaming_fallback_to_non_streaming', {
2513
+ model:
2514
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2515
+ error:
2516
+ streamingError instanceof Error
2517
+ ? (streamingError.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2518
+ : (String(
2519
+ streamingError,
2520
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS),
2521
+ attemptNumber,
2522
+ maxOutputTokens,
2523
+ thinkingType:
2524
+ thinkingConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2525
+ fallback_disabled: true,
2526
+ request_id: (streamRequestId ??
2527
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2528
+ fallback_cause: (streamIdleAborted
2529
+ ? 'watchdog'
2530
+ : 'other') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2531
+ })
2532
+ throw streamingError
2533
+ }
2534
+
2535
+ logForDebugging(
2536
+ `Error streaming, falling back to non-streaming mode: ${errorMessage(streamingError)}`,
2537
+ { level: 'error' },
2538
+ )
2539
+ didFallBackToNonStreaming = true
2540
+ if (options.onStreamingFallback) {
2541
+ options.onStreamingFallback()
2542
+ }
2543
+
2544
+ logEvent('tengu_streaming_fallback_to_non_streaming', {
2545
+ model:
2546
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2547
+ error:
2548
+ streamingError instanceof Error
2549
+ ? (streamingError.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2550
+ : (String(
2551
+ streamingError,
2552
+ ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS),
2553
+ attemptNumber,
2554
+ maxOutputTokens,
2555
+ thinkingType:
2556
+ thinkingConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2557
+ fallback_disabled: false,
2558
+ request_id: (streamRequestId ??
2559
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2560
+ fallback_cause: (streamIdleAborted
2561
+ ? 'watchdog'
2562
+ : 'other') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2563
+ })
2564
+
2565
+ // Fall back to non-streaming mode with retries.
2566
+ // If the streaming failure was itself a 529, count it toward the
2567
+ // consecutive-529 budget so total 529s-before-model-fallback is the
2568
+ // same whether the overload was hit in streaming or non-streaming mode.
2569
+ // This is a speculative fix for https://github.com/anthropics/claude-code/issues/1513
2570
+ // Instrumentation: proves executeNonStreamingRequest was entered (vs. the
2571
+ // fallback event firing but the call itself hanging at dispatch).
2572
+ logForDiagnosticsNoPII('info', 'cli_nonstreaming_fallback_started')
2573
+ logEvent('tengu_nonstreaming_fallback_started', {
2574
+ request_id: (streamRequestId ??
2575
+ 'unknown') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2576
+ model:
2577
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2578
+ fallback_cause: (streamIdleAborted
2579
+ ? 'watchdog'
2580
+ : 'other') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2581
+ })
2582
+ const result = yield* executeNonStreamingRequest(
2583
+ { model: options.model, source: options.querySource },
2584
+ {
2585
+ model: options.model,
2586
+ fallbackModel: options.fallbackModel,
2587
+ thinkingConfig,
2588
+ ...(isFastModeEnabled() && { fastMode: isFastMode }),
2589
+ signal,
2590
+ initialConsecutive529Errors: is529Error(streamingError) ? 1 : 0,
2591
+ querySource: options.querySource,
2592
+ },
2593
+ paramsFromContext,
2594
+ (attempt, _startTime, tokens) => {
2595
+ attemptNumber = attempt
2596
+ maxOutputTokens = tokens
2597
+ },
2598
+ params => captureAPIRequest(params, options.querySource),
2599
+ streamRequestId,
2600
+ )
2601
+
2602
+ const m: AssistantMessage = {
2603
+ message: {
2604
+ ...result,
2605
+ content: normalizeContentFromAPI(
2606
+ result.content,
2607
+ tools,
2608
+ options.agentId,
2609
+ ),
2610
+ },
2611
+ requestId: streamRequestId ?? undefined,
2612
+ type: 'assistant',
2613
+ uuid: randomUUID(),
2614
+ timestamp: new Date().toISOString(),
2615
+ ...(process.env.USER_TYPE === 'ant' &&
2616
+ research !== undefined && {
2617
+ research,
2618
+ }),
2619
+ ...(advisorModel && {
2620
+ advisorModel,
2621
+ }),
2622
+ }
2623
+ newMessages.push(m)
2624
+ fallbackMessage = m
2625
+ yield m
2626
+ } finally {
2627
+ clearStreamIdleTimers()
2628
+ }
2629
+ } catch (errorFromRetry) {
2630
+ // FallbackTriggeredError must propagate to query.ts, which performs the
2631
+ // actual model switch. Swallowing it here would turn the fallback into a
2632
+ // no-op — the user would just see "Model fallback triggered: X -> Y" as
2633
+ // an error message with no actual retry on the fallback model.
2634
+ if (errorFromRetry instanceof FallbackTriggeredError) {
2635
+ throw errorFromRetry
2636
+ }
2637
+
2638
+ // Check if this is a 404 error during stream creation that should trigger
2639
+ // non-streaming fallback. This handles gateways that return 404 for streaming
2640
+ // endpoints but work fine with non-streaming. Before v2.1.8, BetaMessageStream
2641
+ // threw 404s during iteration (caught by inner catch with fallback), but now
2642
+ // with raw streams, 404s are thrown during creation (caught here).
2643
+ const is404StreamCreationError =
2644
+ !didFallBackToNonStreaming &&
2645
+ errorFromRetry instanceof CannotRetryError &&
2646
+ errorFromRetry.originalError instanceof APIError &&
2647
+ errorFromRetry.originalError.status === 404
2648
+
2649
+ if (is404StreamCreationError) {
2650
+ // 404 is thrown at .withResponse() before streamRequestId is assigned,
2651
+ // and CannotRetryError means every retry failed — so grab the failed
2652
+ // request's ID from the error header instead.
2653
+ const failedRequestId =
2654
+ (errorFromRetry.originalError as APIError).requestID ?? 'unknown'
2655
+ logForDebugging(
2656
+ 'Streaming endpoint returned 404, falling back to non-streaming mode',
2657
+ { level: 'warn' },
2658
+ )
2659
+ didFallBackToNonStreaming = true
2660
+ if (options.onStreamingFallback) {
2661
+ options.onStreamingFallback()
2662
+ }
2663
+
2664
+ logEvent('tengu_streaming_fallback_to_non_streaming', {
2665
+ model:
2666
+ options.model as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2667
+ error:
2668
+ '404_stream_creation' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2669
+ attemptNumber,
2670
+ maxOutputTokens,
2671
+ thinkingType:
2672
+ thinkingConfig.type as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2673
+ request_id:
2674
+ failedRequestId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2675
+ fallback_cause:
2676
+ '404_stream_creation' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2677
+ })
2678
+
2679
+ try {
2680
+ // Fall back to non-streaming mode
2681
+ const result = yield* executeNonStreamingRequest(
2682
+ { model: options.model, source: options.querySource },
2683
+ {
2684
+ model: options.model,
2685
+ fallbackModel: options.fallbackModel,
2686
+ thinkingConfig,
2687
+ ...(isFastModeEnabled() && { fastMode: isFastMode }),
2688
+ signal,
2689
+ },
2690
+ paramsFromContext,
2691
+ (attempt, _startTime, tokens) => {
2692
+ attemptNumber = attempt
2693
+ maxOutputTokens = tokens
2694
+ },
2695
+ params => captureAPIRequest(params, options.querySource),
2696
+ failedRequestId,
2697
+ )
2698
+
2699
+ const m: AssistantMessage = {
2700
+ message: {
2701
+ ...result,
2702
+ content: normalizeContentFromAPI(
2703
+ result.content,
2704
+ tools,
2705
+ options.agentId,
2706
+ ),
2707
+ },
2708
+ requestId: streamRequestId ?? undefined,
2709
+ type: 'assistant',
2710
+ uuid: randomUUID(),
2711
+ timestamp: new Date().toISOString(),
2712
+ ...(process.env.USER_TYPE === 'ant' &&
2713
+ research !== undefined && { research }),
2714
+ ...(advisorModel && { advisorModel }),
2715
+ }
2716
+ newMessages.push(m)
2717
+ fallbackMessage = m
2718
+ yield m
2719
+
2720
+ // Continue to success logging below
2721
+ } catch (fallbackError) {
2722
+ // Propagate model-fallback signal to query.ts (see comment above).
2723
+ if (fallbackError instanceof FallbackTriggeredError) {
2724
+ throw fallbackError
2725
+ }
2726
+
2727
+ // Fallback also failed, handle as normal error
2728
+ logForDebugging(
2729
+ `Non-streaming fallback also failed: ${errorMessage(fallbackError)}`,
2730
+ { level: 'error' },
2731
+ )
2732
+
2733
+ let error = fallbackError
2734
+ let errorModel = options.model
2735
+ if (fallbackError instanceof CannotRetryError) {
2736
+ error = fallbackError.originalError
2737
+ errorModel = fallbackError.retryContext.model
2738
+ }
2739
+
2740
+ if (error instanceof APIError) {
2741
+ extractQuotaStatusFromError(error)
2742
+ }
2743
+
2744
+ const requestId =
2745
+ streamRequestId ||
2746
+ (error instanceof APIError ? error.requestID : undefined) ||
2747
+ (error instanceof APIError
2748
+ ? (error.error as { request_id?: string })?.request_id
2749
+ : undefined)
2750
+
2751
+ logAPIError({
2752
+ error,
2753
+ model: errorModel,
2754
+ messageCount: messagesForAPI.length,
2755
+ messageTokens: tokenCountFromLastAPIResponse(messagesForAPI),
2756
+ durationMs: Date.now() - start,
2757
+ durationMsIncludingRetries: Date.now() - startIncludingRetries,
2758
+ attempt: attemptNumber,
2759
+ requestId,
2760
+ clientRequestId,
2761
+ didFallBackToNonStreaming,
2762
+ queryTracking: options.queryTracking,
2763
+ querySource: options.querySource,
2764
+ llmSpan,
2765
+ fastMode: isFastModeRequest,
2766
+ previousRequestId,
2767
+ })
2768
+
2769
+ if (error instanceof APIUserAbortError) {
2770
+ releaseStreamResources()
2771
+ return
2772
+ }
2773
+
2774
+ yield getAssistantMessageFromError(error, errorModel, {
2775
+ messages,
2776
+ messagesForAPI,
2777
+ })
2778
+ releaseStreamResources()
2779
+ return
2780
+ }
2781
+ } else {
2782
+ // Original error handling for non-404 errors
2783
+ logForDebugging(`Error in API request: ${errorMessage(errorFromRetry)}`, {
2784
+ level: 'error',
2785
+ })
2786
+
2787
+ let error = errorFromRetry
2788
+ let errorModel = options.model
2789
+ if (errorFromRetry instanceof CannotRetryError) {
2790
+ error = errorFromRetry.originalError
2791
+ errorModel = errorFromRetry.retryContext.model
2792
+ }
2793
+
2794
+ // Extract quota status from error headers if it's a rate limit error
2795
+ if (error instanceof APIError) {
2796
+ extractQuotaStatusFromError(error)
2797
+ }
2798
+
2799
+ // Extract requestId from stream, error header, or error body
2800
+ const requestId =
2801
+ streamRequestId ||
2802
+ (error instanceof APIError ? error.requestID : undefined) ||
2803
+ (error instanceof APIError
2804
+ ? (error.error as { request_id?: string })?.request_id
2805
+ : undefined)
2806
+
2807
+ logAPIError({
2808
+ error,
2809
+ model: errorModel,
2810
+ messageCount: messagesForAPI.length,
2811
+ messageTokens: tokenCountFromLastAPIResponse(messagesForAPI),
2812
+ durationMs: Date.now() - start,
2813
+ durationMsIncludingRetries: Date.now() - startIncludingRetries,
2814
+ attempt: attemptNumber,
2815
+ requestId,
2816
+ clientRequestId,
2817
+ didFallBackToNonStreaming,
2818
+ queryTracking: options.queryTracking,
2819
+ querySource: options.querySource,
2820
+ llmSpan,
2821
+ fastMode: isFastModeRequest,
2822
+ previousRequestId,
2823
+ })
2824
+
2825
+ // Don't yield an assistant error message for user aborts
2826
+ // The interruption message is handled in query.ts
2827
+ if (error instanceof APIUserAbortError) {
2828
+ releaseStreamResources()
2829
+ return
2830
+ }
2831
+
2832
+ yield getAssistantMessageFromError(error, errorModel, {
2833
+ messages,
2834
+ messagesForAPI,
2835
+ })
2836
+ releaseStreamResources()
2837
+ return
2838
+ }
2839
+ } finally {
2840
+ stopSessionActivity('api_call')
2841
+ // Must be in the finally block: if the generator is terminated early
2842
+ // via .return() (e.g. consumer breaks out of for-await-of, or query.ts
2843
+ // encounters an abort), code after the try/finally never executes.
2844
+ // Without this, the Response object's native TLS/socket buffers leak
2845
+ // until the generator itself is GC'd (see GH #32920).
2846
+ releaseStreamResources()
2847
+
2848
+ // Non-streaming fallback cost: the streaming path tracks cost in the
2849
+ // message_delta handler before any yield. Fallback pushes to newMessages
2850
+ // then yields, so tracking must be here to survive .return() at the yield.
2851
+ if (fallbackMessage) {
2852
+ const fallbackUsage = fallbackMessage.message.usage
2853
+ usage = updateUsage(EMPTY_USAGE, fallbackUsage)
2854
+ stopReason = fallbackMessage.message.stop_reason
2855
+ const fallbackCost = calculateUSDCost(resolvedModel, fallbackUsage)
2856
+ costUSD += addToTotalSessionCost(
2857
+ fallbackCost,
2858
+ fallbackUsage,
2859
+ options.model,
2860
+ )
2861
+ }
2862
+ }
2863
+
2864
+ // Mark all registered tools as sent to API so they become eligible for deletion
2865
+ if (feature('CACHED_MICROCOMPACT') && cachedMCEnabled) {
2866
+ markToolsSentToAPIState()
2867
+ }
2868
+
2869
+ // Track the last requestId for the main conversation chain so shutdown
2870
+ // can send a cache eviction hint to inference. Exclude backgrounded
2871
+ // sessions (Ctrl+B) which share the repl_main_thread querySource but
2872
+ // run inside an agent context — they are independent conversation chains
2873
+ // whose cache should not be evicted when the foreground session clears.
2874
+ if (
2875
+ streamRequestId &&
2876
+ !getAgentContext() &&
2877
+ (options.querySource.startsWith('repl_main_thread') ||
2878
+ options.querySource === 'sdk')
2879
+ ) {
2880
+ setLastMainRequestId(streamRequestId)
2881
+ }
2882
+
2883
+ // Precompute scalars so the fire-and-forget .then() closure doesn't pin the
2884
+ // full messagesForAPI array (the entire conversation up to the context window
2885
+ // limit) until getToolPermissionContext() resolves.
2886
+ const logMessageCount = messagesForAPI.length
2887
+ const logMessageTokens = tokenCountFromLastAPIResponse(messagesForAPI)
2888
+ void options.getToolPermissionContext().then(permissionContext => {
2889
+ logAPISuccessAndDuration({
2890
+ model:
2891
+ newMessages[0]?.message.model ?? partialMessage?.model ?? options.model,
2892
+ preNormalizedModel: options.model,
2893
+ usage,
2894
+ start,
2895
+ startIncludingRetries,
2896
+ attempt: attemptNumber,
2897
+ messageCount: logMessageCount,
2898
+ messageTokens: logMessageTokens,
2899
+ requestId: streamRequestId ?? null,
2900
+ stopReason,
2901
+ ttftMs,
2902
+ didFallBackToNonStreaming,
2903
+ querySource: options.querySource,
2904
+ headers: responseHeaders,
2905
+ costUSD,
2906
+ queryTracking: options.queryTracking,
2907
+ permissionMode: permissionContext.mode,
2908
+ // Pass newMessages for beta tracing - extraction happens in logging.ts
2909
+ // only when beta tracing is enabled
2910
+ newMessages,
2911
+ llmSpan,
2912
+ globalCacheStrategy,
2913
+ requestSetupMs: start - startIncludingRetries,
2914
+ attemptStartTimes,
2915
+ fastMode: isFastModeRequest,
2916
+ previousRequestId,
2917
+ betas: lastRequestBetas,
2918
+ })
2919
+ })
2920
+
2921
+ // Defensive: also release on normal completion (no-op if finally already ran).
2922
+ releaseStreamResources()
2923
+ }
2924
+
2925
+ /**
2926
+ * Cleans up stream resources to prevent memory leaks.
2927
+ * @internal Exported for testing
2928
+ */
2929
+ export function cleanupStream(
2930
+ stream: Stream<BetaRawMessageStreamEvent> | undefined,
2931
+ ): void {
2932
+ if (!stream) {
2933
+ return
2934
+ }
2935
+ try {
2936
+ // Abort the stream via its controller if not already aborted
2937
+ if (!stream.controller.signal.aborted) {
2938
+ stream.controller.abort()
2939
+ }
2940
+ } catch {
2941
+ // Ignore - stream may already be closed
2942
+ }
2943
+ }
2944
+
2945
+ /**
2946
+ * Updates usage statistics with new values from streaming API events.
2947
+ * Note: Anthropic's streaming API provides cumulative usage totals, not incremental deltas.
2948
+ * Each event contains the complete usage up to that point in the stream.
2949
+ *
2950
+ * Input-related tokens (input_tokens, cache_creation_input_tokens, cache_read_input_tokens)
2951
+ * are typically set in message_start and remain constant. message_delta events may send
2952
+ * explicit 0 values for these fields, which should not overwrite the values from message_start.
2953
+ * We only update these fields if they have a non-null, non-zero value.
2954
+ */
2955
+ export function updateUsage(
2956
+ usage: Readonly<NonNullableUsage>,
2957
+ partUsage: BetaMessageDeltaUsage | undefined,
2958
+ ): NonNullableUsage {
2959
+ if (!partUsage) {
2960
+ return { ...usage }
2961
+ }
2962
+ return {
2963
+ input_tokens:
2964
+ partUsage.input_tokens !== null && partUsage.input_tokens > 0
2965
+ ? partUsage.input_tokens
2966
+ : usage.input_tokens,
2967
+ cache_creation_input_tokens:
2968
+ partUsage.cache_creation_input_tokens !== null &&
2969
+ partUsage.cache_creation_input_tokens > 0
2970
+ ? partUsage.cache_creation_input_tokens
2971
+ : usage.cache_creation_input_tokens,
2972
+ cache_read_input_tokens:
2973
+ partUsage.cache_read_input_tokens !== null &&
2974
+ partUsage.cache_read_input_tokens > 0
2975
+ ? partUsage.cache_read_input_tokens
2976
+ : usage.cache_read_input_tokens,
2977
+ output_tokens: partUsage.output_tokens ?? usage.output_tokens,
2978
+ server_tool_use: {
2979
+ web_search_requests:
2980
+ partUsage.server_tool_use?.web_search_requests ??
2981
+ usage.server_tool_use.web_search_requests,
2982
+ web_fetch_requests:
2983
+ partUsage.server_tool_use?.web_fetch_requests ??
2984
+ usage.server_tool_use.web_fetch_requests,
2985
+ },
2986
+ service_tier: usage.service_tier,
2987
+ cache_creation: {
2988
+ // SDK type BetaMessageDeltaUsage is missing cache_creation, but it's real!
2989
+ ephemeral_1h_input_tokens:
2990
+ (partUsage as BetaUsage).cache_creation?.ephemeral_1h_input_tokens ??
2991
+ usage.cache_creation.ephemeral_1h_input_tokens,
2992
+ ephemeral_5m_input_tokens:
2993
+ (partUsage as BetaUsage).cache_creation?.ephemeral_5m_input_tokens ??
2994
+ usage.cache_creation.ephemeral_5m_input_tokens,
2995
+ },
2996
+ // cache_deleted_input_tokens: returned by the API when cache editing
2997
+ // deletes KV cache content, but not in SDK types. Kept off NonNullableUsage
2998
+ // so the string is eliminated from external builds by dead code elimination.
2999
+ // Uses the same > 0 guard as other token fields to prevent message_delta
3000
+ // from overwriting the real value with 0.
3001
+ ...(feature('CACHED_MICROCOMPACT')
3002
+ ? {
3003
+ cache_deleted_input_tokens:
3004
+ (partUsage as unknown as { cache_deleted_input_tokens?: number })
3005
+ .cache_deleted_input_tokens != null &&
3006
+ (partUsage as unknown as { cache_deleted_input_tokens: number })
3007
+ .cache_deleted_input_tokens > 0
3008
+ ? (partUsage as unknown as { cache_deleted_input_tokens: number })
3009
+ .cache_deleted_input_tokens
3010
+ : ((usage as unknown as { cache_deleted_input_tokens?: number })
3011
+ .cache_deleted_input_tokens ?? 0),
3012
+ }
3013
+ : {}),
3014
+ inference_geo: usage.inference_geo,
3015
+ iterations: partUsage.iterations ?? usage.iterations,
3016
+ speed: (partUsage as BetaUsage).speed ?? usage.speed,
3017
+ }
3018
+ }
3019
+
3020
+ /**
3021
+ * Accumulates usage from one message into a total usage object.
3022
+ * Used to track cumulative usage across multiple assistant turns.
3023
+ */
3024
+ export function accumulateUsage(
3025
+ totalUsage: Readonly<NonNullableUsage>,
3026
+ messageUsage: Readonly<NonNullableUsage>,
3027
+ ): NonNullableUsage {
3028
+ return {
3029
+ input_tokens: totalUsage.input_tokens + messageUsage.input_tokens,
3030
+ cache_creation_input_tokens:
3031
+ totalUsage.cache_creation_input_tokens +
3032
+ messageUsage.cache_creation_input_tokens,
3033
+ cache_read_input_tokens:
3034
+ totalUsage.cache_read_input_tokens + messageUsage.cache_read_input_tokens,
3035
+ output_tokens: totalUsage.output_tokens + messageUsage.output_tokens,
3036
+ server_tool_use: {
3037
+ web_search_requests:
3038
+ totalUsage.server_tool_use.web_search_requests +
3039
+ messageUsage.server_tool_use.web_search_requests,
3040
+ web_fetch_requests:
3041
+ totalUsage.server_tool_use.web_fetch_requests +
3042
+ messageUsage.server_tool_use.web_fetch_requests,
3043
+ },
3044
+ service_tier: messageUsage.service_tier, // Use the most recent service tier
3045
+ cache_creation: {
3046
+ ephemeral_1h_input_tokens:
3047
+ totalUsage.cache_creation.ephemeral_1h_input_tokens +
3048
+ messageUsage.cache_creation.ephemeral_1h_input_tokens,
3049
+ ephemeral_5m_input_tokens:
3050
+ totalUsage.cache_creation.ephemeral_5m_input_tokens +
3051
+ messageUsage.cache_creation.ephemeral_5m_input_tokens,
3052
+ },
3053
+ // See comment in updateUsage — field is not on NonNullableUsage to keep
3054
+ // the string out of external builds.
3055
+ ...(feature('CACHED_MICROCOMPACT')
3056
+ ? {
3057
+ cache_deleted_input_tokens:
3058
+ ((totalUsage as unknown as { cache_deleted_input_tokens?: number })
3059
+ .cache_deleted_input_tokens ?? 0) +
3060
+ ((
3061
+ messageUsage as unknown as { cache_deleted_input_tokens?: number }
3062
+ ).cache_deleted_input_tokens ?? 0),
3063
+ }
3064
+ : {}),
3065
+ inference_geo: messageUsage.inference_geo, // Use the most recent
3066
+ iterations: messageUsage.iterations, // Use the most recent
3067
+ speed: messageUsage.speed, // Use the most recent
3068
+ }
3069
+ }
3070
+
3071
+ function isToolResultBlock(
3072
+ block: unknown,
3073
+ ): block is { type: 'tool_result'; tool_use_id: string } {
3074
+ return (
3075
+ block !== null &&
3076
+ typeof block === 'object' &&
3077
+ 'type' in block &&
3078
+ (block as { type: string }).type === 'tool_result' &&
3079
+ 'tool_use_id' in block
3080
+ )
3081
+ }
3082
+
3083
+ type CachedMCEditsBlock = {
3084
+ type: 'cache_edits'
3085
+ edits: { type: 'delete'; cache_reference: string }[]
3086
+ }
3087
+
3088
+ type CachedMCPinnedEdits = {
3089
+ userMessageIndex: number
3090
+ block: CachedMCEditsBlock
3091
+ }
3092
+
3093
+ // Exported for testing cache_reference placement constraints
3094
+ export function addCacheBreakpoints(
3095
+ messages: (UserMessage | AssistantMessage)[],
3096
+ enablePromptCaching: boolean,
3097
+ querySource?: QuerySource,
3098
+ useCachedMC = false,
3099
+ newCacheEdits?: CachedMCEditsBlock | null,
3100
+ pinnedEdits?: CachedMCPinnedEdits[],
3101
+ skipCacheWrite = false,
3102
+ ): MessageParam[] {
3103
+ logEvent('tengu_api_cache_breakpoints', {
3104
+ totalMessageCount: messages.length,
3105
+ cachingEnabled: enablePromptCaching,
3106
+ skipCacheWrite,
3107
+ })
3108
+
3109
+ // Exactly one message-level cache_control marker per request. Mycro's
3110
+ // turn-to-turn eviction (page_manager/index.rs: Index::insert) frees
3111
+ // local-attention KV pages at any cached prefix position NOT in
3112
+ // cache_store_int_token_boundaries. With two markers the second-to-last
3113
+ // position is protected and its locals survive an extra turn even though
3114
+ // nothing will ever resume from there — with one marker they're freed
3115
+ // immediately. For fire-and-forget forks (skipCacheWrite) we shift the
3116
+ // marker to the second-to-last message: that's the last shared-prefix
3117
+ // point, so the write is a no-op merge on mycro (entry already exists)
3118
+ // and the fork doesn't leave its own tail in the KVCC. Dense pages are
3119
+ // refcounted and survive via the new hash either way.
3120
+ const markerIndex = skipCacheWrite ? messages.length - 2 : messages.length - 1
3121
+ const result = messages.map((msg, index) => {
3122
+ const addCache = index === markerIndex
3123
+ if (msg.type === 'user') {
3124
+ return userMessageToMessageParam(
3125
+ msg,
3126
+ addCache,
3127
+ enablePromptCaching,
3128
+ querySource,
3129
+ )
3130
+ }
3131
+ return assistantMessageToMessageParam(
3132
+ msg,
3133
+ addCache,
3134
+ enablePromptCaching,
3135
+ querySource,
3136
+ )
3137
+ })
3138
+
3139
+ if (!useCachedMC) {
3140
+ return result
3141
+ }
3142
+
3143
+ // Track all cache_references being deleted to prevent duplicates across blocks.
3144
+ const seenDeleteRefs = new Set<string>()
3145
+
3146
+ // Helper to deduplicate a cache_edits block against already-seen deletions
3147
+ const deduplicateEdits = (block: CachedMCEditsBlock): CachedMCEditsBlock => {
3148
+ const uniqueEdits = block.edits.filter(edit => {
3149
+ if (seenDeleteRefs.has(edit.cache_reference)) {
3150
+ return false
3151
+ }
3152
+ seenDeleteRefs.add(edit.cache_reference)
3153
+ return true
3154
+ })
3155
+ return { ...block, edits: uniqueEdits }
3156
+ }
3157
+
3158
+ // Re-insert all previously-pinned cache_edits at their original positions
3159
+ for (const pinned of pinnedEdits ?? []) {
3160
+ const msg = result[pinned.userMessageIndex]
3161
+ if (msg && msg.role === 'user') {
3162
+ if (!Array.isArray(msg.content)) {
3163
+ msg.content = [{ type: 'text', text: msg.content as string }]
3164
+ }
3165
+ const dedupedBlock = deduplicateEdits(pinned.block)
3166
+ if (dedupedBlock.edits.length > 0) {
3167
+ insertBlockAfterToolResults(msg.content, dedupedBlock)
3168
+ }
3169
+ }
3170
+ }
3171
+
3172
+ // Insert new cache_edits into the last user message and pin them
3173
+ if (newCacheEdits && result.length > 0) {
3174
+ const dedupedNewEdits = deduplicateEdits(newCacheEdits)
3175
+ if (dedupedNewEdits.edits.length > 0) {
3176
+ for (let i = result.length - 1; i >= 0; i--) {
3177
+ const msg = result[i]
3178
+ if (msg && msg.role === 'user') {
3179
+ if (!Array.isArray(msg.content)) {
3180
+ msg.content = [{ type: 'text', text: msg.content as string }]
3181
+ }
3182
+ insertBlockAfterToolResults(msg.content, dedupedNewEdits)
3183
+ // Pin so this block is re-sent at the same position in future calls
3184
+ pinCacheEdits(i, newCacheEdits)
3185
+
3186
+ logForDebugging(
3187
+ `Added cache_edits block with ${dedupedNewEdits.edits.length} deletion(s) to message[${i}]: ${dedupedNewEdits.edits.map(e => e.cache_reference).join(', ')}`,
3188
+ )
3189
+ break
3190
+ }
3191
+ }
3192
+ }
3193
+ }
3194
+
3195
+ // Add cache_reference to tool_result blocks that are within the cached prefix.
3196
+ // Must be done AFTER cache_edits insertion since that modifies content arrays.
3197
+ if (enablePromptCaching) {
3198
+ // Find the last message containing a cache_control marker
3199
+ let lastCCMsg = -1
3200
+ for (let i = 0; i < result.length; i++) {
3201
+ const msg = result[i]!
3202
+ if (Array.isArray(msg.content)) {
3203
+ for (const block of msg.content) {
3204
+ if (block && typeof block === 'object' && 'cache_control' in block) {
3205
+ lastCCMsg = i
3206
+ }
3207
+ }
3208
+ }
3209
+ }
3210
+
3211
+ // Add cache_reference to tool_result blocks that are strictly before
3212
+ // the last cache_control marker. The API requires cache_reference to
3213
+ // appear "before or on" the last cache_control — we use strict "before"
3214
+ // to avoid edge cases where cache_edits splicing shifts block indices.
3215
+ //
3216
+ // Create new objects instead of mutating in-place to avoid contaminating
3217
+ // blocks reused by secondary queries that use models without cache_editing support.
3218
+ if (lastCCMsg >= 0) {
3219
+ for (let i = 0; i < lastCCMsg; i++) {
3220
+ const msg = result[i]!
3221
+ if (msg.role !== 'user' || !Array.isArray(msg.content)) {
3222
+ continue
3223
+ }
3224
+ let cloned = false
3225
+ for (let j = 0; j < msg.content.length; j++) {
3226
+ const block = msg.content[j]
3227
+ if (block && isToolResultBlock(block)) {
3228
+ if (!cloned) {
3229
+ msg.content = [...msg.content]
3230
+ cloned = true
3231
+ }
3232
+ msg.content[j] = Object.assign({}, block, {
3233
+ cache_reference: block.tool_use_id,
3234
+ })
3235
+ }
3236
+ }
3237
+ }
3238
+ }
3239
+ }
3240
+
3241
+ return result
3242
+ }
3243
+
3244
+ export function buildSystemPromptBlocks(
3245
+ systemPrompt: SystemPrompt,
3246
+ enablePromptCaching: boolean,
3247
+ options?: {
3248
+ skipGlobalCacheForSystemPrompt?: boolean
3249
+ querySource?: QuerySource
3250
+ },
3251
+ ): TextBlockParam[] {
3252
+ // IMPORTANT: Do not add any more blocks for caching or you will get a 400
3253
+ return splitSysPromptPrefix(systemPrompt, {
3254
+ skipGlobalCacheForSystemPrompt: options?.skipGlobalCacheForSystemPrompt,
3255
+ }).map(block => {
3256
+ return {
3257
+ type: 'text' as const,
3258
+ text: block.text,
3259
+ ...(enablePromptCaching &&
3260
+ block.cacheScope !== null && {
3261
+ cache_control: getCacheControl({
3262
+ scope: block.cacheScope,
3263
+ querySource: options?.querySource,
3264
+ }),
3265
+ }),
3266
+ }
3267
+ })
3268
+ }
3269
+
3270
+ type HaikuOptions = Omit<Options, 'model' | 'getToolPermissionContext'>
3271
+
3272
+ export async function queryHaiku({
3273
+ systemPrompt = asSystemPrompt([]),
3274
+ userPrompt,
3275
+ outputFormat,
3276
+ signal,
3277
+ options,
3278
+ }: {
3279
+ systemPrompt: SystemPrompt
3280
+ userPrompt: string
3281
+ outputFormat?: BetaJSONOutputFormat
3282
+ signal: AbortSignal
3283
+ options: HaikuOptions
3284
+ }): Promise<AssistantMessage> {
3285
+ const result = await withVCR(
3286
+ [
3287
+ createUserMessage({
3288
+ content: systemPrompt.map(text => ({ type: 'text', text })),
3289
+ }),
3290
+ createUserMessage({
3291
+ content: userPrompt,
3292
+ }),
3293
+ ],
3294
+ async () => {
3295
+ const messages = [
3296
+ createUserMessage({
3297
+ content: userPrompt,
3298
+ }),
3299
+ ]
3300
+
3301
+ const result = await queryModelWithoutStreaming({
3302
+ messages,
3303
+ systemPrompt,
3304
+ thinkingConfig: { type: 'disabled' },
3305
+ tools: [],
3306
+ signal,
3307
+ options: {
3308
+ ...options,
3309
+ model: getSmallFastModel(),
3310
+ enablePromptCaching: options.enablePromptCaching ?? false,
3311
+ outputFormat,
3312
+ async getToolPermissionContext() {
3313
+ return getEmptyToolPermissionContext()
3314
+ },
3315
+ },
3316
+ })
3317
+ return [result]
3318
+ },
3319
+ )
3320
+ // We don't use streaming for Haiku so this is safe
3321
+ return result[0]! as AssistantMessage
3322
+ }
3323
+
3324
+ type QueryWithModelOptions = Omit<Options, 'getToolPermissionContext'>
3325
+
3326
+ /**
3327
+ * Query a specific model through the Claude Code infrastructure.
3328
+ * This goes through the full query pipeline including proper authentication,
3329
+ * betas, and headers - unlike direct API calls.
3330
+ */
3331
+ export async function queryWithModel({
3332
+ systemPrompt = asSystemPrompt([]),
3333
+ userPrompt,
3334
+ outputFormat,
3335
+ signal,
3336
+ options,
3337
+ }: {
3338
+ systemPrompt: SystemPrompt
3339
+ userPrompt: string
3340
+ outputFormat?: BetaJSONOutputFormat
3341
+ signal: AbortSignal
3342
+ options: QueryWithModelOptions
3343
+ }): Promise<AssistantMessage> {
3344
+ const result = await withVCR(
3345
+ [
3346
+ createUserMessage({
3347
+ content: systemPrompt.map(text => ({ type: 'text', text })),
3348
+ }),
3349
+ createUserMessage({
3350
+ content: userPrompt,
3351
+ }),
3352
+ ],
3353
+ async () => {
3354
+ const messages = [
3355
+ createUserMessage({
3356
+ content: userPrompt,
3357
+ }),
3358
+ ]
3359
+
3360
+ const result = await queryModelWithoutStreaming({
3361
+ messages,
3362
+ systemPrompt,
3363
+ thinkingConfig: { type: 'disabled' },
3364
+ tools: [],
3365
+ signal,
3366
+ options: {
3367
+ ...options,
3368
+ enablePromptCaching: options.enablePromptCaching ?? false,
3369
+ outputFormat,
3370
+ async getToolPermissionContext() {
3371
+ return getEmptyToolPermissionContext()
3372
+ },
3373
+ },
3374
+ })
3375
+ return [result]
3376
+ },
3377
+ )
3378
+ return result[0]! as AssistantMessage
3379
+ }
3380
+
3381
+ // Non-streaming requests have a 10min max per the docs:
3382
+ // https://platform.claude.com/docs/en/api/errors#long-requests
3383
+ // The SDK's 21333-token cap is derived from 10min × 128k tokens/hour, but we
3384
+ // bypass it by setting a client-level timeout, so we can cap higher.
3385
+ export const MAX_NON_STREAMING_TOKENS = 64_000
3386
+
3387
+ /**
3388
+ * Adjusts thinking budget when max_tokens is capped for non-streaming fallback.
3389
+ * Ensures the API constraint: max_tokens > thinking.budget_tokens
3390
+ *
3391
+ * @param params - The parameters that will be sent to the API
3392
+ * @param maxTokensCap - The maximum allowed tokens (MAX_NON_STREAMING_TOKENS)
3393
+ * @returns Adjusted parameters with thinking budget capped if needed
3394
+ */
3395
+ export function adjustParamsForNonStreaming<
3396
+ T extends {
3397
+ max_tokens: number
3398
+ thinking?: BetaMessageStreamParams['thinking']
3399
+ },
3400
+ >(params: T, maxTokensCap: number): T {
3401
+ const cappedMaxTokens = Math.min(params.max_tokens, maxTokensCap)
3402
+
3403
+ // Adjust thinking budget if it would exceed capped max_tokens
3404
+ // to maintain the constraint: max_tokens > thinking.budget_tokens
3405
+ const adjustedParams = { ...params }
3406
+ if (
3407
+ adjustedParams.thinking?.type === 'enabled' &&
3408
+ adjustedParams.thinking.budget_tokens
3409
+ ) {
3410
+ adjustedParams.thinking = {
3411
+ ...adjustedParams.thinking,
3412
+ budget_tokens: Math.min(
3413
+ adjustedParams.thinking.budget_tokens,
3414
+ cappedMaxTokens - 1, // Must be at least 1 less than max_tokens
3415
+ ),
3416
+ }
3417
+ }
3418
+
3419
+ return {
3420
+ ...adjustedParams,
3421
+ max_tokens: cappedMaxTokens,
3422
+ }
3423
+ }
3424
+
3425
+ function isMaxTokensCapEnabled(): boolean {
3426
+ // 3P default: false (not validated on Bedrock/Vertex)
3427
+ return getFeatureValue_CACHED_MAY_BE_STALE('tengu_otk_slot_v1', false)
3428
+ }
3429
+
3430
+ export function getMaxOutputTokensForModel(model: string): number {
3431
+ const maxOutputTokens = getModelMaxOutputTokens(model)
3432
+
3433
+ // Slot-reservation cap: drop default to 8k for all models. BQ p99 output
3434
+ // = 4,911 tokens; 32k/64k defaults over-reserve 8-16× slot capacity.
3435
+ // Requests hitting the cap get one clean retry at 64k (query.ts
3436
+ // max_output_tokens_escalate). Math.min keeps models with lower native
3437
+ // defaults (e.g. claude-3-opus at 4k) at their native value. Applied
3438
+ // before the env-var override so CLAUDE_CODE_MAX_OUTPUT_TOKENS still wins.
3439
+ const defaultTokens = isMaxTokensCapEnabled()
3440
+ ? Math.min(maxOutputTokens.default, CAPPED_DEFAULT_MAX_TOKENS)
3441
+ : maxOutputTokens.default
3442
+
3443
+ const result = validateBoundedIntEnvVar(
3444
+ 'CLAUDE_CODE_MAX_OUTPUT_TOKENS',
3445
+ process.env.CLAUDE_CODE_MAX_OUTPUT_TOKENS,
3446
+ defaultTokens,
3447
+ maxOutputTokens.upperLimit,
3448
+ )
3449
+ return result.effective
3450
+ }