ashishcode 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 (1936) hide show
  1. package/README.md +189 -0
  2. package/bun.lock +895 -0
  3. package/package.json +109 -0
  4. package/public/claude-files.png +0 -0
  5. package/public/leak-tweet.png +0 -0
  6. package/scripts/ashishcode.mjs +41 -0
  7. package/scripts/claude-local.mjs +41 -0
  8. package/src/QueryEngine.ts +1295 -0
  9. package/src/Task.ts +125 -0
  10. package/src/Tool.ts +792 -0
  11. package/src/assistant/sessionHistory.ts +87 -0
  12. package/src/bootstrap/state.ts +1758 -0
  13. package/src/bridge/bridgeApi.ts +539 -0
  14. package/src/bridge/bridgeConfig.ts +48 -0
  15. package/src/bridge/bridgeDebug.ts +135 -0
  16. package/src/bridge/bridgeEnabled.ts +203 -0
  17. package/src/bridge/bridgeMain.ts +3000 -0
  18. package/src/bridge/bridgeMessaging.ts +461 -0
  19. package/src/bridge/bridgePermissionCallbacks.ts +43 -0
  20. package/src/bridge/bridgePointer.ts +210 -0
  21. package/src/bridge/bridgeStatusUtil.ts +163 -0
  22. package/src/bridge/bridgeUI.ts +531 -0
  23. package/src/bridge/capacityWake.ts +56 -0
  24. package/src/bridge/codeSessionApi.ts +168 -0
  25. package/src/bridge/createSession.ts +384 -0
  26. package/src/bridge/debugUtils.ts +141 -0
  27. package/src/bridge/envLessBridgeConfig.ts +166 -0
  28. package/src/bridge/flushGate.ts +71 -0
  29. package/src/bridge/inboundAttachments.ts +175 -0
  30. package/src/bridge/inboundMessages.ts +80 -0
  31. package/src/bridge/initReplBridge.ts +569 -0
  32. package/src/bridge/jwtUtils.ts +256 -0
  33. package/src/bridge/pollConfig.ts +110 -0
  34. package/src/bridge/pollConfigDefaults.ts +82 -0
  35. package/src/bridge/remoteBridgeCore.ts +1008 -0
  36. package/src/bridge/replBridge.ts +2407 -0
  37. package/src/bridge/replBridgeHandle.ts +36 -0
  38. package/src/bridge/replBridgeTransport.ts +370 -0
  39. package/src/bridge/sessionIdCompat.ts +57 -0
  40. package/src/bridge/sessionRunner.ts +550 -0
  41. package/src/bridge/trustedDevice.ts +210 -0
  42. package/src/bridge/types.ts +262 -0
  43. package/src/bridge/workSecret.ts +127 -0
  44. package/src/buddy/CompanionSprite.tsx +371 -0
  45. package/src/buddy/companion.ts +133 -0
  46. package/src/buddy/prompt.ts +36 -0
  47. package/src/buddy/sprites.ts +514 -0
  48. package/src/buddy/types.ts +148 -0
  49. package/src/buddy/useBuddyNotification.tsx +98 -0
  50. package/src/cli/exit.ts +31 -0
  51. package/src/cli/handlers/agents.ts +70 -0
  52. package/src/cli/handlers/auth.ts +482 -0
  53. package/src/cli/handlers/autoMode.ts +170 -0
  54. package/src/cli/handlers/mcp.tsx +362 -0
  55. package/src/cli/handlers/plugins.ts +878 -0
  56. package/src/cli/handlers/util.tsx +110 -0
  57. package/src/cli/ndjsonSafeStringify.ts +32 -0
  58. package/src/cli/print.ts +5594 -0
  59. package/src/cli/remoteIO.ts +255 -0
  60. package/src/cli/structuredIO.ts +859 -0
  61. package/src/cli/transports/HybridTransport.ts +282 -0
  62. package/src/cli/transports/SSETransport.ts +711 -0
  63. package/src/cli/transports/SerialBatchEventUploader.ts +275 -0
  64. package/src/cli/transports/WebSocketTransport.ts +800 -0
  65. package/src/cli/transports/WorkerStateUploader.ts +131 -0
  66. package/src/cli/transports/ccrClient.ts +998 -0
  67. package/src/cli/transports/transportUtils.ts +45 -0
  68. package/src/cli/update.ts +423 -0
  69. package/src/commands/add-dir/add-dir.tsx +126 -0
  70. package/src/commands/add-dir/index.ts +11 -0
  71. package/src/commands/add-dir/validation.ts +110 -0
  72. package/src/commands/advisor.ts +109 -0
  73. package/src/commands/agents/agents.tsx +12 -0
  74. package/src/commands/agents/index.ts +10 -0
  75. package/src/commands/ant-trace/index.js +1 -0
  76. package/src/commands/autofix-pr/index.js +1 -0
  77. package/src/commands/backfill-sessions/index.js +1 -0
  78. package/src/commands/branch/branch.ts +296 -0
  79. package/src/commands/branch/index.ts +14 -0
  80. package/src/commands/break-cache/index.js +1 -0
  81. package/src/commands/bridge/bridge.tsx +509 -0
  82. package/src/commands/bridge/index.ts +26 -0
  83. package/src/commands/bridge-kick.ts +200 -0
  84. package/src/commands/brief.ts +130 -0
  85. package/src/commands/btw/btw.tsx +243 -0
  86. package/src/commands/btw/index.ts +13 -0
  87. package/src/commands/bughunter/index.js +1 -0
  88. package/src/commands/chrome/chrome.tsx +285 -0
  89. package/src/commands/chrome/index.ts +13 -0
  90. package/src/commands/clear/caches.ts +144 -0
  91. package/src/commands/clear/clear.ts +7 -0
  92. package/src/commands/clear/conversation.ts +251 -0
  93. package/src/commands/clear/index.ts +19 -0
  94. package/src/commands/color/color.ts +93 -0
  95. package/src/commands/color/index.ts +16 -0
  96. package/src/commands/commit-push-pr.ts +158 -0
  97. package/src/commands/commit.ts +92 -0
  98. package/src/commands/compact/compact.ts +287 -0
  99. package/src/commands/compact/index.ts +15 -0
  100. package/src/commands/config/config.tsx +7 -0
  101. package/src/commands/config/index.ts +11 -0
  102. package/src/commands/context/context-noninteractive.ts +325 -0
  103. package/src/commands/context/context.tsx +64 -0
  104. package/src/commands/context/index.ts +24 -0
  105. package/src/commands/copy/copy.tsx +371 -0
  106. package/src/commands/copy/index.ts +15 -0
  107. package/src/commands/cost/cost.ts +24 -0
  108. package/src/commands/cost/index.ts +23 -0
  109. package/src/commands/createMovedToPluginCommand.ts +65 -0
  110. package/src/commands/ctx_viz/index.js +1 -0
  111. package/src/commands/debug-tool-call/index.js +1 -0
  112. package/src/commands/desktop/desktop.tsx +9 -0
  113. package/src/commands/desktop/index.ts +26 -0
  114. package/src/commands/diff/diff.tsx +9 -0
  115. package/src/commands/diff/index.ts +8 -0
  116. package/src/commands/doctor/doctor.tsx +7 -0
  117. package/src/commands/doctor/index.ts +12 -0
  118. package/src/commands/effort/effort.tsx +183 -0
  119. package/src/commands/effort/index.ts +13 -0
  120. package/src/commands/env/index.js +1 -0
  121. package/src/commands/exit/exit.tsx +33 -0
  122. package/src/commands/exit/index.ts +12 -0
  123. package/src/commands/export/export.tsx +91 -0
  124. package/src/commands/export/index.ts +11 -0
  125. package/src/commands/extra-usage/extra-usage-core.ts +118 -0
  126. package/src/commands/extra-usage/extra-usage-noninteractive.ts +16 -0
  127. package/src/commands/extra-usage/extra-usage.tsx +17 -0
  128. package/src/commands/extra-usage/index.ts +31 -0
  129. package/src/commands/fast/fast.tsx +269 -0
  130. package/src/commands/fast/index.ts +26 -0
  131. package/src/commands/feedback/feedback.tsx +25 -0
  132. package/src/commands/feedback/index.ts +28 -0
  133. package/src/commands/files/files.ts +19 -0
  134. package/src/commands/files/index.ts +12 -0
  135. package/src/commands/good-claude/index.js +1 -0
  136. package/src/commands/heapdump/heapdump.ts +17 -0
  137. package/src/commands/heapdump/index.ts +12 -0
  138. package/src/commands/help/help.tsx +11 -0
  139. package/src/commands/help/index.ts +10 -0
  140. package/src/commands/hooks/hooks.tsx +13 -0
  141. package/src/commands/hooks/index.ts +11 -0
  142. package/src/commands/ide/ide.tsx +646 -0
  143. package/src/commands/ide/index.ts +11 -0
  144. package/src/commands/init-verifiers.ts +262 -0
  145. package/src/commands/init.ts +256 -0
  146. package/src/commands/insights.ts +3201 -0
  147. package/src/commands/install-github-app/ApiKeyStep.tsx +231 -0
  148. package/src/commands/install-github-app/CheckExistingSecretStep.tsx +190 -0
  149. package/src/commands/install-github-app/CheckGitHubStep.tsx +15 -0
  150. package/src/commands/install-github-app/ChooseRepoStep.tsx +211 -0
  151. package/src/commands/install-github-app/CreatingStep.tsx +65 -0
  152. package/src/commands/install-github-app/ErrorStep.tsx +85 -0
  153. package/src/commands/install-github-app/ExistingWorkflowStep.tsx +103 -0
  154. package/src/commands/install-github-app/InstallAppStep.tsx +94 -0
  155. package/src/commands/install-github-app/OAuthFlowStep.tsx +276 -0
  156. package/src/commands/install-github-app/SuccessStep.tsx +96 -0
  157. package/src/commands/install-github-app/WarningsStep.tsx +73 -0
  158. package/src/commands/install-github-app/index.ts +13 -0
  159. package/src/commands/install-github-app/install-github-app.tsx +587 -0
  160. package/src/commands/install-github-app/setupGitHubActions.ts +325 -0
  161. package/src/commands/install-slack-app/index.ts +12 -0
  162. package/src/commands/install-slack-app/install-slack-app.ts +30 -0
  163. package/src/commands/install.tsx +300 -0
  164. package/src/commands/issue/index.js +1 -0
  165. package/src/commands/keybindings/index.ts +13 -0
  166. package/src/commands/keybindings/keybindings.ts +53 -0
  167. package/src/commands/login/index.ts +14 -0
  168. package/src/commands/login/login.tsx +147 -0
  169. package/src/commands/logout/index.ts +10 -0
  170. package/src/commands/logout/logout.tsx +82 -0
  171. package/src/commands/mcp/addCommand.ts +280 -0
  172. package/src/commands/mcp/index.ts +12 -0
  173. package/src/commands/mcp/mcp.tsx +85 -0
  174. package/src/commands/mcp/xaaIdpCommand.ts +266 -0
  175. package/src/commands/memory/index.ts +10 -0
  176. package/src/commands/memory/memory.tsx +90 -0
  177. package/src/commands/mobile/index.ts +11 -0
  178. package/src/commands/mobile/mobile.tsx +274 -0
  179. package/src/commands/mock-limits/index.js +1 -0
  180. package/src/commands/model/index.ts +16 -0
  181. package/src/commands/model/model.tsx +383 -0
  182. package/src/commands/oauth-refresh/index.js +1 -0
  183. package/src/commands/onboarding/index.js +1 -0
  184. package/src/commands/output-style/index.ts +11 -0
  185. package/src/commands/output-style/output-style.tsx +7 -0
  186. package/src/commands/passes/index.ts +22 -0
  187. package/src/commands/passes/passes.tsx +24 -0
  188. package/src/commands/perf-issue/index.js +1 -0
  189. package/src/commands/permissions/index.ts +11 -0
  190. package/src/commands/permissions/permissions.tsx +10 -0
  191. package/src/commands/plan/index.ts +11 -0
  192. package/src/commands/plan/plan.tsx +122 -0
  193. package/src/commands/plugin/AddMarketplace.tsx +162 -0
  194. package/src/commands/plugin/BrowseMarketplace.tsx +802 -0
  195. package/src/commands/plugin/DiscoverPlugins.tsx +781 -0
  196. package/src/commands/plugin/ManageMarketplaces.tsx +838 -0
  197. package/src/commands/plugin/ManagePlugins.tsx +2215 -0
  198. package/src/commands/plugin/PluginErrors.tsx +124 -0
  199. package/src/commands/plugin/PluginOptionsDialog.tsx +357 -0
  200. package/src/commands/plugin/PluginOptionsFlow.tsx +135 -0
  201. package/src/commands/plugin/PluginSettings.tsx +1072 -0
  202. package/src/commands/plugin/PluginTrustWarning.tsx +32 -0
  203. package/src/commands/plugin/UnifiedInstalledCell.tsx +565 -0
  204. package/src/commands/plugin/ValidatePlugin.tsx +98 -0
  205. package/src/commands/plugin/index.tsx +11 -0
  206. package/src/commands/plugin/parseArgs.ts +103 -0
  207. package/src/commands/plugin/plugin.tsx +7 -0
  208. package/src/commands/plugin/pluginDetailsHelpers.tsx +117 -0
  209. package/src/commands/plugin/usePagination.ts +171 -0
  210. package/src/commands/pr_comments/index.ts +50 -0
  211. package/src/commands/privacy-settings/index.ts +14 -0
  212. package/src/commands/privacy-settings/privacy-settings.tsx +58 -0
  213. package/src/commands/provider/index.ts +27 -0
  214. package/src/commands/provider/provider.tsx +686 -0
  215. package/src/commands/rate-limit-options/index.ts +19 -0
  216. package/src/commands/rate-limit-options/rate-limit-options.tsx +210 -0
  217. package/src/commands/release-notes/index.ts +11 -0
  218. package/src/commands/release-notes/release-notes.ts +50 -0
  219. package/src/commands/reload-plugins/index.ts +18 -0
  220. package/src/commands/reload-plugins/reload-plugins.ts +61 -0
  221. package/src/commands/remote-env/index.ts +15 -0
  222. package/src/commands/remote-env/remote-env.tsx +7 -0
  223. package/src/commands/remote-setup/api.ts +182 -0
  224. package/src/commands/remote-setup/index.ts +20 -0
  225. package/src/commands/remote-setup/remote-setup.tsx +187 -0
  226. package/src/commands/rename/generateSessionName.ts +67 -0
  227. package/src/commands/rename/index.ts +12 -0
  228. package/src/commands/rename/rename.ts +87 -0
  229. package/src/commands/reset-limits/index.js +4 -0
  230. package/src/commands/resume/index.ts +12 -0
  231. package/src/commands/resume/resume.tsx +275 -0
  232. package/src/commands/review/UltrareviewOverageDialog.tsx +96 -0
  233. package/src/commands/review/reviewRemote.ts +316 -0
  234. package/src/commands/review/ultrareviewCommand.tsx +58 -0
  235. package/src/commands/review/ultrareviewEnabled.ts +14 -0
  236. package/src/commands/review.ts +57 -0
  237. package/src/commands/rewind/index.ts +13 -0
  238. package/src/commands/rewind/rewind.ts +13 -0
  239. package/src/commands/sandbox-toggle/index.ts +50 -0
  240. package/src/commands/sandbox-toggle/sandbox-toggle.tsx +83 -0
  241. package/src/commands/security-review.ts +243 -0
  242. package/src/commands/session/index.ts +16 -0
  243. package/src/commands/session/session.tsx +140 -0
  244. package/src/commands/share/index.js +1 -0
  245. package/src/commands/skills/index.ts +10 -0
  246. package/src/commands/skills/skills.tsx +8 -0
  247. package/src/commands/stats/index.ts +10 -0
  248. package/src/commands/stats/stats.tsx +7 -0
  249. package/src/commands/status/index.ts +12 -0
  250. package/src/commands/status/status.tsx +8 -0
  251. package/src/commands/statusline.tsx +24 -0
  252. package/src/commands/stickers/index.ts +11 -0
  253. package/src/commands/stickers/stickers.ts +16 -0
  254. package/src/commands/summary/index.js +1 -0
  255. package/src/commands/tag/index.ts +12 -0
  256. package/src/commands/tag/tag.tsx +215 -0
  257. package/src/commands/tasks/index.ts +11 -0
  258. package/src/commands/tasks/tasks.tsx +8 -0
  259. package/src/commands/teleport/index.js +1 -0
  260. package/src/commands/terminalSetup/index.ts +23 -0
  261. package/src/commands/terminalSetup/terminalSetup.tsx +531 -0
  262. package/src/commands/theme/index.ts +10 -0
  263. package/src/commands/theme/theme.tsx +57 -0
  264. package/src/commands/thinkback/index.ts +13 -0
  265. package/src/commands/thinkback/thinkback.tsx +554 -0
  266. package/src/commands/thinkback-play/index.ts +17 -0
  267. package/src/commands/thinkback-play/thinkback-play.ts +43 -0
  268. package/src/commands/ultraplan.tsx +471 -0
  269. package/src/commands/upgrade/index.ts +16 -0
  270. package/src/commands/upgrade/upgrade.tsx +38 -0
  271. package/src/commands/usage/index.ts +9 -0
  272. package/src/commands/usage/usage.tsx +7 -0
  273. package/src/commands/version.ts +24 -0
  274. package/src/commands/vim/index.ts +11 -0
  275. package/src/commands/vim/vim.ts +38 -0
  276. package/src/commands/voice/index.ts +20 -0
  277. package/src/commands/voice/voice.ts +150 -0
  278. package/src/commands.ts +756 -0
  279. package/src/components/AgentProgressLine.tsx +136 -0
  280. package/src/components/App.tsx +56 -0
  281. package/src/components/ApproveApiKey.tsx +123 -0
  282. package/src/components/AutoModeOptInDialog.tsx +142 -0
  283. package/src/components/AutoUpdater.tsx +199 -0
  284. package/src/components/AutoUpdaterWrapper.tsx +91 -0
  285. package/src/components/AwsAuthStatusBox.tsx +82 -0
  286. package/src/components/BaseTextInput.tsx +136 -0
  287. package/src/components/BashModeProgress.tsx +56 -0
  288. package/src/components/BridgeDialog.tsx +401 -0
  289. package/src/components/BypassPermissionsModeDialog.tsx +87 -0
  290. package/src/components/ChannelDowngradeDialog.tsx +102 -0
  291. package/src/components/ClaudeCodeHint/PluginHintMenu.tsx +78 -0
  292. package/src/components/ClaudeInChromeOnboarding.tsx +121 -0
  293. package/src/components/ClaudeMdExternalIncludesDialog.tsx +137 -0
  294. package/src/components/ClickableImageRef.tsx +73 -0
  295. package/src/components/CompactSummary.tsx +118 -0
  296. package/src/components/ConfigurableShortcutHint.tsx +57 -0
  297. package/src/components/ConsoleOAuthFlow.tsx +631 -0
  298. package/src/components/ContextSuggestions.tsx +47 -0
  299. package/src/components/ContextVisualization.tsx +489 -0
  300. package/src/components/CoordinatorAgentStatus.tsx +273 -0
  301. package/src/components/CostThresholdDialog.tsx +50 -0
  302. package/src/components/CtrlOToExpand.tsx +51 -0
  303. package/src/components/CustomSelect/SelectMulti.tsx +213 -0
  304. package/src/components/CustomSelect/index.ts +3 -0
  305. package/src/components/CustomSelect/option-map.ts +50 -0
  306. package/src/components/CustomSelect/select-input-option.tsx +488 -0
  307. package/src/components/CustomSelect/select-option.tsx +68 -0
  308. package/src/components/CustomSelect/select.tsx +690 -0
  309. package/src/components/CustomSelect/use-multi-select-state.ts +414 -0
  310. package/src/components/CustomSelect/use-select-input.ts +287 -0
  311. package/src/components/CustomSelect/use-select-navigation.ts +653 -0
  312. package/src/components/CustomSelect/use-select-state.ts +157 -0
  313. package/src/components/DesktopHandoff.tsx +193 -0
  314. package/src/components/DesktopUpsell/DesktopUpsellStartup.tsx +171 -0
  315. package/src/components/DevBar.tsx +49 -0
  316. package/src/components/DevChannelsDialog.tsx +105 -0
  317. package/src/components/DiagnosticsDisplay.tsx +95 -0
  318. package/src/components/EffortCallout.tsx +265 -0
  319. package/src/components/EffortIndicator.ts +42 -0
  320. package/src/components/ExitFlow.tsx +48 -0
  321. package/src/components/ExportDialog.tsx +128 -0
  322. package/src/components/FallbackToolUseErrorMessage.tsx +116 -0
  323. package/src/components/FallbackToolUseRejectedMessage.tsx +16 -0
  324. package/src/components/FastIcon.tsx +46 -0
  325. package/src/components/Feedback.tsx +593 -0
  326. package/src/components/FeedbackSurvey/FeedbackSurvey.tsx +174 -0
  327. package/src/components/FeedbackSurvey/FeedbackSurveyView.tsx +108 -0
  328. package/src/components/FeedbackSurvey/TranscriptSharePrompt.tsx +88 -0
  329. package/src/components/FeedbackSurvey/submitTranscriptShare.ts +113 -0
  330. package/src/components/FeedbackSurvey/useDebouncedDigitInput.ts +82 -0
  331. package/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +296 -0
  332. package/src/components/FeedbackSurvey/useMemorySurvey.tsx +213 -0
  333. package/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +206 -0
  334. package/src/components/FeedbackSurvey/useSurveyState.tsx +100 -0
  335. package/src/components/FileEditToolDiff.tsx +181 -0
  336. package/src/components/FileEditToolUpdatedMessage.tsx +124 -0
  337. package/src/components/FileEditToolUseRejectedMessage.tsx +170 -0
  338. package/src/components/FilePathLink.tsx +43 -0
  339. package/src/components/FullscreenLayout.tsx +637 -0
  340. package/src/components/GlobalSearchDialog.tsx +343 -0
  341. package/src/components/HelpV2/Commands.tsx +82 -0
  342. package/src/components/HelpV2/General.tsx +23 -0
  343. package/src/components/HelpV2/HelpV2.tsx +185 -0
  344. package/src/components/HighlightedCode/Fallback.tsx +193 -0
  345. package/src/components/HighlightedCode.tsx +190 -0
  346. package/src/components/HistorySearchDialog.tsx +118 -0
  347. package/src/components/IdeAutoConnectDialog.tsx +154 -0
  348. package/src/components/IdeOnboardingDialog.tsx +167 -0
  349. package/src/components/IdeStatusIndicator.tsx +58 -0
  350. package/src/components/IdleReturnDialog.tsx +118 -0
  351. package/src/components/InterruptedByUser.tsx +15 -0
  352. package/src/components/InvalidConfigDialog.tsx +156 -0
  353. package/src/components/InvalidSettingsDialog.tsx +89 -0
  354. package/src/components/KeybindingWarnings.tsx +55 -0
  355. package/src/components/LanguagePicker.tsx +86 -0
  356. package/src/components/LogSelector.tsx +1575 -0
  357. package/src/components/LogoV2/AnimatedAsterisk.tsx +50 -0
  358. package/src/components/LogoV2/AnimatedClawd.tsx +124 -0
  359. package/src/components/LogoV2/ChannelsNotice.tsx +266 -0
  360. package/src/components/LogoV2/Clawd.tsx +238 -0
  361. package/src/components/LogoV2/CondensedLogo.tsx +161 -0
  362. package/src/components/LogoV2/EmergencyTip.tsx +58 -0
  363. package/src/components/LogoV2/Feed.tsx +112 -0
  364. package/src/components/LogoV2/FeedColumn.tsx +59 -0
  365. package/src/components/LogoV2/GuestPassesUpsell.tsx +70 -0
  366. package/src/components/LogoV2/LogoV2.tsx +544 -0
  367. package/src/components/LogoV2/Opus1mMergeNotice.tsx +55 -0
  368. package/src/components/LogoV2/OverageCreditUpsell.tsx +166 -0
  369. package/src/components/LogoV2/VoiceModeNotice.tsx +68 -0
  370. package/src/components/LogoV2/WelcomeV2.tsx +434 -0
  371. package/src/components/LogoV2/feedConfigs.tsx +92 -0
  372. package/src/components/LspRecommendation/LspRecommendationMenu.tsx +88 -0
  373. package/src/components/MCPServerApprovalDialog.tsx +115 -0
  374. package/src/components/MCPServerDesktopImportDialog.tsx +203 -0
  375. package/src/components/MCPServerDialogCopy.tsx +15 -0
  376. package/src/components/MCPServerMultiselectDialog.tsx +133 -0
  377. package/src/components/ManagedSettingsSecurityDialog/ManagedSettingsSecurityDialog.tsx +149 -0
  378. package/src/components/ManagedSettingsSecurityDialog/utils.ts +144 -0
  379. package/src/components/Markdown.tsx +236 -0
  380. package/src/components/MarkdownTable.tsx +322 -0
  381. package/src/components/MemoryUsageIndicator.tsx +37 -0
  382. package/src/components/Message.tsx +627 -0
  383. package/src/components/MessageModel.tsx +43 -0
  384. package/src/components/MessageResponse.tsx +78 -0
  385. package/src/components/MessageRow.tsx +383 -0
  386. package/src/components/MessageSelector.tsx +831 -0
  387. package/src/components/MessageTimestamp.tsx +63 -0
  388. package/src/components/Messages.tsx +834 -0
  389. package/src/components/ModelPicker.tsx +458 -0
  390. package/src/components/NativeAutoUpdater.tsx +194 -0
  391. package/src/components/NotebookEditToolUseRejectedMessage.tsx +92 -0
  392. package/src/components/OffscreenFreeze.tsx +44 -0
  393. package/src/components/Onboarding.tsx +635 -0
  394. package/src/components/OutputStylePicker.tsx +112 -0
  395. package/src/components/PackageManagerAutoUpdater.tsx +105 -0
  396. package/src/components/Passes/Passes.tsx +184 -0
  397. package/src/components/PrBadge.tsx +97 -0
  398. package/src/components/PressEnterToContinue.tsx +15 -0
  399. package/src/components/PromptInput/HistorySearchInput.tsx +51 -0
  400. package/src/components/PromptInput/IssueFlagBanner.tsx +12 -0
  401. package/src/components/PromptInput/Notifications.tsx +332 -0
  402. package/src/components/PromptInput/PromptInput.tsx +2366 -0
  403. package/src/components/PromptInput/PromptInputFooter.tsx +191 -0
  404. package/src/components/PromptInput/PromptInputFooterLeftSide.tsx +517 -0
  405. package/src/components/PromptInput/PromptInputFooterSuggestions.tsx +293 -0
  406. package/src/components/PromptInput/PromptInputHelpMenu.tsx +358 -0
  407. package/src/components/PromptInput/PromptInputModeIndicator.tsx +93 -0
  408. package/src/components/PromptInput/PromptInputQueuedCommands.tsx +117 -0
  409. package/src/components/PromptInput/PromptInputStashNotice.tsx +25 -0
  410. package/src/components/PromptInput/SandboxPromptFooterHint.tsx +64 -0
  411. package/src/components/PromptInput/ShimmeredInput.tsx +143 -0
  412. package/src/components/PromptInput/VoiceIndicator.tsx +137 -0
  413. package/src/components/PromptInput/inputModes.ts +33 -0
  414. package/src/components/PromptInput/inputPaste.ts +90 -0
  415. package/src/components/PromptInput/useMaybeTruncateInput.ts +58 -0
  416. package/src/components/PromptInput/usePromptInputPlaceholder.ts +76 -0
  417. package/src/components/PromptInput/useShowFastIconHint.ts +31 -0
  418. package/src/components/PromptInput/useSwarmBanner.ts +155 -0
  419. package/src/components/PromptInput/utils.ts +60 -0
  420. package/src/components/ProviderPicker.tsx +111 -0
  421. package/src/components/QuickOpenDialog.tsx +244 -0
  422. package/src/components/RemoteCallout.tsx +76 -0
  423. package/src/components/RemoteEnvironmentDialog.tsx +340 -0
  424. package/src/components/ResumeTask.tsx +268 -0
  425. package/src/components/SandboxViolationExpandedView.tsx +99 -0
  426. package/src/components/ScrollKeybindingHandler.tsx +1012 -0
  427. package/src/components/SearchBox.tsx +72 -0
  428. package/src/components/SentryErrorBoundary.ts +28 -0
  429. package/src/components/SessionBackgroundHint.tsx +108 -0
  430. package/src/components/SessionPreview.tsx +194 -0
  431. package/src/components/Settings/Config.tsx +1903 -0
  432. package/src/components/Settings/Settings.tsx +137 -0
  433. package/src/components/Settings/Status.tsx +242 -0
  434. package/src/components/Settings/Usage.tsx +377 -0
  435. package/src/components/ShowInIDEPrompt.tsx +170 -0
  436. package/src/components/SkillImprovementSurvey.tsx +152 -0
  437. package/src/components/Spinner/FlashingChar.tsx +61 -0
  438. package/src/components/Spinner/GlimmerMessage.tsx +328 -0
  439. package/src/components/Spinner/ShimmerChar.tsx +36 -0
  440. package/src/components/Spinner/SpinnerAnimationRow.tsx +265 -0
  441. package/src/components/Spinner/SpinnerGlyph.tsx +80 -0
  442. package/src/components/Spinner/TeammateSpinnerLine.tsx +233 -0
  443. package/src/components/Spinner/TeammateSpinnerTree.tsx +272 -0
  444. package/src/components/Spinner/index.ts +10 -0
  445. package/src/components/Spinner/teammateSelectHint.ts +1 -0
  446. package/src/components/Spinner/useShimmerAnimation.ts +31 -0
  447. package/src/components/Spinner/useStalledAnimation.ts +75 -0
  448. package/src/components/Spinner/utils.ts +84 -0
  449. package/src/components/Spinner.tsx +562 -0
  450. package/src/components/Stats.tsx +1228 -0
  451. package/src/components/StatusLine.tsx +325 -0
  452. package/src/components/StatusNotices.tsx +55 -0
  453. package/src/components/StructuredDiff/Fallback.tsx +487 -0
  454. package/src/components/StructuredDiff/colorDiff.ts +37 -0
  455. package/src/components/StructuredDiff.tsx +190 -0
  456. package/src/components/StructuredDiffList.tsx +30 -0
  457. package/src/components/TagTabs.tsx +139 -0
  458. package/src/components/TaskListV2.tsx +378 -0
  459. package/src/components/TeammateViewHeader.tsx +82 -0
  460. package/src/components/TeleportError.tsx +189 -0
  461. package/src/components/TeleportProgress.tsx +140 -0
  462. package/src/components/TeleportRepoMismatchDialog.tsx +104 -0
  463. package/src/components/TeleportResumeWrapper.tsx +167 -0
  464. package/src/components/TeleportStash.tsx +116 -0
  465. package/src/components/TextInput.tsx +124 -0
  466. package/src/components/ThemePicker.tsx +333 -0
  467. package/src/components/ThinkingToggle.tsx +153 -0
  468. package/src/components/TokenWarning.tsx +179 -0
  469. package/src/components/ToolUseLoader.tsx +42 -0
  470. package/src/components/TrustDialog/TrustDialog.tsx +290 -0
  471. package/src/components/TrustDialog/utils.ts +245 -0
  472. package/src/components/ValidationErrorsList.tsx +148 -0
  473. package/src/components/VimTextInput.tsx +140 -0
  474. package/src/components/VirtualMessageList.tsx +1082 -0
  475. package/src/components/WorkflowMultiselectDialog.tsx +128 -0
  476. package/src/components/WorktreeExitDialog.tsx +231 -0
  477. package/src/components/agents/AgentDetail.tsx +220 -0
  478. package/src/components/agents/AgentEditor.tsx +178 -0
  479. package/src/components/agents/AgentNavigationFooter.tsx +26 -0
  480. package/src/components/agents/AgentsList.tsx +440 -0
  481. package/src/components/agents/AgentsMenu.tsx +800 -0
  482. package/src/components/agents/ColorPicker.tsx +112 -0
  483. package/src/components/agents/ModelSelector.tsx +68 -0
  484. package/src/components/agents/ToolSelector.tsx +568 -0
  485. package/src/components/agents/agentFileUtils.ts +272 -0
  486. package/src/components/agents/generateAgent.ts +197 -0
  487. package/src/components/agents/new-agent-creation/CreateAgentWizard.tsx +97 -0
  488. package/src/components/agents/new-agent-creation/wizard-steps/ColorStep.tsx +84 -0
  489. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStep.tsx +378 -0
  490. package/src/components/agents/new-agent-creation/wizard-steps/ConfirmStepWrapper.tsx +74 -0
  491. package/src/components/agents/new-agent-creation/wizard-steps/DescriptionStep.tsx +123 -0
  492. package/src/components/agents/new-agent-creation/wizard-steps/GenerateStep.tsx +143 -0
  493. package/src/components/agents/new-agent-creation/wizard-steps/LocationStep.tsx +80 -0
  494. package/src/components/agents/new-agent-creation/wizard-steps/MemoryStep.tsx +113 -0
  495. package/src/components/agents/new-agent-creation/wizard-steps/MethodStep.tsx +80 -0
  496. package/src/components/agents/new-agent-creation/wizard-steps/ModelStep.tsx +52 -0
  497. package/src/components/agents/new-agent-creation/wizard-steps/PromptStep.tsx +128 -0
  498. package/src/components/agents/new-agent-creation/wizard-steps/ToolsStep.tsx +61 -0
  499. package/src/components/agents/new-agent-creation/wizard-steps/TypeStep.tsx +103 -0
  500. package/src/components/agents/types.ts +27 -0
  501. package/src/components/agents/utils.ts +18 -0
  502. package/src/components/agents/validateAgent.ts +109 -0
  503. package/src/components/design-system/Byline.tsx +77 -0
  504. package/src/components/design-system/Dialog.tsx +138 -0
  505. package/src/components/design-system/Divider.tsx +149 -0
  506. package/src/components/design-system/FuzzyPicker.tsx +312 -0
  507. package/src/components/design-system/KeyboardShortcutHint.tsx +81 -0
  508. package/src/components/design-system/ListItem.tsx +244 -0
  509. package/src/components/design-system/LoadingState.tsx +94 -0
  510. package/src/components/design-system/Pane.tsx +77 -0
  511. package/src/components/design-system/ProgressBar.tsx +86 -0
  512. package/src/components/design-system/Ratchet.tsx +80 -0
  513. package/src/components/design-system/StatusIcon.tsx +95 -0
  514. package/src/components/design-system/Tabs.tsx +340 -0
  515. package/src/components/design-system/ThemeProvider.tsx +170 -0
  516. package/src/components/design-system/ThemedBox.tsx +156 -0
  517. package/src/components/design-system/ThemedText.tsx +124 -0
  518. package/src/components/design-system/color.ts +30 -0
  519. package/src/components/diff/DiffDetailView.tsx +281 -0
  520. package/src/components/diff/DiffDialog.tsx +383 -0
  521. package/src/components/diff/DiffFileList.tsx +292 -0
  522. package/src/components/grove/Grove.tsx +463 -0
  523. package/src/components/hooks/HooksConfigMenu.tsx +578 -0
  524. package/src/components/hooks/PromptDialog.tsx +90 -0
  525. package/src/components/hooks/SelectEventMode.tsx +127 -0
  526. package/src/components/hooks/SelectHookMode.tsx +112 -0
  527. package/src/components/hooks/SelectMatcherMode.tsx +144 -0
  528. package/src/components/hooks/ViewHookMode.tsx +199 -0
  529. package/src/components/mcp/CapabilitiesSection.tsx +61 -0
  530. package/src/components/mcp/ElicitationDialog.tsx +1169 -0
  531. package/src/components/mcp/MCPAgentServerMenu.tsx +183 -0
  532. package/src/components/mcp/MCPListPanel.tsx +504 -0
  533. package/src/components/mcp/MCPReconnect.tsx +167 -0
  534. package/src/components/mcp/MCPRemoteServerMenu.tsx +649 -0
  535. package/src/components/mcp/MCPSettings.tsx +398 -0
  536. package/src/components/mcp/MCPStdioServerMenu.tsx +177 -0
  537. package/src/components/mcp/MCPToolDetailView.tsx +212 -0
  538. package/src/components/mcp/MCPToolListView.tsx +141 -0
  539. package/src/components/mcp/McpParsingWarnings.tsx +213 -0
  540. package/src/components/mcp/index.ts +9 -0
  541. package/src/components/mcp/utils/reconnectHelpers.tsx +49 -0
  542. package/src/components/memory/MemoryFileSelector.tsx +438 -0
  543. package/src/components/memory/MemoryUpdateNotification.tsx +45 -0
  544. package/src/components/messageActions.tsx +450 -0
  545. package/src/components/messages/AdvisorMessage.tsx +158 -0
  546. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +31 -0
  547. package/src/components/messages/AssistantTextMessage.tsx +270 -0
  548. package/src/components/messages/AssistantThinkingMessage.tsx +86 -0
  549. package/src/components/messages/AssistantToolUseMessage.tsx +368 -0
  550. package/src/components/messages/AttachmentMessage.tsx +536 -0
  551. package/src/components/messages/CollapsedReadSearchContent.tsx +484 -0
  552. package/src/components/messages/CompactBoundaryMessage.tsx +18 -0
  553. package/src/components/messages/GroupedToolUseContent.tsx +58 -0
  554. package/src/components/messages/HighlightedThinkingText.tsx +162 -0
  555. package/src/components/messages/HookProgressMessage.tsx +116 -0
  556. package/src/components/messages/PlanApprovalMessage.tsx +222 -0
  557. package/src/components/messages/RateLimitMessage.tsx +161 -0
  558. package/src/components/messages/ShutdownMessage.tsx +132 -0
  559. package/src/components/messages/SystemAPIErrorMessage.tsx +141 -0
  560. package/src/components/messages/SystemTextMessage.tsx +827 -0
  561. package/src/components/messages/TaskAssignmentMessage.tsx +76 -0
  562. package/src/components/messages/UserAgentNotificationMessage.tsx +83 -0
  563. package/src/components/messages/UserBashInputMessage.tsx +58 -0
  564. package/src/components/messages/UserBashOutputMessage.tsx +54 -0
  565. package/src/components/messages/UserChannelMessage.tsx +137 -0
  566. package/src/components/messages/UserCommandMessage.tsx +108 -0
  567. package/src/components/messages/UserImageMessage.tsx +59 -0
  568. package/src/components/messages/UserLocalCommandOutputMessage.tsx +167 -0
  569. package/src/components/messages/UserMemoryInputMessage.tsx +75 -0
  570. package/src/components/messages/UserPlanMessage.tsx +42 -0
  571. package/src/components/messages/UserPromptMessage.tsx +80 -0
  572. package/src/components/messages/UserResourceUpdateMessage.tsx +121 -0
  573. package/src/components/messages/UserTeammateMessage.tsx +206 -0
  574. package/src/components/messages/UserTextMessage.tsx +275 -0
  575. package/src/components/messages/UserToolResultMessage/RejectedPlanMessage.tsx +31 -0
  576. package/src/components/messages/UserToolResultMessage/RejectedToolUseMessage.tsx +16 -0
  577. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +16 -0
  578. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +103 -0
  579. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +95 -0
  580. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +106 -0
  581. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +104 -0
  582. package/src/components/messages/UserToolResultMessage/utils.tsx +44 -0
  583. package/src/components/messages/nullRenderingAttachments.ts +70 -0
  584. package/src/components/messages/teamMemCollapsed.tsx +140 -0
  585. package/src/components/messages/teamMemSaved.ts +19 -0
  586. package/src/components/modelPickerHelpers.test.ts +74 -0
  587. package/src/components/modelPickerHelpers.ts +67 -0
  588. package/src/components/permissions/AskUserQuestionPermissionRequest/AskUserQuestionPermissionRequest.tsx +645 -0
  589. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewBox.tsx +229 -0
  590. package/src/components/permissions/AskUserQuestionPermissionRequest/PreviewQuestionView.tsx +328 -0
  591. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionNavigationBar.tsx +178 -0
  592. package/src/components/permissions/AskUserQuestionPermissionRequest/QuestionView.tsx +465 -0
  593. package/src/components/permissions/AskUserQuestionPermissionRequest/SubmitQuestionsView.tsx +144 -0
  594. package/src/components/permissions/AskUserQuestionPermissionRequest/use-multiple-choice-state.ts +179 -0
  595. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +482 -0
  596. package/src/components/permissions/BashPermissionRequest/bashToolUseOptions.tsx +147 -0
  597. package/src/components/permissions/ComputerUseApproval/ComputerUseApproval.tsx +441 -0
  598. package/src/components/permissions/EnterPlanModePermissionRequest/EnterPlanModePermissionRequest.tsx +122 -0
  599. package/src/components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.tsx +768 -0
  600. package/src/components/permissions/FallbackPermissionRequest.tsx +333 -0
  601. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  602. package/src/components/permissions/FilePermissionDialog/FilePermissionDialog.tsx +204 -0
  603. package/src/components/permissions/FilePermissionDialog/ideDiffConfig.ts +42 -0
  604. package/src/components/permissions/FilePermissionDialog/permissionOptions.tsx +177 -0
  605. package/src/components/permissions/FilePermissionDialog/useFilePermissionDialog.ts +212 -0
  606. package/src/components/permissions/FilePermissionDialog/usePermissionHandler.ts +185 -0
  607. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +161 -0
  608. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +89 -0
  609. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +115 -0
  610. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditPermissionRequest.tsx +166 -0
  611. package/src/components/permissions/NotebookEditPermissionRequest/NotebookEditToolDiff.tsx +235 -0
  612. package/src/components/permissions/PermissionDecisionDebugInfo.tsx +460 -0
  613. package/src/components/permissions/PermissionDialog.tsx +72 -0
  614. package/src/components/permissions/PermissionExplanation.tsx +272 -0
  615. package/src/components/permissions/PermissionPrompt.tsx +336 -0
  616. package/src/components/permissions/PermissionRequest.tsx +217 -0
  617. package/src/components/permissions/PermissionRequestTitle.tsx +66 -0
  618. package/src/components/permissions/PermissionRuleExplanation.tsx +121 -0
  619. package/src/components/permissions/PowerShellPermissionRequest/PowerShellPermissionRequest.tsx +235 -0
  620. package/src/components/permissions/PowerShellPermissionRequest/powershellToolUseOptions.tsx +91 -0
  621. package/src/components/permissions/SandboxPermissionRequest.tsx +163 -0
  622. package/src/components/permissions/SedEditPermissionRequest/SedEditPermissionRequest.tsx +230 -0
  623. package/src/components/permissions/SkillPermissionRequest/SkillPermissionRequest.tsx +369 -0
  624. package/src/components/permissions/WebFetchPermissionRequest/WebFetchPermissionRequest.tsx +258 -0
  625. package/src/components/permissions/WorkerBadge.tsx +49 -0
  626. package/src/components/permissions/WorkerPendingPermission.tsx +105 -0
  627. package/src/components/permissions/hooks.ts +209 -0
  628. package/src/components/permissions/rules/AddPermissionRules.tsx +180 -0
  629. package/src/components/permissions/rules/AddWorkspaceDirectory.tsx +340 -0
  630. package/src/components/permissions/rules/PermissionRuleDescription.tsx +76 -0
  631. package/src/components/permissions/rules/PermissionRuleInput.tsx +138 -0
  632. package/src/components/permissions/rules/PermissionRuleList.tsx +1179 -0
  633. package/src/components/permissions/rules/RecentDenialsTab.tsx +207 -0
  634. package/src/components/permissions/rules/RemoveWorkspaceDirectory.tsx +110 -0
  635. package/src/components/permissions/rules/WorkspaceTab.tsx +150 -0
  636. package/src/components/permissions/shellPermissionHelpers.tsx +164 -0
  637. package/src/components/permissions/useShellPermissionFeedback.ts +148 -0
  638. package/src/components/permissions/utils.ts +25 -0
  639. package/src/components/sandbox/SandboxConfigTab.tsx +45 -0
  640. package/src/components/sandbox/SandboxDependenciesTab.tsx +120 -0
  641. package/src/components/sandbox/SandboxDoctorSection.tsx +46 -0
  642. package/src/components/sandbox/SandboxOverridesTab.tsx +193 -0
  643. package/src/components/sandbox/SandboxSettings.tsx +296 -0
  644. package/src/components/shell/ExpandShellOutputContext.tsx +36 -0
  645. package/src/components/shell/OutputLine.tsx +118 -0
  646. package/src/components/shell/ShellProgressMessage.tsx +150 -0
  647. package/src/components/shell/ShellTimeDisplay.tsx +74 -0
  648. package/src/components/skills/SkillsMenu.tsx +237 -0
  649. package/src/components/tasks/AsyncAgentDetailDialog.tsx +229 -0
  650. package/src/components/tasks/BackgroundTask.tsx +345 -0
  651. package/src/components/tasks/BackgroundTaskStatus.tsx +429 -0
  652. package/src/components/tasks/BackgroundTasksDialog.tsx +652 -0
  653. package/src/components/tasks/DreamDetailDialog.tsx +251 -0
  654. package/src/components/tasks/InProcessTeammateDetailDialog.tsx +266 -0
  655. package/src/components/tasks/RemoteSessionDetailDialog.tsx +904 -0
  656. package/src/components/tasks/RemoteSessionProgress.tsx +243 -0
  657. package/src/components/tasks/ShellDetailDialog.tsx +404 -0
  658. package/src/components/tasks/ShellProgress.tsx +87 -0
  659. package/src/components/tasks/renderToolActivity.tsx +33 -0
  660. package/src/components/tasks/taskStatusUtils.tsx +107 -0
  661. package/src/components/teams/TeamStatus.tsx +80 -0
  662. package/src/components/teams/TeamsDialog.tsx +715 -0
  663. package/src/components/ui/OrderedList.tsx +71 -0
  664. package/src/components/ui/OrderedListItem.tsx +45 -0
  665. package/src/components/ui/TreeSelect.tsx +397 -0
  666. package/src/components/wizard/WizardDialogLayout.tsx +65 -0
  667. package/src/components/wizard/WizardNavigationFooter.tsx +24 -0
  668. package/src/components/wizard/WizardProvider.tsx +213 -0
  669. package/src/components/wizard/index.ts +9 -0
  670. package/src/components/wizard/useWizard.ts +13 -0
  671. package/src/constants/apiLimits.ts +94 -0
  672. package/src/constants/betas.ts +52 -0
  673. package/src/constants/common.ts +33 -0
  674. package/src/constants/cyberRiskInstruction.ts +24 -0
  675. package/src/constants/errorIds.ts +15 -0
  676. package/src/constants/figures.ts +45 -0
  677. package/src/constants/files.ts +156 -0
  678. package/src/constants/github-app.ts +144 -0
  679. package/src/constants/keys.ts +11 -0
  680. package/src/constants/messages.ts +1 -0
  681. package/src/constants/oauth.ts +234 -0
  682. package/src/constants/outputStyles.ts +216 -0
  683. package/src/constants/product.ts +76 -0
  684. package/src/constants/prompts.ts +915 -0
  685. package/src/constants/spinnerVerbs.ts +204 -0
  686. package/src/constants/system.ts +96 -0
  687. package/src/constants/systemPromptSections.ts +68 -0
  688. package/src/constants/toolLimits.ts +56 -0
  689. package/src/constants/tools.ts +112 -0
  690. package/src/constants/turnCompletionVerbs.ts +12 -0
  691. package/src/constants/xml.ts +86 -0
  692. package/src/context/QueuedMessageContext.tsx +63 -0
  693. package/src/context/fpsMetrics.tsx +30 -0
  694. package/src/context/mailbox.tsx +38 -0
  695. package/src/context/modalContext.tsx +58 -0
  696. package/src/context/notifications.tsx +240 -0
  697. package/src/context/overlayContext.tsx +151 -0
  698. package/src/context/promptOverlayContext.tsx +125 -0
  699. package/src/context/stats.tsx +220 -0
  700. package/src/context/voice.tsx +88 -0
  701. package/src/context.ts +189 -0
  702. package/src/coordinator/coordinatorMode.ts +369 -0
  703. package/src/cost-tracker.ts +323 -0
  704. package/src/costHook.ts +22 -0
  705. package/src/dialogLaunchers.tsx +133 -0
  706. package/src/entrypoints/agentSdkTypes.ts +443 -0
  707. package/src/entrypoints/cli.tsx +327 -0
  708. package/src/entrypoints/init.ts +340 -0
  709. package/src/entrypoints/mcp.ts +197 -0
  710. package/src/entrypoints/sandboxTypes.ts +156 -0
  711. package/src/entrypoints/sdk/controlSchemas.ts +663 -0
  712. package/src/entrypoints/sdk/controlTypes.ts +87 -0
  713. package/src/entrypoints/sdk/coreSchemas.ts +1889 -0
  714. package/src/entrypoints/sdk/coreTypes.generated.ts +255 -0
  715. package/src/entrypoints/sdk/coreTypes.ts +62 -0
  716. package/src/entrypoints/sdk/runtimeTypes.ts +29 -0
  717. package/src/entrypoints/sdk/settingsTypes.generated.ts +5 -0
  718. package/src/entrypoints/sdk/toolTypes.ts +5 -0
  719. package/src/history.ts +464 -0
  720. package/src/hooks/fileSuggestions.ts +811 -0
  721. package/src/hooks/notifs/useAutoModeUnavailableNotification.ts +56 -0
  722. package/src/hooks/notifs/useCanSwitchToExistingSubscription.tsx +60 -0
  723. package/src/hooks/notifs/useDeprecationWarningNotification.tsx +44 -0
  724. package/src/hooks/notifs/useFastModeNotification.tsx +162 -0
  725. package/src/hooks/notifs/useIDEStatusIndicator.tsx +186 -0
  726. package/src/hooks/notifs/useInstallMessages.tsx +26 -0
  727. package/src/hooks/notifs/useLspInitializationNotification.tsx +143 -0
  728. package/src/hooks/notifs/useMcpConnectivityStatus.tsx +88 -0
  729. package/src/hooks/notifs/useModelMigrationNotifications.tsx +52 -0
  730. package/src/hooks/notifs/useNpmDeprecationNotification.tsx +25 -0
  731. package/src/hooks/notifs/usePluginAutoupdateNotification.tsx +83 -0
  732. package/src/hooks/notifs/usePluginInstallationStatus.tsx +128 -0
  733. package/src/hooks/notifs/useRateLimitWarningNotification.tsx +114 -0
  734. package/src/hooks/notifs/useSettingsErrors.tsx +69 -0
  735. package/src/hooks/notifs/useStartupNotification.ts +41 -0
  736. package/src/hooks/notifs/useTeammateShutdownNotification.ts +78 -0
  737. package/src/hooks/renderPlaceholder.ts +51 -0
  738. package/src/hooks/toolPermission/PermissionContext.ts +388 -0
  739. package/src/hooks/toolPermission/handlers/coordinatorHandler.ts +65 -0
  740. package/src/hooks/toolPermission/handlers/interactiveHandler.ts +536 -0
  741. package/src/hooks/toolPermission/handlers/swarmWorkerHandler.ts +159 -0
  742. package/src/hooks/toolPermission/permissionLogging.ts +238 -0
  743. package/src/hooks/unifiedSuggestions.ts +202 -0
  744. package/src/hooks/useAfterFirstRender.ts +17 -0
  745. package/src/hooks/useApiKeyVerification.ts +84 -0
  746. package/src/hooks/useArrowKeyHistory.tsx +229 -0
  747. package/src/hooks/useAssistantHistory.ts +250 -0
  748. package/src/hooks/useAwaySummary.ts +125 -0
  749. package/src/hooks/useBackgroundTaskNavigation.ts +251 -0
  750. package/src/hooks/useBlink.ts +34 -0
  751. package/src/hooks/useCanUseTool.tsx +204 -0
  752. package/src/hooks/useCancelRequest.ts +276 -0
  753. package/src/hooks/useChromeExtensionNotification.tsx +50 -0
  754. package/src/hooks/useClaudeCodeHintRecommendation.tsx +129 -0
  755. package/src/hooks/useClipboardImageHint.ts +77 -0
  756. package/src/hooks/useCommandKeybindings.tsx +108 -0
  757. package/src/hooks/useCommandQueue.ts +15 -0
  758. package/src/hooks/useCopyOnSelect.ts +98 -0
  759. package/src/hooks/useDeferredHookMessages.ts +46 -0
  760. package/src/hooks/useDiffData.ts +110 -0
  761. package/src/hooks/useDiffInIDE.ts +379 -0
  762. package/src/hooks/useDirectConnect.ts +229 -0
  763. package/src/hooks/useDoublePress.ts +62 -0
  764. package/src/hooks/useDynamicConfig.ts +22 -0
  765. package/src/hooks/useElapsedTime.ts +37 -0
  766. package/src/hooks/useExitOnCtrlCD.ts +95 -0
  767. package/src/hooks/useExitOnCtrlCDWithKeybindings.ts +24 -0
  768. package/src/hooks/useFileHistorySnapshotInit.ts +25 -0
  769. package/src/hooks/useGlobalKeybindings.tsx +249 -0
  770. package/src/hooks/useHistorySearch.ts +303 -0
  771. package/src/hooks/useIDEIntegration.tsx +70 -0
  772. package/src/hooks/useIdeAtMentioned.ts +76 -0
  773. package/src/hooks/useIdeConnectionStatus.ts +33 -0
  774. package/src/hooks/useIdeLogging.ts +41 -0
  775. package/src/hooks/useIdeSelection.ts +150 -0
  776. package/src/hooks/useInboxPoller.ts +969 -0
  777. package/src/hooks/useInputBuffer.ts +132 -0
  778. package/src/hooks/useIssueFlagBanner.ts +133 -0
  779. package/src/hooks/useLogMessages.ts +119 -0
  780. package/src/hooks/useLspPluginRecommendation.tsx +194 -0
  781. package/src/hooks/useMailboxBridge.ts +21 -0
  782. package/src/hooks/useMainLoopModel.ts +34 -0
  783. package/src/hooks/useManagePlugins.ts +304 -0
  784. package/src/hooks/useMemoryUsage.ts +39 -0
  785. package/src/hooks/useMergedClients.ts +23 -0
  786. package/src/hooks/useMergedCommands.ts +15 -0
  787. package/src/hooks/useMergedTools.ts +44 -0
  788. package/src/hooks/useMinDisplayTime.ts +35 -0
  789. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  790. package/src/hooks/useOfficialMarketplaceNotification.tsx +48 -0
  791. package/src/hooks/usePasteHandler.ts +285 -0
  792. package/src/hooks/usePluginRecommendationBase.tsx +105 -0
  793. package/src/hooks/usePrStatus.ts +106 -0
  794. package/src/hooks/usePromptSuggestion.ts +177 -0
  795. package/src/hooks/usePromptsFromClaudeInChrome.tsx +71 -0
  796. package/src/hooks/useQueueProcessor.ts +68 -0
  797. package/src/hooks/useRemoteSession.ts +605 -0
  798. package/src/hooks/useReplBridge.tsx +723 -0
  799. package/src/hooks/useSSHSession.ts +241 -0
  800. package/src/hooks/useScheduledTasks.ts +139 -0
  801. package/src/hooks/useSearchInput.ts +364 -0
  802. package/src/hooks/useSessionBackgrounding.ts +158 -0
  803. package/src/hooks/useSettings.ts +17 -0
  804. package/src/hooks/useSettingsChange.ts +25 -0
  805. package/src/hooks/useSkillImprovementSurvey.ts +105 -0
  806. package/src/hooks/useSkillsChange.ts +62 -0
  807. package/src/hooks/useSwarmInitialization.ts +81 -0
  808. package/src/hooks/useSwarmPermissionPoller.ts +330 -0
  809. package/src/hooks/useTaskListWatcher.ts +221 -0
  810. package/src/hooks/useTasksV2.ts +250 -0
  811. package/src/hooks/useTeammateViewAutoExit.ts +63 -0
  812. package/src/hooks/useTeleportResume.tsx +85 -0
  813. package/src/hooks/useTerminalSize.ts +15 -0
  814. package/src/hooks/useTextInput.ts +529 -0
  815. package/src/hooks/useTimeout.ts +14 -0
  816. package/src/hooks/useTurnDiffs.ts +213 -0
  817. package/src/hooks/useTypeahead.tsx +1385 -0
  818. package/src/hooks/useUpdateNotification.ts +35 -0
  819. package/src/hooks/useVimInput.ts +316 -0
  820. package/src/hooks/useVirtualScroll.ts +721 -0
  821. package/src/hooks/useVoice.ts +1144 -0
  822. package/src/hooks/useVoiceEnabled.ts +25 -0
  823. package/src/hooks/useVoiceIntegration.tsx +677 -0
  824. package/src/ink/Ansi.tsx +292 -0
  825. package/src/ink/bidi.ts +139 -0
  826. package/src/ink/clearTerminal.ts +74 -0
  827. package/src/ink/colorize.ts +231 -0
  828. package/src/ink/components/AlternateScreen.tsx +80 -0
  829. package/src/ink/components/App.tsx +658 -0
  830. package/src/ink/components/AppContext.ts +21 -0
  831. package/src/ink/components/Box.tsx +214 -0
  832. package/src/ink/components/Button.tsx +192 -0
  833. package/src/ink/components/ClockContext.tsx +112 -0
  834. package/src/ink/components/CursorDeclarationContext.ts +32 -0
  835. package/src/ink/components/ErrorOverview.tsx +109 -0
  836. package/src/ink/components/Link.tsx +42 -0
  837. package/src/ink/components/Newline.tsx +39 -0
  838. package/src/ink/components/NoSelect.tsx +68 -0
  839. package/src/ink/components/RawAnsi.tsx +57 -0
  840. package/src/ink/components/ScrollBox.tsx +237 -0
  841. package/src/ink/components/Spacer.tsx +20 -0
  842. package/src/ink/components/StdinContext.ts +49 -0
  843. package/src/ink/components/TerminalFocusContext.tsx +52 -0
  844. package/src/ink/components/TerminalSizeContext.tsx +7 -0
  845. package/src/ink/components/Text.tsx +254 -0
  846. package/src/ink/constants.ts +2 -0
  847. package/src/ink/dom.ts +484 -0
  848. package/src/ink/events/click-event.ts +38 -0
  849. package/src/ink/events/dispatcher.ts +233 -0
  850. package/src/ink/events/emitter.ts +39 -0
  851. package/src/ink/events/event-handlers.ts +73 -0
  852. package/src/ink/events/event.ts +11 -0
  853. package/src/ink/events/focus-event.ts +21 -0
  854. package/src/ink/events/input-event.ts +205 -0
  855. package/src/ink/events/keyboard-event.ts +51 -0
  856. package/src/ink/events/terminal-event.ts +107 -0
  857. package/src/ink/events/terminal-focus-event.ts +19 -0
  858. package/src/ink/focus.ts +181 -0
  859. package/src/ink/frame.ts +124 -0
  860. package/src/ink/get-max-width.ts +27 -0
  861. package/src/ink/global.ts +2 -0
  862. package/src/ink/hit-test.ts +130 -0
  863. package/src/ink/hooks/use-animation-frame.ts +57 -0
  864. package/src/ink/hooks/use-app.ts +8 -0
  865. package/src/ink/hooks/use-declared-cursor.ts +73 -0
  866. package/src/ink/hooks/use-input.ts +92 -0
  867. package/src/ink/hooks/use-interval.ts +67 -0
  868. package/src/ink/hooks/use-search-highlight.ts +53 -0
  869. package/src/ink/hooks/use-selection.ts +104 -0
  870. package/src/ink/hooks/use-stdin.ts +8 -0
  871. package/src/ink/hooks/use-tab-status.ts +72 -0
  872. package/src/ink/hooks/use-terminal-focus.ts +16 -0
  873. package/src/ink/hooks/use-terminal-title.ts +31 -0
  874. package/src/ink/hooks/use-terminal-viewport.ts +96 -0
  875. package/src/ink/ink.tsx +1723 -0
  876. package/src/ink/instances.ts +10 -0
  877. package/src/ink/layout/engine.ts +6 -0
  878. package/src/ink/layout/geometry.ts +97 -0
  879. package/src/ink/layout/node.ts +152 -0
  880. package/src/ink/layout/yoga.ts +308 -0
  881. package/src/ink/line-width-cache.ts +24 -0
  882. package/src/ink/log-update.ts +773 -0
  883. package/src/ink/measure-element.ts +23 -0
  884. package/src/ink/measure-text.ts +47 -0
  885. package/src/ink/node-cache.ts +54 -0
  886. package/src/ink/optimizer.ts +93 -0
  887. package/src/ink/output.ts +797 -0
  888. package/src/ink/parse-keypress.ts +801 -0
  889. package/src/ink/reconciler.ts +512 -0
  890. package/src/ink/render-border.ts +231 -0
  891. package/src/ink/render-node-to-output.ts +1462 -0
  892. package/src/ink/render-to-screen.ts +231 -0
  893. package/src/ink/renderer.ts +178 -0
  894. package/src/ink/root.ts +184 -0
  895. package/src/ink/screen.ts +1486 -0
  896. package/src/ink/searchHighlight.ts +93 -0
  897. package/src/ink/selection.ts +917 -0
  898. package/src/ink/squash-text-nodes.ts +92 -0
  899. package/src/ink/stringWidth.ts +222 -0
  900. package/src/ink/styles.ts +771 -0
  901. package/src/ink/supports-hyperlinks.ts +57 -0
  902. package/src/ink/tabstops.ts +46 -0
  903. package/src/ink/terminal-focus-state.ts +47 -0
  904. package/src/ink/terminal-querier.ts +212 -0
  905. package/src/ink/terminal.ts +248 -0
  906. package/src/ink/termio/ansi.ts +75 -0
  907. package/src/ink/termio/csi.ts +319 -0
  908. package/src/ink/termio/dec.ts +60 -0
  909. package/src/ink/termio/esc.ts +67 -0
  910. package/src/ink/termio/osc.ts +493 -0
  911. package/src/ink/termio/parser.ts +394 -0
  912. package/src/ink/termio/sgr.ts +308 -0
  913. package/src/ink/termio/tokenize.ts +319 -0
  914. package/src/ink/termio/types.ts +236 -0
  915. package/src/ink/termio.ts +42 -0
  916. package/src/ink/useTerminalNotification.ts +126 -0
  917. package/src/ink/warn.ts +9 -0
  918. package/src/ink/widest-line.ts +19 -0
  919. package/src/ink/wrap-text.ts +74 -0
  920. package/src/ink/wrapAnsi.ts +20 -0
  921. package/src/ink.ts +85 -0
  922. package/src/interactiveHelpers.tsx +367 -0
  923. package/src/keybindings/KeybindingContext.tsx +243 -0
  924. package/src/keybindings/KeybindingProviderSetup.tsx +308 -0
  925. package/src/keybindings/defaultBindings.ts +340 -0
  926. package/src/keybindings/loadUserBindings.ts +472 -0
  927. package/src/keybindings/match.ts +120 -0
  928. package/src/keybindings/parser.ts +203 -0
  929. package/src/keybindings/reservedShortcuts.ts +127 -0
  930. package/src/keybindings/resolver.ts +244 -0
  931. package/src/keybindings/schema.ts +236 -0
  932. package/src/keybindings/shortcutFormat.ts +63 -0
  933. package/src/keybindings/template.ts +52 -0
  934. package/src/keybindings/useKeybinding.ts +196 -0
  935. package/src/keybindings/useShortcutDisplay.ts +59 -0
  936. package/src/keybindings/validate.ts +498 -0
  937. package/src/main.tsx +4756 -0
  938. package/src/memdir/findRelevantMemories.ts +141 -0
  939. package/src/memdir/memdir.ts +507 -0
  940. package/src/memdir/memoryAge.ts +53 -0
  941. package/src/memdir/memoryScan.ts +94 -0
  942. package/src/memdir/memoryTypes.ts +271 -0
  943. package/src/memdir/paths.ts +278 -0
  944. package/src/memdir/teamMemPaths.ts +292 -0
  945. package/src/memdir/teamMemPrompts.ts +100 -0
  946. package/src/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  947. package/src/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  948. package/src/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  949. package/src/migrations/migrateFennecToOpus.ts +45 -0
  950. package/src/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  951. package/src/migrations/migrateOpusToOpus1m.ts +43 -0
  952. package/src/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  953. package/src/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  954. package/src/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  955. package/src/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  956. package/src/migrations/resetProToOpusDefault.ts +51 -0
  957. package/src/moreright/useMoreRight.tsx +26 -0
  958. package/src/native-ts/color-diff/index.ts +999 -0
  959. package/src/native-ts/file-index/index.ts +370 -0
  960. package/src/native-ts/yoga-layout/enums.ts +134 -0
  961. package/src/native-ts/yoga-layout/index.ts +2578 -0
  962. package/src/outputStyles/loadOutputStylesDir.ts +98 -0
  963. package/src/plugins/builtinPlugins.ts +159 -0
  964. package/src/plugins/bundled/index.ts +23 -0
  965. package/src/projectOnboardingState.ts +83 -0
  966. package/src/query/config.ts +48 -0
  967. package/src/query/deps.ts +40 -0
  968. package/src/query/stopHooks.ts +473 -0
  969. package/src/query/tokenBudget.ts +93 -0
  970. package/src/query.ts +1729 -0
  971. package/src/remote/RemoteSessionManager.ts +343 -0
  972. package/src/remote/SessionsWebSocket.ts +404 -0
  973. package/src/remote/remotePermissionBridge.ts +78 -0
  974. package/src/remote/sdkMessageAdapter.ts +302 -0
  975. package/src/replLauncher.tsx +22 -0
  976. package/src/schemas/hooks.ts +222 -0
  977. package/src/screens/Doctor.tsx +575 -0
  978. package/src/screens/REPL.tsx +5052 -0
  979. package/src/screens/ResumeConversation.tsx +399 -0
  980. package/src/server/createDirectConnectSession.ts +88 -0
  981. package/src/server/directConnectManager.ts +213 -0
  982. package/src/server/types.ts +57 -0
  983. package/src/services/AgentSummary/agentSummary.ts +179 -0
  984. package/src/services/MagicDocs/magicDocs.ts +254 -0
  985. package/src/services/MagicDocs/prompts.ts +127 -0
  986. package/src/services/PromptSuggestion/promptSuggestion.ts +523 -0
  987. package/src/services/PromptSuggestion/speculation.ts +991 -0
  988. package/src/services/SessionMemory/prompts.ts +324 -0
  989. package/src/services/SessionMemory/sessionMemory.ts +495 -0
  990. package/src/services/SessionMemory/sessionMemoryUtils.ts +207 -0
  991. package/src/services/analytics/config.ts +40 -0
  992. package/src/services/analytics/datadog.ts +307 -0
  993. package/src/services/analytics/firstPartyEventLogger.ts +450 -0
  994. package/src/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
  995. package/src/services/analytics/growthbook.ts +1155 -0
  996. package/src/services/analytics/index.ts +173 -0
  997. package/src/services/analytics/metadata.ts +974 -0
  998. package/src/services/analytics/sink.ts +114 -0
  999. package/src/services/analytics/sinkKillswitch.ts +25 -0
  1000. package/src/services/api/adminRequests.ts +119 -0
  1001. package/src/services/api/bootstrap.ts +190 -0
  1002. package/src/services/api/claude.ts +3419 -0
  1003. package/src/services/api/client.ts +425 -0
  1004. package/src/services/api/dumpPrompts.ts +226 -0
  1005. package/src/services/api/emptyUsage.ts +22 -0
  1006. package/src/services/api/errorUtils.ts +260 -0
  1007. package/src/services/api/errors.ts +1207 -0
  1008. package/src/services/api/filesApi.ts +748 -0
  1009. package/src/services/api/firstTokenDate.ts +60 -0
  1010. package/src/services/api/grove.ts +357 -0
  1011. package/src/services/api/logging.ts +788 -0
  1012. package/src/services/api/metricsOptOut.ts +159 -0
  1013. package/src/services/api/overageCreditGrant.ts +137 -0
  1014. package/src/services/api/promptCacheBreakDetection.ts +727 -0
  1015. package/src/services/api/referral.ts +281 -0
  1016. package/src/services/api/sessionIngress.ts +514 -0
  1017. package/src/services/api/ultrareviewQuota.ts +38 -0
  1018. package/src/services/api/usage.ts +63 -0
  1019. package/src/services/api/withRetry.ts +822 -0
  1020. package/src/services/autoDream/autoDream.ts +324 -0
  1021. package/src/services/autoDream/config.ts +21 -0
  1022. package/src/services/autoDream/consolidationLock.ts +140 -0
  1023. package/src/services/autoDream/consolidationPrompt.ts +65 -0
  1024. package/src/services/awaySummary.ts +74 -0
  1025. package/src/services/claudeAiLimits.ts +515 -0
  1026. package/src/services/claudeAiLimitsHook.ts +23 -0
  1027. package/src/services/compact/apiMicrocompact.ts +153 -0
  1028. package/src/services/compact/autoCompact.ts +351 -0
  1029. package/src/services/compact/compact.ts +1705 -0
  1030. package/src/services/compact/compactWarningHook.ts +16 -0
  1031. package/src/services/compact/compactWarningState.ts +18 -0
  1032. package/src/services/compact/grouping.ts +63 -0
  1033. package/src/services/compact/microCompact.ts +530 -0
  1034. package/src/services/compact/postCompactCleanup.ts +77 -0
  1035. package/src/services/compact/prompt.ts +374 -0
  1036. package/src/services/compact/sessionMemoryCompact.ts +630 -0
  1037. package/src/services/compact/timeBasedMCConfig.ts +43 -0
  1038. package/src/services/copilot/provider.ts +1603 -0
  1039. package/src/services/diagnosticTracking.ts +397 -0
  1040. package/src/services/extractMemories/extractMemories.ts +615 -0
  1041. package/src/services/extractMemories/prompts.ts +154 -0
  1042. package/src/services/internalLogging.ts +90 -0
  1043. package/src/services/lsp/LSPClient.ts +447 -0
  1044. package/src/services/lsp/LSPDiagnosticRegistry.ts +386 -0
  1045. package/src/services/lsp/LSPServerInstance.ts +511 -0
  1046. package/src/services/lsp/LSPServerManager.ts +420 -0
  1047. package/src/services/lsp/config.ts +79 -0
  1048. package/src/services/lsp/manager.ts +289 -0
  1049. package/src/services/lsp/passiveFeedback.ts +328 -0
  1050. package/src/services/mcp/InProcessTransport.ts +63 -0
  1051. package/src/services/mcp/MCPConnectionManager.tsx +73 -0
  1052. package/src/services/mcp/SdkControlTransport.ts +136 -0
  1053. package/src/services/mcp/auth.ts +2465 -0
  1054. package/src/services/mcp/channelAllowlist.ts +76 -0
  1055. package/src/services/mcp/channelNotification.ts +316 -0
  1056. package/src/services/mcp/channelPermissions.ts +240 -0
  1057. package/src/services/mcp/claudeai.ts +164 -0
  1058. package/src/services/mcp/client.ts +3349 -0
  1059. package/src/services/mcp/config.ts +1578 -0
  1060. package/src/services/mcp/elicitationHandler.ts +313 -0
  1061. package/src/services/mcp/envExpansion.ts +38 -0
  1062. package/src/services/mcp/headersHelper.ts +138 -0
  1063. package/src/services/mcp/mcpStringUtils.ts +106 -0
  1064. package/src/services/mcp/normalization.ts +23 -0
  1065. package/src/services/mcp/oauthPort.ts +78 -0
  1066. package/src/services/mcp/officialRegistry.ts +72 -0
  1067. package/src/services/mcp/types.ts +258 -0
  1068. package/src/services/mcp/useManageMCPConnections.ts +1141 -0
  1069. package/src/services/mcp/utils.ts +575 -0
  1070. package/src/services/mcp/vscodeSdkMcp.ts +112 -0
  1071. package/src/services/mcp/xaa.ts +511 -0
  1072. package/src/services/mcp/xaaIdpLogin.ts +487 -0
  1073. package/src/services/mcpServerApproval.tsx +41 -0
  1074. package/src/services/mockRateLimits.ts +882 -0
  1075. package/src/services/notifier.ts +156 -0
  1076. package/src/services/oauth/auth-code-listener.ts +211 -0
  1077. package/src/services/oauth/client.ts +566 -0
  1078. package/src/services/oauth/crypto.ts +23 -0
  1079. package/src/services/oauth/getOauthProfile.ts +53 -0
  1080. package/src/services/oauth/index.ts +198 -0
  1081. package/src/services/openrouter/provider.ts +129 -0
  1082. package/src/services/plugins/PluginInstallationManager.ts +184 -0
  1083. package/src/services/plugins/pluginCliCommands.ts +344 -0
  1084. package/src/services/plugins/pluginOperations.ts +1088 -0
  1085. package/src/services/policyLimits/index.ts +663 -0
  1086. package/src/services/policyLimits/types.ts +27 -0
  1087. package/src/services/preventSleep.ts +165 -0
  1088. package/src/services/providerCatalog.test.ts +124 -0
  1089. package/src/services/rateLimitMessages.ts +344 -0
  1090. package/src/services/rateLimitMocking.ts +144 -0
  1091. package/src/services/remoteManagedSettings/index.ts +638 -0
  1092. package/src/services/remoteManagedSettings/securityCheck.tsx +74 -0
  1093. package/src/services/remoteManagedSettings/syncCache.ts +112 -0
  1094. package/src/services/remoteManagedSettings/syncCacheState.ts +96 -0
  1095. package/src/services/remoteManagedSettings/types.ts +31 -0
  1096. package/src/services/settingsSync/index.ts +581 -0
  1097. package/src/services/settingsSync/types.ts +67 -0
  1098. package/src/services/teamMemorySync/index.ts +1256 -0
  1099. package/src/services/teamMemorySync/secretScanner.ts +324 -0
  1100. package/src/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
  1101. package/src/services/teamMemorySync/types.ts +156 -0
  1102. package/src/services/teamMemorySync/watcher.ts +387 -0
  1103. package/src/services/tips/tipHistory.ts +17 -0
  1104. package/src/services/tips/tipRegistry.ts +686 -0
  1105. package/src/services/tips/tipScheduler.ts +58 -0
  1106. package/src/services/tokenEstimation.ts +503 -0
  1107. package/src/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
  1108. package/src/services/tools/StreamingToolExecutor.ts +530 -0
  1109. package/src/services/tools/toolExecution.ts +1745 -0
  1110. package/src/services/tools/toolHooks.ts +650 -0
  1111. package/src/services/tools/toolOrchestration.ts +188 -0
  1112. package/src/services/vcr.ts +406 -0
  1113. package/src/services/voice.ts +525 -0
  1114. package/src/services/voiceKeyterms.ts +106 -0
  1115. package/src/services/voiceStreamSTT.ts +544 -0
  1116. package/src/setup.ts +477 -0
  1117. package/src/skills/bundled/batch.ts +124 -0
  1118. package/src/skills/bundled/claudeApi.ts +196 -0
  1119. package/src/skills/bundled/claudeApiContent.ts +75 -0
  1120. package/src/skills/bundled/claudeInChrome.ts +34 -0
  1121. package/src/skills/bundled/debug.ts +103 -0
  1122. package/src/skills/bundled/index.ts +79 -0
  1123. package/src/skills/bundled/keybindings.ts +339 -0
  1124. package/src/skills/bundled/loop.ts +92 -0
  1125. package/src/skills/bundled/loremIpsum.ts +282 -0
  1126. package/src/skills/bundled/remember.ts +82 -0
  1127. package/src/skills/bundled/scheduleRemoteAgents.ts +447 -0
  1128. package/src/skills/bundled/simplify.ts +69 -0
  1129. package/src/skills/bundled/skillify.ts +197 -0
  1130. package/src/skills/bundled/stuck.ts +79 -0
  1131. package/src/skills/bundled/updateConfig.ts +475 -0
  1132. package/src/skills/bundled/verify/SKILL.md +3 -0
  1133. package/src/skills/bundled/verify/examples/cli.md +3 -0
  1134. package/src/skills/bundled/verify/examples/server.md +3 -0
  1135. package/src/skills/bundled/verify.ts +30 -0
  1136. package/src/skills/bundled/verifyContent.ts +13 -0
  1137. package/src/skills/bundledSkills.ts +220 -0
  1138. package/src/skills/loadSkillsDir.ts +1086 -0
  1139. package/src/skills/mcpSkillBuilders.ts +44 -0
  1140. package/src/state/AppState.tsx +200 -0
  1141. package/src/state/AppStateStore.ts +572 -0
  1142. package/src/state/onChangeAppState.ts +181 -0
  1143. package/src/state/selectors.ts +76 -0
  1144. package/src/state/store.ts +34 -0
  1145. package/src/state/teammateViewHelpers.ts +141 -0
  1146. package/src/tasks/DreamTask/DreamTask.ts +157 -0
  1147. package/src/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  1148. package/src/tasks/InProcessTeammateTask/types.ts +121 -0
  1149. package/src/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  1150. package/src/tasks/LocalMainSessionTask.ts +479 -0
  1151. package/src/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  1152. package/src/tasks/LocalShellTask/guards.ts +41 -0
  1153. package/src/tasks/LocalShellTask/killShellTasks.ts +76 -0
  1154. package/src/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  1155. package/src/tasks/pillLabel.ts +82 -0
  1156. package/src/tasks/stopTask.ts +100 -0
  1157. package/src/tasks/types.ts +46 -0
  1158. package/src/tasks.ts +39 -0
  1159. package/src/tools/AgentTool/AgentTool.tsx +1398 -0
  1160. package/src/tools/AgentTool/UI.tsx +872 -0
  1161. package/src/tools/AgentTool/agentColorManager.ts +66 -0
  1162. package/src/tools/AgentTool/agentDisplay.ts +104 -0
  1163. package/src/tools/AgentTool/agentMemory.ts +177 -0
  1164. package/src/tools/AgentTool/agentMemorySnapshot.ts +197 -0
  1165. package/src/tools/AgentTool/agentToolUtils.ts +686 -0
  1166. package/src/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
  1167. package/src/tools/AgentTool/built-in/exploreAgent.ts +83 -0
  1168. package/src/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
  1169. package/src/tools/AgentTool/built-in/planAgent.ts +92 -0
  1170. package/src/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
  1171. package/src/tools/AgentTool/built-in/verificationAgent.ts +152 -0
  1172. package/src/tools/AgentTool/builtInAgents.ts +72 -0
  1173. package/src/tools/AgentTool/constants.ts +12 -0
  1174. package/src/tools/AgentTool/forkSubagent.ts +210 -0
  1175. package/src/tools/AgentTool/loadAgentsDir.ts +755 -0
  1176. package/src/tools/AgentTool/prompt.ts +287 -0
  1177. package/src/tools/AgentTool/resumeAgent.ts +265 -0
  1178. package/src/tools/AgentTool/runAgent.ts +973 -0
  1179. package/src/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
  1180. package/src/tools/AskUserQuestionTool/prompt.ts +44 -0
  1181. package/src/tools/BashTool/BashTool.tsx +1144 -0
  1182. package/src/tools/BashTool/BashToolResultMessage.tsx +191 -0
  1183. package/src/tools/BashTool/UI.tsx +185 -0
  1184. package/src/tools/BashTool/bashCommandHelpers.ts +265 -0
  1185. package/src/tools/BashTool/bashPermissions.ts +2621 -0
  1186. package/src/tools/BashTool/bashSecurity.ts +2592 -0
  1187. package/src/tools/BashTool/commandSemantics.ts +140 -0
  1188. package/src/tools/BashTool/commentLabel.ts +13 -0
  1189. package/src/tools/BashTool/destructiveCommandWarning.ts +102 -0
  1190. package/src/tools/BashTool/modeValidation.ts +115 -0
  1191. package/src/tools/BashTool/pathValidation.ts +1303 -0
  1192. package/src/tools/BashTool/prompt.ts +369 -0
  1193. package/src/tools/BashTool/readOnlyValidation.ts +1990 -0
  1194. package/src/tools/BashTool/sedEditParser.ts +322 -0
  1195. package/src/tools/BashTool/sedValidation.ts +684 -0
  1196. package/src/tools/BashTool/shouldUseSandbox.ts +153 -0
  1197. package/src/tools/BashTool/toolName.ts +2 -0
  1198. package/src/tools/BashTool/utils.ts +223 -0
  1199. package/src/tools/BriefTool/BriefTool.ts +204 -0
  1200. package/src/tools/BriefTool/UI.tsx +101 -0
  1201. package/src/tools/BriefTool/attachments.ts +110 -0
  1202. package/src/tools/BriefTool/prompt.ts +22 -0
  1203. package/src/tools/BriefTool/upload.ts +174 -0
  1204. package/src/tools/ConfigTool/ConfigTool.ts +467 -0
  1205. package/src/tools/ConfigTool/UI.tsx +38 -0
  1206. package/src/tools/ConfigTool/constants.ts +1 -0
  1207. package/src/tools/ConfigTool/prompt.ts +93 -0
  1208. package/src/tools/ConfigTool/supportedSettings.ts +211 -0
  1209. package/src/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
  1210. package/src/tools/EnterPlanModeTool/UI.tsx +33 -0
  1211. package/src/tools/EnterPlanModeTool/constants.ts +1 -0
  1212. package/src/tools/EnterPlanModeTool/prompt.ts +170 -0
  1213. package/src/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
  1214. package/src/tools/EnterWorktreeTool/UI.tsx +20 -0
  1215. package/src/tools/EnterWorktreeTool/constants.ts +1 -0
  1216. package/src/tools/EnterWorktreeTool/prompt.ts +30 -0
  1217. package/src/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
  1218. package/src/tools/ExitPlanModeTool/UI.tsx +82 -0
  1219. package/src/tools/ExitPlanModeTool/constants.ts +2 -0
  1220. package/src/tools/ExitPlanModeTool/prompt.ts +29 -0
  1221. package/src/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
  1222. package/src/tools/ExitWorktreeTool/UI.tsx +25 -0
  1223. package/src/tools/ExitWorktreeTool/constants.ts +1 -0
  1224. package/src/tools/ExitWorktreeTool/prompt.ts +32 -0
  1225. package/src/tools/FileEditTool/FileEditTool.ts +628 -0
  1226. package/src/tools/FileEditTool/UI.tsx +289 -0
  1227. package/src/tools/FileEditTool/constants.ts +11 -0
  1228. package/src/tools/FileEditTool/prompt.ts +28 -0
  1229. package/src/tools/FileEditTool/types.ts +85 -0
  1230. package/src/tools/FileEditTool/utils.ts +775 -0
  1231. package/src/tools/FileReadTool/FileReadTool.ts +1183 -0
  1232. package/src/tools/FileReadTool/UI.tsx +185 -0
  1233. package/src/tools/FileReadTool/imageProcessor.ts +94 -0
  1234. package/src/tools/FileReadTool/limits.ts +92 -0
  1235. package/src/tools/FileReadTool/prompt.ts +49 -0
  1236. package/src/tools/FileWriteTool/FileWriteTool.ts +437 -0
  1237. package/src/tools/FileWriteTool/UI.tsx +405 -0
  1238. package/src/tools/FileWriteTool/prompt.ts +18 -0
  1239. package/src/tools/GlobTool/GlobTool.ts +198 -0
  1240. package/src/tools/GlobTool/UI.tsx +63 -0
  1241. package/src/tools/GlobTool/prompt.ts +7 -0
  1242. package/src/tools/GrepTool/GrepTool.ts +577 -0
  1243. package/src/tools/GrepTool/UI.tsx +201 -0
  1244. package/src/tools/GrepTool/prompt.ts +18 -0
  1245. package/src/tools/LSPTool/LSPTool.ts +860 -0
  1246. package/src/tools/LSPTool/UI.tsx +228 -0
  1247. package/src/tools/LSPTool/formatters.ts +592 -0
  1248. package/src/tools/LSPTool/prompt.ts +21 -0
  1249. package/src/tools/LSPTool/schemas.ts +215 -0
  1250. package/src/tools/LSPTool/symbolContext.ts +90 -0
  1251. package/src/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
  1252. package/src/tools/ListMcpResourcesTool/UI.tsx +29 -0
  1253. package/src/tools/ListMcpResourcesTool/prompt.ts +20 -0
  1254. package/src/tools/MCPTool/MCPTool.ts +77 -0
  1255. package/src/tools/MCPTool/UI.tsx +403 -0
  1256. package/src/tools/MCPTool/classifyForCollapse.ts +604 -0
  1257. package/src/tools/MCPTool/prompt.ts +3 -0
  1258. package/src/tools/McpAuthTool/McpAuthTool.ts +215 -0
  1259. package/src/tools/NotebookEditTool/NotebookEditTool.ts +493 -0
  1260. package/src/tools/NotebookEditTool/UI.tsx +93 -0
  1261. package/src/tools/NotebookEditTool/constants.ts +2 -0
  1262. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  1263. package/src/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
  1264. package/src/tools/PowerShellTool/UI.tsx +131 -0
  1265. package/src/tools/PowerShellTool/clmTypes.ts +211 -0
  1266. package/src/tools/PowerShellTool/commandSemantics.ts +142 -0
  1267. package/src/tools/PowerShellTool/commonParameters.ts +30 -0
  1268. package/src/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
  1269. package/src/tools/PowerShellTool/gitSafety.ts +176 -0
  1270. package/src/tools/PowerShellTool/modeValidation.ts +404 -0
  1271. package/src/tools/PowerShellTool/pathValidation.ts +2049 -0
  1272. package/src/tools/PowerShellTool/powershellPermissions.ts +1648 -0
  1273. package/src/tools/PowerShellTool/powershellSecurity.ts +1090 -0
  1274. package/src/tools/PowerShellTool/prompt.ts +145 -0
  1275. package/src/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
  1276. package/src/tools/PowerShellTool/toolName.ts +2 -0
  1277. package/src/tools/REPLTool/constants.ts +46 -0
  1278. package/src/tools/REPLTool/primitiveTools.ts +39 -0
  1279. package/src/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
  1280. package/src/tools/ReadMcpResourceTool/UI.tsx +37 -0
  1281. package/src/tools/ReadMcpResourceTool/prompt.ts +16 -0
  1282. package/src/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
  1283. package/src/tools/RemoteTriggerTool/UI.tsx +17 -0
  1284. package/src/tools/RemoteTriggerTool/prompt.ts +15 -0
  1285. package/src/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
  1286. package/src/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
  1287. package/src/tools/ScheduleCronTool/CronListTool.ts +97 -0
  1288. package/src/tools/ScheduleCronTool/UI.tsx +60 -0
  1289. package/src/tools/ScheduleCronTool/prompt.ts +135 -0
  1290. package/src/tools/SendMessageTool/SendMessageTool.ts +917 -0
  1291. package/src/tools/SendMessageTool/UI.tsx +31 -0
  1292. package/src/tools/SendMessageTool/constants.ts +1 -0
  1293. package/src/tools/SendMessageTool/prompt.ts +49 -0
  1294. package/src/tools/SkillTool/SkillTool.ts +1108 -0
  1295. package/src/tools/SkillTool/UI.tsx +128 -0
  1296. package/src/tools/SkillTool/constants.ts +1 -0
  1297. package/src/tools/SkillTool/prompt.ts +241 -0
  1298. package/src/tools/SleepTool/prompt.ts +17 -0
  1299. package/src/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
  1300. package/src/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
  1301. package/src/tools/TaskCreateTool/constants.ts +1 -0
  1302. package/src/tools/TaskCreateTool/prompt.ts +56 -0
  1303. package/src/tools/TaskGetTool/TaskGetTool.ts +128 -0
  1304. package/src/tools/TaskGetTool/constants.ts +1 -0
  1305. package/src/tools/TaskGetTool/prompt.ts +24 -0
  1306. package/src/tools/TaskListTool/TaskListTool.ts +116 -0
  1307. package/src/tools/TaskListTool/constants.ts +1 -0
  1308. package/src/tools/TaskListTool/prompt.ts +49 -0
  1309. package/src/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
  1310. package/src/tools/TaskOutputTool/constants.ts +1 -0
  1311. package/src/tools/TaskStopTool/TaskStopTool.ts +131 -0
  1312. package/src/tools/TaskStopTool/UI.tsx +41 -0
  1313. package/src/tools/TaskStopTool/prompt.ts +8 -0
  1314. package/src/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
  1315. package/src/tools/TaskUpdateTool/constants.ts +1 -0
  1316. package/src/tools/TaskUpdateTool/prompt.ts +77 -0
  1317. package/src/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
  1318. package/src/tools/TeamCreateTool/UI.tsx +6 -0
  1319. package/src/tools/TeamCreateTool/constants.ts +1 -0
  1320. package/src/tools/TeamCreateTool/prompt.ts +113 -0
  1321. package/src/tools/TeamDeleteTool/TeamDeleteTool.ts +139 -0
  1322. package/src/tools/TeamDeleteTool/UI.tsx +20 -0
  1323. package/src/tools/TeamDeleteTool/constants.ts +1 -0
  1324. package/src/tools/TeamDeleteTool/prompt.ts +16 -0
  1325. package/src/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
  1326. package/src/tools/TodoWriteTool/constants.ts +1 -0
  1327. package/src/tools/TodoWriteTool/prompt.ts +184 -0
  1328. package/src/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
  1329. package/src/tools/ToolSearchTool/constants.ts +1 -0
  1330. package/src/tools/ToolSearchTool/prompt.ts +121 -0
  1331. package/src/tools/WebFetchTool/UI.tsx +72 -0
  1332. package/src/tools/WebFetchTool/WebFetchTool.ts +318 -0
  1333. package/src/tools/WebFetchTool/preapproved.ts +166 -0
  1334. package/src/tools/WebFetchTool/prompt.ts +46 -0
  1335. package/src/tools/WebFetchTool/utils.ts +530 -0
  1336. package/src/tools/WebSearchTool/UI.tsx +101 -0
  1337. package/src/tools/WebSearchTool/WebSearchTool.ts +444 -0
  1338. package/src/tools/WebSearchTool/prompt.ts +34 -0
  1339. package/src/tools/WorkflowTool/constants.ts +2 -0
  1340. package/src/tools/shared/gitOperationTracking.ts +277 -0
  1341. package/src/tools/shared/spawnMultiAgent.ts +1026 -0
  1342. package/src/tools/testing/TestingPermissionTool.tsx +74 -0
  1343. package/src/tools/utils.ts +40 -0
  1344. package/src/tools.ts +392 -0
  1345. package/src/types/command.ts +216 -0
  1346. package/src/types/connectorText.ts +24 -0
  1347. package/src/types/generated/events_mono/claude_code/v1/claude_code_internal_event.ts +865 -0
  1348. package/src/types/generated/events_mono/common/v1/auth.ts +100 -0
  1349. package/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +223 -0
  1350. package/src/types/generated/google/protobuf/timestamp.ts +187 -0
  1351. package/src/types/hooks.ts +290 -0
  1352. package/src/types/ids.ts +44 -0
  1353. package/src/types/logs.ts +330 -0
  1354. package/src/types/permissions.ts +441 -0
  1355. package/src/types/plugin.ts +363 -0
  1356. package/src/types/textInputTypes.ts +387 -0
  1357. package/src/upstreamproxy/relay.ts +455 -0
  1358. package/src/upstreamproxy/upstreamproxy.ts +285 -0
  1359. package/src/utils/CircularBuffer.ts +84 -0
  1360. package/src/utils/Cursor.ts +1530 -0
  1361. package/src/utils/QueryGuard.ts +121 -0
  1362. package/src/utils/Shell.ts +474 -0
  1363. package/src/utils/ShellCommand.ts +465 -0
  1364. package/src/utils/abortController.ts +99 -0
  1365. package/src/utils/activityManager.ts +164 -0
  1366. package/src/utils/advisor.ts +145 -0
  1367. package/src/utils/agentContext.ts +178 -0
  1368. package/src/utils/agentId.ts +99 -0
  1369. package/src/utils/agentSwarmsEnabled.ts +44 -0
  1370. package/src/utils/agenticSessionSearch.ts +307 -0
  1371. package/src/utils/analyzeContext.ts +1382 -0
  1372. package/src/utils/ansiToPng.ts +334 -0
  1373. package/src/utils/ansiToSvg.ts +272 -0
  1374. package/src/utils/api.ts +718 -0
  1375. package/src/utils/apiPreconnect.ts +75 -0
  1376. package/src/utils/appVersion.ts +6 -0
  1377. package/src/utils/appleTerminalBackup.ts +124 -0
  1378. package/src/utils/argumentSubstitution.ts +145 -0
  1379. package/src/utils/array.ts +13 -0
  1380. package/src/utils/asciicast.ts +239 -0
  1381. package/src/utils/attachments.ts +3997 -0
  1382. package/src/utils/attribution.ts +393 -0
  1383. package/src/utils/auth.ts +2056 -0
  1384. package/src/utils/authFileDescriptor.ts +196 -0
  1385. package/src/utils/authPortable.ts +19 -0
  1386. package/src/utils/autoModeDenials.ts +26 -0
  1387. package/src/utils/autoRunIssue.tsx +122 -0
  1388. package/src/utils/autoUpdater.ts +567 -0
  1389. package/src/utils/aws.ts +74 -0
  1390. package/src/utils/awsAuthStatusManager.ts +81 -0
  1391. package/src/utils/background/remote/preconditions.ts +235 -0
  1392. package/src/utils/background/remote/remoteSession.ts +98 -0
  1393. package/src/utils/backgroundHousekeeping.ts +94 -0
  1394. package/src/utils/bash/ParsedCommand.ts +318 -0
  1395. package/src/utils/bash/ShellSnapshot.ts +582 -0
  1396. package/src/utils/bash/ast.ts +2679 -0
  1397. package/src/utils/bash/bashParser.ts +4436 -0
  1398. package/src/utils/bash/bashPipeCommand.ts +294 -0
  1399. package/src/utils/bash/commands.ts +1339 -0
  1400. package/src/utils/bash/heredoc.ts +733 -0
  1401. package/src/utils/bash/parser.ts +230 -0
  1402. package/src/utils/bash/prefix.ts +204 -0
  1403. package/src/utils/bash/registry.ts +53 -0
  1404. package/src/utils/bash/shellCompletion.ts +259 -0
  1405. package/src/utils/bash/shellPrefix.ts +28 -0
  1406. package/src/utils/bash/shellQuote.ts +304 -0
  1407. package/src/utils/bash/shellQuoting.ts +128 -0
  1408. package/src/utils/bash/specs/alias.ts +14 -0
  1409. package/src/utils/bash/specs/index.ts +18 -0
  1410. package/src/utils/bash/specs/nohup.ts +13 -0
  1411. package/src/utils/bash/specs/pyright.ts +91 -0
  1412. package/src/utils/bash/specs/sleep.ts +13 -0
  1413. package/src/utils/bash/specs/srun.ts +31 -0
  1414. package/src/utils/bash/specs/time.ts +13 -0
  1415. package/src/utils/bash/specs/timeout.ts +20 -0
  1416. package/src/utils/bash/treeSitterAnalysis.ts +506 -0
  1417. package/src/utils/betas.ts +440 -0
  1418. package/src/utils/billing.ts +78 -0
  1419. package/src/utils/binaryCheck.ts +53 -0
  1420. package/src/utils/browser.ts +68 -0
  1421. package/src/utils/bufferedWriter.ts +100 -0
  1422. package/src/utils/bundledMode.ts +22 -0
  1423. package/src/utils/caCerts.ts +115 -0
  1424. package/src/utils/caCertsConfig.ts +88 -0
  1425. package/src/utils/cachePaths.ts +38 -0
  1426. package/src/utils/classifierApprovals.ts +88 -0
  1427. package/src/utils/classifierApprovalsHook.ts +17 -0
  1428. package/src/utils/claudeCodeHints.ts +193 -0
  1429. package/src/utils/claudeDesktop.ts +152 -0
  1430. package/src/utils/claudeInChrome/browserTools.ts +19 -0
  1431. package/src/utils/claudeInChrome/chromeNativeHost.ts +527 -0
  1432. package/src/utils/claudeInChrome/common.ts +540 -0
  1433. package/src/utils/claudeInChrome/mcpServer.ts +293 -0
  1434. package/src/utils/claudeInChrome/prompt.ts +83 -0
  1435. package/src/utils/claudeInChrome/setup.ts +400 -0
  1436. package/src/utils/claudeInChrome/setupPortable.ts +233 -0
  1437. package/src/utils/claudeInChrome/toolRendering.tsx +262 -0
  1438. package/src/utils/claudemd.ts +1479 -0
  1439. package/src/utils/cleanup.ts +602 -0
  1440. package/src/utils/cleanupRegistry.ts +25 -0
  1441. package/src/utils/cliArgs.ts +60 -0
  1442. package/src/utils/cliHighlight.ts +54 -0
  1443. package/src/utils/cliSessionEnv.ts +42 -0
  1444. package/src/utils/codeIndexing.ts +206 -0
  1445. package/src/utils/collapseBackgroundBashNotifications.ts +84 -0
  1446. package/src/utils/collapseHookSummaries.ts +59 -0
  1447. package/src/utils/collapseReadSearch.ts +1109 -0
  1448. package/src/utils/collapseTeammateShutdowns.ts +55 -0
  1449. package/src/utils/combinedAbortSignal.ts +47 -0
  1450. package/src/utils/commandLifecycle.ts +21 -0
  1451. package/src/utils/commitAttribution.ts +961 -0
  1452. package/src/utils/completionCache.ts +166 -0
  1453. package/src/utils/computerUse/appNames.ts +196 -0
  1454. package/src/utils/computerUse/cleanup.ts +86 -0
  1455. package/src/utils/computerUse/common.ts +61 -0
  1456. package/src/utils/computerUse/computerUseLock.ts +215 -0
  1457. package/src/utils/computerUse/drainRunLoop.ts +79 -0
  1458. package/src/utils/computerUse/escHotkey.ts +54 -0
  1459. package/src/utils/computerUse/executor.ts +658 -0
  1460. package/src/utils/computerUse/gates.ts +72 -0
  1461. package/src/utils/computerUse/hostAdapter.ts +69 -0
  1462. package/src/utils/computerUse/inputLoader.ts +30 -0
  1463. package/src/utils/computerUse/mcpServer.ts +106 -0
  1464. package/src/utils/computerUse/setup.ts +53 -0
  1465. package/src/utils/computerUse/swiftLoader.ts +23 -0
  1466. package/src/utils/computerUse/toolRendering.tsx +125 -0
  1467. package/src/utils/computerUse/wrapper.tsx +336 -0
  1468. package/src/utils/concurrentSessions.ts +204 -0
  1469. package/src/utils/config.ts +1817 -0
  1470. package/src/utils/configConstants.ts +21 -0
  1471. package/src/utils/contentArray.ts +51 -0
  1472. package/src/utils/context.ts +221 -0
  1473. package/src/utils/contextAnalysis.ts +272 -0
  1474. package/src/utils/contextSuggestions.ts +235 -0
  1475. package/src/utils/controlMessageCompat.ts +32 -0
  1476. package/src/utils/conversationRecovery.ts +597 -0
  1477. package/src/utils/cron.ts +308 -0
  1478. package/src/utils/cronJitterConfig.ts +75 -0
  1479. package/src/utils/cronScheduler.ts +565 -0
  1480. package/src/utils/cronTasks.ts +458 -0
  1481. package/src/utils/cronTasksLock.ts +195 -0
  1482. package/src/utils/crossProjectResume.ts +75 -0
  1483. package/src/utils/crypto.ts +13 -0
  1484. package/src/utils/cwd.ts +32 -0
  1485. package/src/utils/debug.ts +268 -0
  1486. package/src/utils/debugFilter.ts +157 -0
  1487. package/src/utils/deepLink/banner.ts +123 -0
  1488. package/src/utils/deepLink/parseDeepLink.ts +170 -0
  1489. package/src/utils/deepLink/protocolHandler.ts +136 -0
  1490. package/src/utils/deepLink/registerProtocol.ts +348 -0
  1491. package/src/utils/deepLink/terminalLauncher.ts +557 -0
  1492. package/src/utils/deepLink/terminalPreference.ts +54 -0
  1493. package/src/utils/desktopDeepLink.ts +236 -0
  1494. package/src/utils/detectRepository.ts +178 -0
  1495. package/src/utils/diagLogs.ts +94 -0
  1496. package/src/utils/diff.ts +177 -0
  1497. package/src/utils/directMemberMessage.ts +69 -0
  1498. package/src/utils/displayTags.ts +51 -0
  1499. package/src/utils/doctorContextWarnings.ts +265 -0
  1500. package/src/utils/doctorDiagnostic.ts +626 -0
  1501. package/src/utils/dxt/helpers.ts +88 -0
  1502. package/src/utils/dxt/zip.ts +226 -0
  1503. package/src/utils/earlyInput.ts +221 -0
  1504. package/src/utils/editor.ts +183 -0
  1505. package/src/utils/effort.ts +329 -0
  1506. package/src/utils/embeddedTools.ts +29 -0
  1507. package/src/utils/env.ts +347 -0
  1508. package/src/utils/envDynamic.ts +151 -0
  1509. package/src/utils/envUtils.ts +183 -0
  1510. package/src/utils/envValidation.ts +38 -0
  1511. package/src/utils/errorLogSink.ts +236 -0
  1512. package/src/utils/errors.ts +238 -0
  1513. package/src/utils/exampleCommands.ts +184 -0
  1514. package/src/utils/execFileNoThrow.ts +150 -0
  1515. package/src/utils/execFileNoThrowPortable.ts +89 -0
  1516. package/src/utils/execSyncWrapper.ts +38 -0
  1517. package/src/utils/exportRenderer.tsx +98 -0
  1518. package/src/utils/extraUsage.ts +23 -0
  1519. package/src/utils/fastMode.ts +532 -0
  1520. package/src/utils/file.ts +584 -0
  1521. package/src/utils/fileHistory.ts +1115 -0
  1522. package/src/utils/fileOperationAnalytics.ts +71 -0
  1523. package/src/utils/filePersistence/filePersistence.ts +287 -0
  1524. package/src/utils/filePersistence/outputsScanner.ts +126 -0
  1525. package/src/utils/fileRead.ts +102 -0
  1526. package/src/utils/fileReadCache.ts +96 -0
  1527. package/src/utils/fileStateCache.ts +142 -0
  1528. package/src/utils/findExecutable.ts +17 -0
  1529. package/src/utils/fingerprint.ts +77 -0
  1530. package/src/utils/forkedAgent.ts +689 -0
  1531. package/src/utils/format.ts +308 -0
  1532. package/src/utils/formatBriefTimestamp.ts +81 -0
  1533. package/src/utils/fpsTracker.ts +47 -0
  1534. package/src/utils/frontmatterParser.ts +370 -0
  1535. package/src/utils/fsOperations.ts +770 -0
  1536. package/src/utils/fullscreen.ts +202 -0
  1537. package/src/utils/generatedFiles.ts +136 -0
  1538. package/src/utils/generators.ts +88 -0
  1539. package/src/utils/genericProcessUtils.ts +184 -0
  1540. package/src/utils/getWorktreePaths.ts +70 -0
  1541. package/src/utils/getWorktreePathsPortable.ts +27 -0
  1542. package/src/utils/ghPrStatus.ts +106 -0
  1543. package/src/utils/git/gitConfigParser.ts +277 -0
  1544. package/src/utils/git/gitFilesystem.ts +699 -0
  1545. package/src/utils/git/gitignore.ts +99 -0
  1546. package/src/utils/git.ts +926 -0
  1547. package/src/utils/gitDiff.ts +532 -0
  1548. package/src/utils/gitSettings.ts +18 -0
  1549. package/src/utils/github/ghAuthStatus.ts +29 -0
  1550. package/src/utils/githubRepoPathMapping.ts +162 -0
  1551. package/src/utils/glob.ts +130 -0
  1552. package/src/utils/gracefulShutdown.ts +529 -0
  1553. package/src/utils/groupToolUses.ts +182 -0
  1554. package/src/utils/handlePromptSubmit.ts +610 -0
  1555. package/src/utils/hash.ts +46 -0
  1556. package/src/utils/headlessProfiler.ts +178 -0
  1557. package/src/utils/heapDumpService.ts +304 -0
  1558. package/src/utils/heatmap.ts +198 -0
  1559. package/src/utils/highlightMatch.tsx +28 -0
  1560. package/src/utils/hooks/AsyncHookRegistry.ts +309 -0
  1561. package/src/utils/hooks/apiQueryHookHelper.ts +141 -0
  1562. package/src/utils/hooks/execAgentHook.ts +339 -0
  1563. package/src/utils/hooks/execHttpHook.ts +242 -0
  1564. package/src/utils/hooks/execPromptHook.ts +211 -0
  1565. package/src/utils/hooks/fileChangedWatcher.ts +191 -0
  1566. package/src/utils/hooks/hookEvents.ts +192 -0
  1567. package/src/utils/hooks/hookHelpers.ts +83 -0
  1568. package/src/utils/hooks/hooksConfigManager.ts +400 -0
  1569. package/src/utils/hooks/hooksConfigSnapshot.ts +133 -0
  1570. package/src/utils/hooks/hooksSettings.ts +271 -0
  1571. package/src/utils/hooks/postSamplingHooks.ts +70 -0
  1572. package/src/utils/hooks/registerFrontmatterHooks.ts +67 -0
  1573. package/src/utils/hooks/registerSkillHooks.ts +64 -0
  1574. package/src/utils/hooks/sessionHooks.ts +447 -0
  1575. package/src/utils/hooks/skillImprovement.ts +267 -0
  1576. package/src/utils/hooks/ssrfGuard.ts +294 -0
  1577. package/src/utils/hooks.ts +5022 -0
  1578. package/src/utils/horizontalScroll.ts +137 -0
  1579. package/src/utils/http.ts +137 -0
  1580. package/src/utils/hyperlink.ts +39 -0
  1581. package/src/utils/iTermBackup.ts +73 -0
  1582. package/src/utils/ide.ts +1495 -0
  1583. package/src/utils/idePathConversion.ts +90 -0
  1584. package/src/utils/idleTimeout.ts +53 -0
  1585. package/src/utils/imagePaste.ts +416 -0
  1586. package/src/utils/imageResizer.ts +880 -0
  1587. package/src/utils/imageStore.ts +167 -0
  1588. package/src/utils/imageValidation.ts +104 -0
  1589. package/src/utils/immediateCommand.ts +15 -0
  1590. package/src/utils/inProcessTeammateHelpers.ts +102 -0
  1591. package/src/utils/ink.ts +26 -0
  1592. package/src/utils/intl.ts +94 -0
  1593. package/src/utils/jetbrains.ts +191 -0
  1594. package/src/utils/json.ts +277 -0
  1595. package/src/utils/jsonRead.ts +16 -0
  1596. package/src/utils/keyboardShortcuts.ts +14 -0
  1597. package/src/utils/lazySchema.ts +8 -0
  1598. package/src/utils/listSessionsImpl.ts +454 -0
  1599. package/src/utils/localInstaller.ts +162 -0
  1600. package/src/utils/lockfile.ts +43 -0
  1601. package/src/utils/log.ts +364 -0
  1602. package/src/utils/logoV2Utils.ts +351 -0
  1603. package/src/utils/mailbox.ts +73 -0
  1604. package/src/utils/managedEnv.ts +199 -0
  1605. package/src/utils/managedEnvConstants.ts +205 -0
  1606. package/src/utils/markdown.ts +381 -0
  1607. package/src/utils/markdownConfigLoader.ts +600 -0
  1608. package/src/utils/mcp/dateTimeParser.ts +121 -0
  1609. package/src/utils/mcp/elicitationValidation.ts +336 -0
  1610. package/src/utils/mcpInstructionsDelta.ts +130 -0
  1611. package/src/utils/mcpOutputStorage.ts +189 -0
  1612. package/src/utils/mcpValidation.ts +208 -0
  1613. package/src/utils/mcpWebSocketTransport.ts +200 -0
  1614. package/src/utils/memoize.ts +269 -0
  1615. package/src/utils/memory/types.ts +12 -0
  1616. package/src/utils/memory/versions.ts +8 -0
  1617. package/src/utils/memoryFileDetection.ts +289 -0
  1618. package/src/utils/messagePredicates.ts +8 -0
  1619. package/src/utils/messageQueueManager.ts +547 -0
  1620. package/src/utils/messages/mappers.ts +290 -0
  1621. package/src/utils/messages/systemInit.ts +97 -0
  1622. package/src/utils/messages.ts +5512 -0
  1623. package/src/utils/model/agent.ts +157 -0
  1624. package/src/utils/model/aliases.ts +25 -0
  1625. package/src/utils/model/antModels.ts +64 -0
  1626. package/src/utils/model/bedrock.ts +265 -0
  1627. package/src/utils/model/check1mAccess.ts +72 -0
  1628. package/src/utils/model/configs.ts +140 -0
  1629. package/src/utils/model/contextWindowUpgradeCheck.ts +47 -0
  1630. package/src/utils/model/deprecation.ts +104 -0
  1631. package/src/utils/model/model.ts +630 -0
  1632. package/src/utils/model/modelAllowlist.ts +170 -0
  1633. package/src/utils/model/modelCapabilities.ts +118 -0
  1634. package/src/utils/model/modelOptions.ts +651 -0
  1635. package/src/utils/model/modelStrings.ts +181 -0
  1636. package/src/utils/model/modelSupportOverrides.ts +50 -0
  1637. package/src/utils/model/providers.ts +50 -0
  1638. package/src/utils/model/validateModel.ts +173 -0
  1639. package/src/utils/modelCost.ts +231 -0
  1640. package/src/utils/modelSelection.ts +43 -0
  1641. package/src/utils/modifiers.ts +36 -0
  1642. package/src/utils/mtls.ts +179 -0
  1643. package/src/utils/nativeInstaller/download.ts +523 -0
  1644. package/src/utils/nativeInstaller/index.ts +18 -0
  1645. package/src/utils/nativeInstaller/installer.ts +1709 -0
  1646. package/src/utils/nativeInstaller/packageManagers.ts +336 -0
  1647. package/src/utils/nativeInstaller/pidLock.ts +433 -0
  1648. package/src/utils/notebook.ts +224 -0
  1649. package/src/utils/objectGroupBy.ts +18 -0
  1650. package/src/utils/pasteStore.ts +104 -0
  1651. package/src/utils/path.ts +155 -0
  1652. package/src/utils/pdf.ts +300 -0
  1653. package/src/utils/pdfUtils.ts +70 -0
  1654. package/src/utils/peerAddress.ts +21 -0
  1655. package/src/utils/permissions/PermissionMode.ts +166 -0
  1656. package/src/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
  1657. package/src/utils/permissions/PermissionResult.ts +35 -0
  1658. package/src/utils/permissions/PermissionRule.ts +40 -0
  1659. package/src/utils/permissions/PermissionUpdate.ts +389 -0
  1660. package/src/utils/permissions/PermissionUpdateSchema.ts +78 -0
  1661. package/src/utils/permissions/autoModeState.ts +39 -0
  1662. package/src/utils/permissions/bashClassifier.ts +61 -0
  1663. package/src/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
  1664. package/src/utils/permissions/classifierDecision.ts +98 -0
  1665. package/src/utils/permissions/classifierShared.ts +39 -0
  1666. package/src/utils/permissions/dangerousPatterns.ts +80 -0
  1667. package/src/utils/permissions/denialTracking.ts +45 -0
  1668. package/src/utils/permissions/filesystem.ts +1778 -0
  1669. package/src/utils/permissions/getNextPermissionMode.ts +101 -0
  1670. package/src/utils/permissions/pathValidation.ts +485 -0
  1671. package/src/utils/permissions/permissionExplainer.ts +250 -0
  1672. package/src/utils/permissions/permissionRuleParser.ts +198 -0
  1673. package/src/utils/permissions/permissionSetup.ts +1544 -0
  1674. package/src/utils/permissions/permissions.ts +1486 -0
  1675. package/src/utils/permissions/permissionsLoader.ts +296 -0
  1676. package/src/utils/permissions/shadowedRuleDetection.ts +234 -0
  1677. package/src/utils/permissions/shellRuleMatching.ts +228 -0
  1678. package/src/utils/permissions/yoloClassifier.ts +1495 -0
  1679. package/src/utils/planModeV2.ts +95 -0
  1680. package/src/utils/plans.ts +397 -0
  1681. package/src/utils/platform.ts +150 -0
  1682. package/src/utils/plugins/addDirPluginSettings.ts +71 -0
  1683. package/src/utils/plugins/cacheUtils.ts +196 -0
  1684. package/src/utils/plugins/dependencyResolver.ts +305 -0
  1685. package/src/utils/plugins/fetchTelemetry.ts +135 -0
  1686. package/src/utils/plugins/gitAvailability.ts +69 -0
  1687. package/src/utils/plugins/headlessPluginInstall.ts +174 -0
  1688. package/src/utils/plugins/hintRecommendation.ts +164 -0
  1689. package/src/utils/plugins/installCounts.ts +292 -0
  1690. package/src/utils/plugins/installedPluginsManager.ts +1268 -0
  1691. package/src/utils/plugins/loadPluginAgents.ts +348 -0
  1692. package/src/utils/plugins/loadPluginCommands.ts +946 -0
  1693. package/src/utils/plugins/loadPluginHooks.ts +287 -0
  1694. package/src/utils/plugins/loadPluginOutputStyles.ts +178 -0
  1695. package/src/utils/plugins/lspPluginIntegration.ts +387 -0
  1696. package/src/utils/plugins/lspRecommendation.ts +374 -0
  1697. package/src/utils/plugins/managedPlugins.ts +27 -0
  1698. package/src/utils/plugins/marketplaceHelpers.ts +592 -0
  1699. package/src/utils/plugins/marketplaceManager.ts +2643 -0
  1700. package/src/utils/plugins/mcpPluginIntegration.ts +634 -0
  1701. package/src/utils/plugins/mcpbHandler.ts +968 -0
  1702. package/src/utils/plugins/officialMarketplace.ts +25 -0
  1703. package/src/utils/plugins/officialMarketplaceGcs.ts +216 -0
  1704. package/src/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
  1705. package/src/utils/plugins/orphanedPluginFilter.ts +114 -0
  1706. package/src/utils/plugins/parseMarketplaceInput.ts +162 -0
  1707. package/src/utils/plugins/performStartupChecks.tsx +70 -0
  1708. package/src/utils/plugins/pluginAutoupdate.ts +284 -0
  1709. package/src/utils/plugins/pluginBlocklist.ts +127 -0
  1710. package/src/utils/plugins/pluginDirectories.ts +178 -0
  1711. package/src/utils/plugins/pluginFlagging.ts +208 -0
  1712. package/src/utils/plugins/pluginIdentifier.ts +123 -0
  1713. package/src/utils/plugins/pluginInstallationHelpers.ts +595 -0
  1714. package/src/utils/plugins/pluginLoader.ts +3302 -0
  1715. package/src/utils/plugins/pluginOptionsStorage.ts +400 -0
  1716. package/src/utils/plugins/pluginPolicy.ts +20 -0
  1717. package/src/utils/plugins/pluginStartupCheck.ts +341 -0
  1718. package/src/utils/plugins/pluginVersioning.ts +157 -0
  1719. package/src/utils/plugins/reconciler.ts +265 -0
  1720. package/src/utils/plugins/refresh.ts +215 -0
  1721. package/src/utils/plugins/schemas.ts +1681 -0
  1722. package/src/utils/plugins/validatePlugin.ts +903 -0
  1723. package/src/utils/plugins/walkPluginMarkdown.ts +69 -0
  1724. package/src/utils/plugins/zipCache.ts +406 -0
  1725. package/src/utils/plugins/zipCacheAdapters.ts +164 -0
  1726. package/src/utils/powershell/dangerousCmdlets.ts +185 -0
  1727. package/src/utils/powershell/parser.ts +1804 -0
  1728. package/src/utils/powershell/staticPrefix.ts +316 -0
  1729. package/src/utils/preflightChecks.tsx +151 -0
  1730. package/src/utils/privacyLevel.ts +55 -0
  1731. package/src/utils/process.ts +68 -0
  1732. package/src/utils/processUserInput/processBashCommand.tsx +140 -0
  1733. package/src/utils/processUserInput/processSlashCommand.tsx +922 -0
  1734. package/src/utils/processUserInput/processTextPrompt.ts +100 -0
  1735. package/src/utils/processUserInput/processUserInput.ts +605 -0
  1736. package/src/utils/profilerBase.ts +46 -0
  1737. package/src/utils/promptCategory.ts +49 -0
  1738. package/src/utils/promptEditor.ts +188 -0
  1739. package/src/utils/promptShellExecution.ts +183 -0
  1740. package/src/utils/providerModelCache.ts +30 -0
  1741. package/src/utils/providerSelection.ts +364 -0
  1742. package/src/utils/proxy.ts +426 -0
  1743. package/src/utils/queryContext.ts +179 -0
  1744. package/src/utils/queryHelpers.ts +552 -0
  1745. package/src/utils/queryProfiler.ts +301 -0
  1746. package/src/utils/queueProcessor.ts +95 -0
  1747. package/src/utils/readEditContext.ts +227 -0
  1748. package/src/utils/readFileInRange.ts +383 -0
  1749. package/src/utils/releaseNotes.ts +361 -0
  1750. package/src/utils/renderOptions.ts +77 -0
  1751. package/src/utils/ripgrep.ts +679 -0
  1752. package/src/utils/sandbox/sandbox-adapter.ts +985 -0
  1753. package/src/utils/sandbox/sandbox-ui-utils.ts +12 -0
  1754. package/src/utils/sanitization.ts +91 -0
  1755. package/src/utils/screenshotClipboard.ts +121 -0
  1756. package/src/utils/sdkEventQueue.ts +134 -0
  1757. package/src/utils/secureStorage/fallbackStorage.ts +70 -0
  1758. package/src/utils/secureStorage/index.ts +17 -0
  1759. package/src/utils/secureStorage/keychainPrefetch.ts +116 -0
  1760. package/src/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
  1761. package/src/utils/secureStorage/macOsKeychainStorage.ts +231 -0
  1762. package/src/utils/secureStorage/plainTextStorage.ts +84 -0
  1763. package/src/utils/semanticBoolean.ts +29 -0
  1764. package/src/utils/semanticNumber.ts +36 -0
  1765. package/src/utils/semver.ts +59 -0
  1766. package/src/utils/sequential.ts +56 -0
  1767. package/src/utils/sessionActivity.ts +133 -0
  1768. package/src/utils/sessionEnvVars.ts +22 -0
  1769. package/src/utils/sessionEnvironment.ts +166 -0
  1770. package/src/utils/sessionFileAccessHooks.ts +250 -0
  1771. package/src/utils/sessionIngressAuth.ts +140 -0
  1772. package/src/utils/sessionRestore.ts +551 -0
  1773. package/src/utils/sessionStart.ts +232 -0
  1774. package/src/utils/sessionState.ts +150 -0
  1775. package/src/utils/sessionStorage.ts +5106 -0
  1776. package/src/utils/sessionStoragePortable.ts +793 -0
  1777. package/src/utils/sessionTitle.ts +129 -0
  1778. package/src/utils/sessionUrl.ts +64 -0
  1779. package/src/utils/set.ts +53 -0
  1780. package/src/utils/settings/allErrors.ts +32 -0
  1781. package/src/utils/settings/applySettingsChange.ts +92 -0
  1782. package/src/utils/settings/changeDetector.ts +488 -0
  1783. package/src/utils/settings/constants.ts +202 -0
  1784. package/src/utils/settings/internalWrites.ts +37 -0
  1785. package/src/utils/settings/managedPath.ts +34 -0
  1786. package/src/utils/settings/mdm/constants.ts +81 -0
  1787. package/src/utils/settings/mdm/rawRead.ts +130 -0
  1788. package/src/utils/settings/mdm/settings.ts +316 -0
  1789. package/src/utils/settings/permissionValidation.ts +262 -0
  1790. package/src/utils/settings/pluginOnlyPolicy.ts +60 -0
  1791. package/src/utils/settings/schemaOutput.ts +8 -0
  1792. package/src/utils/settings/settings.ts +1015 -0
  1793. package/src/utils/settings/settingsCache.ts +80 -0
  1794. package/src/utils/settings/toolValidationConfig.ts +103 -0
  1795. package/src/utils/settings/types.ts +1171 -0
  1796. package/src/utils/settings/validateEditTool.ts +45 -0
  1797. package/src/utils/settings/validation.ts +265 -0
  1798. package/src/utils/settings/validationTips.ts +164 -0
  1799. package/src/utils/shell/bashProvider.ts +255 -0
  1800. package/src/utils/shell/outputLimits.ts +14 -0
  1801. package/src/utils/shell/powershellDetection.ts +107 -0
  1802. package/src/utils/shell/powershellProvider.ts +123 -0
  1803. package/src/utils/shell/prefix.ts +367 -0
  1804. package/src/utils/shell/readOnlyCommandValidation.ts +1893 -0
  1805. package/src/utils/shell/resolveDefaultShell.ts +14 -0
  1806. package/src/utils/shell/shellProvider.ts +33 -0
  1807. package/src/utils/shell/shellToolUtils.ts +22 -0
  1808. package/src/utils/shell/specPrefix.ts +241 -0
  1809. package/src/utils/shellConfig.ts +167 -0
  1810. package/src/utils/sideQuery.ts +223 -0
  1811. package/src/utils/sideQuestion.ts +155 -0
  1812. package/src/utils/signal.ts +43 -0
  1813. package/src/utils/sinks.ts +16 -0
  1814. package/src/utils/skills/skillChangeDetector.ts +311 -0
  1815. package/src/utils/slashCommandParsing.ts +60 -0
  1816. package/src/utils/sleep.ts +84 -0
  1817. package/src/utils/sliceAnsi.ts +91 -0
  1818. package/src/utils/slowOperations.ts +286 -0
  1819. package/src/utils/standaloneAgent.ts +23 -0
  1820. package/src/utils/startupProfiler.ts +194 -0
  1821. package/src/utils/staticRender.tsx +116 -0
  1822. package/src/utils/stats.ts +1061 -0
  1823. package/src/utils/statsCache.ts +434 -0
  1824. package/src/utils/status.tsx +372 -0
  1825. package/src/utils/statusNoticeDefinitions.tsx +198 -0
  1826. package/src/utils/statusNoticeHelpers.ts +20 -0
  1827. package/src/utils/stream.ts +76 -0
  1828. package/src/utils/streamJsonStdoutGuard.ts +123 -0
  1829. package/src/utils/streamlinedTransform.ts +201 -0
  1830. package/src/utils/stringUtils.ts +235 -0
  1831. package/src/utils/subprocessEnv.ts +99 -0
  1832. package/src/utils/suggestions/commandSuggestions.ts +567 -0
  1833. package/src/utils/suggestions/directoryCompletion.ts +263 -0
  1834. package/src/utils/suggestions/shellHistoryCompletion.ts +119 -0
  1835. package/src/utils/suggestions/skillUsageTracking.ts +55 -0
  1836. package/src/utils/suggestions/slackChannelSuggestions.ts +209 -0
  1837. package/src/utils/swarm/It2SetupPrompt.tsx +380 -0
  1838. package/src/utils/swarm/backends/ITermBackend.ts +370 -0
  1839. package/src/utils/swarm/backends/InProcessBackend.ts +339 -0
  1840. package/src/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
  1841. package/src/utils/swarm/backends/TmuxBackend.ts +764 -0
  1842. package/src/utils/swarm/backends/detection.ts +128 -0
  1843. package/src/utils/swarm/backends/it2Setup.ts +245 -0
  1844. package/src/utils/swarm/backends/registry.ts +464 -0
  1845. package/src/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
  1846. package/src/utils/swarm/backends/types.ts +311 -0
  1847. package/src/utils/swarm/constants.ts +33 -0
  1848. package/src/utils/swarm/inProcessRunner.ts +1552 -0
  1849. package/src/utils/swarm/leaderPermissionBridge.ts +54 -0
  1850. package/src/utils/swarm/permissionSync.ts +928 -0
  1851. package/src/utils/swarm/reconnection.ts +119 -0
  1852. package/src/utils/swarm/spawnInProcess.ts +328 -0
  1853. package/src/utils/swarm/spawnUtils.ts +168 -0
  1854. package/src/utils/swarm/teamHelpers.ts +683 -0
  1855. package/src/utils/swarm/teammateInit.ts +129 -0
  1856. package/src/utils/swarm/teammateLayoutManager.ts +107 -0
  1857. package/src/utils/swarm/teammateModel.ts +10 -0
  1858. package/src/utils/swarm/teammatePromptAddendum.ts +18 -0
  1859. package/src/utils/systemDirectories.ts +74 -0
  1860. package/src/utils/systemPrompt.ts +123 -0
  1861. package/src/utils/systemPromptType.ts +14 -0
  1862. package/src/utils/systemTheme.ts +119 -0
  1863. package/src/utils/taggedId.ts +54 -0
  1864. package/src/utils/task/TaskOutput.ts +390 -0
  1865. package/src/utils/task/diskOutput.ts +451 -0
  1866. package/src/utils/task/framework.ts +308 -0
  1867. package/src/utils/task/outputFormatting.ts +38 -0
  1868. package/src/utils/task/sdkProgress.ts +36 -0
  1869. package/src/utils/tasks.ts +862 -0
  1870. package/src/utils/teamDiscovery.ts +81 -0
  1871. package/src/utils/teamMemoryOps.ts +88 -0
  1872. package/src/utils/teammate.ts +292 -0
  1873. package/src/utils/teammateContext.ts +96 -0
  1874. package/src/utils/teammateMailbox.ts +1183 -0
  1875. package/src/utils/telemetry/betaSessionTracing.ts +491 -0
  1876. package/src/utils/telemetry/bigqueryExporter.ts +252 -0
  1877. package/src/utils/telemetry/events.ts +75 -0
  1878. package/src/utils/telemetry/instrumentation.ts +826 -0
  1879. package/src/utils/telemetry/logger.ts +26 -0
  1880. package/src/utils/telemetry/perfettoTracing.ts +1120 -0
  1881. package/src/utils/telemetry/pluginTelemetry.ts +289 -0
  1882. package/src/utils/telemetry/sessionTracing.ts +927 -0
  1883. package/src/utils/telemetry/skillLoadedEvent.ts +39 -0
  1884. package/src/utils/telemetryAttributes.ts +72 -0
  1885. package/src/utils/teleport/api.ts +466 -0
  1886. package/src/utils/teleport/environmentSelection.ts +77 -0
  1887. package/src/utils/teleport/environments.ts +120 -0
  1888. package/src/utils/teleport/gitBundle.ts +292 -0
  1889. package/src/utils/teleport.tsx +1226 -0
  1890. package/src/utils/tempfile.ts +31 -0
  1891. package/src/utils/terminal.ts +131 -0
  1892. package/src/utils/terminalPanel.ts +191 -0
  1893. package/src/utils/textHighlighting.ts +166 -0
  1894. package/src/utils/theme.ts +639 -0
  1895. package/src/utils/thinking.ts +170 -0
  1896. package/src/utils/timeouts.ts +39 -0
  1897. package/src/utils/tmuxSocket.ts +427 -0
  1898. package/src/utils/todo/types.ts +18 -0
  1899. package/src/utils/tokenBudget.ts +73 -0
  1900. package/src/utils/tokens.ts +261 -0
  1901. package/src/utils/toolErrors.ts +132 -0
  1902. package/src/utils/toolPool.ts +79 -0
  1903. package/src/utils/toolResultStorage.ts +1040 -0
  1904. package/src/utils/toolSchemaCache.ts +26 -0
  1905. package/src/utils/toolSearch.ts +756 -0
  1906. package/src/utils/transcriptSearch.ts +202 -0
  1907. package/src/utils/treeify.ts +170 -0
  1908. package/src/utils/truncate.ts +179 -0
  1909. package/src/utils/ultraplan/ccrSession.ts +349 -0
  1910. package/src/utils/ultraplan/keyword.ts +127 -0
  1911. package/src/utils/ultraplan/prompt.txt +12 -0
  1912. package/src/utils/unaryLogging.ts +39 -0
  1913. package/src/utils/undercover.ts +89 -0
  1914. package/src/utils/user.ts +195 -0
  1915. package/src/utils/userAgent.ts +11 -0
  1916. package/src/utils/userPromptKeywords.ts +27 -0
  1917. package/src/utils/uuid.ts +27 -0
  1918. package/src/utils/warningHandler.ts +121 -0
  1919. package/src/utils/webSearchEnabled.ts +16 -0
  1920. package/src/utils/which.ts +82 -0
  1921. package/src/utils/windowsPaths.ts +173 -0
  1922. package/src/utils/withResolvers.ts +13 -0
  1923. package/src/utils/words.ts +800 -0
  1924. package/src/utils/workloadContext.ts +57 -0
  1925. package/src/utils/worktree.ts +1519 -0
  1926. package/src/utils/worktreeModeEnabled.ts +11 -0
  1927. package/src/utils/xdg.ts +65 -0
  1928. package/src/utils/xml.ts +16 -0
  1929. package/src/utils/yaml.ts +15 -0
  1930. package/src/utils/zodToJsonSchema.ts +23 -0
  1931. package/src/vim/motions.ts +82 -0
  1932. package/src/vim/operators.ts +556 -0
  1933. package/src/vim/textObjects.ts +186 -0
  1934. package/src/vim/transitions.ts +490 -0
  1935. package/src/vim/types.ts +199 -0
  1936. package/src/voice/voiceModeEnabled.ts +54 -0
@@ -0,0 +1,2407 @@
1
+ import { getAppVersion } from '../utils/appVersion.js'
2
+ // biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered
3
+ import { randomUUID } from 'crypto'
4
+ import {
5
+ createBridgeApiClient,
6
+ BridgeFatalError,
7
+ isExpiredErrorType,
8
+ isSuppressible403,
9
+ } from './bridgeApi.js'
10
+ import type { BridgeConfig, BridgeApiClient } from './types.js'
11
+ import { logForDebugging } from '../utils/debug.js'
12
+ import { logForDiagnosticsNoPII } from '../utils/diagLogs.js'
13
+ import {
14
+ type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
15
+ logEvent,
16
+ } from '../services/analytics/index.js'
17
+ import { registerCleanup } from '../utils/cleanupRegistry.js'
18
+ import {
19
+ handleIngressMessage,
20
+ handleServerControlRequest,
21
+ makeResultMessage,
22
+ isEligibleBridgeMessage,
23
+ extractTitleText,
24
+ BoundedUUIDSet,
25
+ } from './bridgeMessaging.js'
26
+ import {
27
+ decodeWorkSecret,
28
+ buildSdkUrl,
29
+ buildCCRv2SdkUrl,
30
+ sameSessionId,
31
+ } from './workSecret.js'
32
+ import { toCompatSessionId, toInfraSessionId } from './sessionIdCompat.js'
33
+ import { updateSessionBridgeId } from '../utils/concurrentSessions.js'
34
+ import { getTrustedDeviceToken } from './trustedDevice.js'
35
+ import { HybridTransport } from '../cli/transports/HybridTransport.js'
36
+ import {
37
+ type ReplBridgeTransport,
38
+ createV1ReplTransport,
39
+ createV2ReplTransport,
40
+ } from './replBridgeTransport.js'
41
+ import { updateSessionIngressAuthToken } from '../utils/sessionIngressAuth.js'
42
+ import { isEnvTruthy, isInProtectedNamespace } from '../utils/envUtils.js'
43
+ import { validateBridgeId } from './bridgeApi.js'
44
+ import {
45
+ describeAxiosError,
46
+ extractHttpStatus,
47
+ logBridgeSkip,
48
+ } from './debugUtils.js'
49
+ import type { Message } from '../types/message.js'
50
+ import type { SDKMessage } from '../entrypoints/agentSdkTypes.js'
51
+ import type { PermissionMode } from '../utils/permissions/PermissionMode.js'
52
+ import type {
53
+ SDKControlRequest,
54
+ SDKControlResponse,
55
+ } from '../entrypoints/sdk/controlTypes.js'
56
+ import { createCapacityWake, type CapacitySignal } from './capacityWake.js'
57
+ import { FlushGate } from './flushGate.js'
58
+ import {
59
+ DEFAULT_POLL_CONFIG,
60
+ type PollIntervalConfig,
61
+ } from './pollConfigDefaults.js'
62
+ import { errorMessage } from '../utils/errors.js'
63
+ import { sleep } from '../utils/sleep.js'
64
+ import {
65
+ wrapApiForFaultInjection,
66
+ registerBridgeDebugHandle,
67
+ clearBridgeDebugHandle,
68
+ injectBridgeFault,
69
+ } from './bridgeDebug.js'
70
+
71
+ export type ReplBridgeHandle = {
72
+ bridgeSessionId: string
73
+ environmentId: string
74
+ sessionIngressUrl: string
75
+ writeMessages(messages: Message[]): void
76
+ writeSdkMessages(messages: SDKMessage[]): void
77
+ sendControlRequest(request: SDKControlRequest): void
78
+ sendControlResponse(response: SDKControlResponse): void
79
+ sendControlCancelRequest(requestId: string): void
80
+ sendResult(): void
81
+ teardown(): Promise<void>
82
+ }
83
+
84
+ export type BridgeState = 'ready' | 'connected' | 'reconnecting' | 'failed'
85
+
86
+ /**
87
+ * Explicit-param input to initBridgeCore. Everything initReplBridge reads
88
+ * from bootstrap state (cwd, session ID, git, OAuth) becomes a field here.
89
+ * A daemon caller (Agent SDK, PR 4) that never runs main.tsx fills these
90
+ * in itself.
91
+ */
92
+ export type BridgeCoreParams = {
93
+ dir: string
94
+ machineName: string
95
+ branch: string
96
+ gitRepoUrl: string | null
97
+ title: string
98
+ baseUrl: string
99
+ sessionIngressUrl: string
100
+ /**
101
+ * Opaque string sent as metadata.worker_type. Use BridgeWorkerType for
102
+ * the two CLI-originated values; daemon callers may send any string the
103
+ * backend recognizes (it's just a filter key on the web side).
104
+ */
105
+ workerType: string
106
+ getAccessToken: () => string | undefined
107
+ /**
108
+ * POST /v1/sessions. Injected because `createSession.ts` lazy-loads
109
+ * `auth.ts`/`model.ts`/`oauth/client.ts` and `bun --outfile` inlines
110
+ * dynamic imports — the lazy-load doesn't help, the whole REPL tree ends
111
+ * up in the Agent SDK bundle.
112
+ *
113
+ * REPL wrapper passes `createBridgeSession` from `createSession.ts`.
114
+ * Daemon wrapper passes `createBridgeSessionLean` from `sessionApi.ts`
115
+ * (HTTP-only, orgUUID+model supplied by the daemon caller).
116
+ *
117
+ * Receives `gitRepoUrl`+`branch` so the REPL wrapper can build the git
118
+ * source/outcome for claude.ai's session card. Daemon ignores them.
119
+ */
120
+ createSession: (opts: {
121
+ environmentId: string
122
+ title: string
123
+ gitRepoUrl: string | null
124
+ branch: string
125
+ signal: AbortSignal
126
+ }) => Promise<string | null>
127
+ /**
128
+ * POST /v1/sessions/{id}/archive. Same injection rationale. Best-effort;
129
+ * the callback MUST NOT throw.
130
+ */
131
+ archiveSession: (sessionId: string) => Promise<void>
132
+ /**
133
+ * Invoked on reconnect-after-env-lost to refresh the title. REPL wrapper
134
+ * reads session storage (picks up /rename); daemon returns the static
135
+ * title. Defaults to () => title.
136
+ */
137
+ getCurrentTitle?: () => string
138
+ /**
139
+ * Converts internal Message[] → SDKMessage[] for writeMessages() and the
140
+ * initial-flush/drain paths. REPL wrapper passes the real toSDKMessages
141
+ * from utils/messages/mappers.ts. Daemon callers that only use
142
+ * writeSdkMessages() and pass no initialMessages can omit this — those
143
+ * code paths are unreachable.
144
+ *
145
+ * Injected rather than imported because mappers.ts transitively pulls in
146
+ * src/commands.ts via messages.ts → api.ts → prompts.ts, dragging the
147
+ * entire command registry + React tree into the Agent SDK bundle.
148
+ */
149
+ toSDKMessages?: (messages: Message[]) => SDKMessage[]
150
+ /**
151
+ * OAuth 401 refresh handler passed to createBridgeApiClient. REPL wrapper
152
+ * passes handleOAuth401Error; daemon passes its AuthManager's handler.
153
+ * Injected because utils/auth.ts transitively pulls in the command
154
+ * registry via config.ts → file.ts → permissions/filesystem.ts →
155
+ * sessionStorage.ts → commands.ts.
156
+ */
157
+ onAuth401?: (staleAccessToken: string) => Promise<boolean>
158
+ /**
159
+ * Poll interval config getter for the work-poll heartbeat loop. REPL
160
+ * wrapper passes the GrowthBook-backed getPollIntervalConfig (allows ops
161
+ * to live-tune poll rates fleet-wide). Daemon passes a static config
162
+ * with a 60s heartbeat (5× headroom under the 300s work-lease TTL).
163
+ * Injected because growthbook.ts transitively pulls in the command
164
+ * registry via the same config.ts chain.
165
+ */
166
+ getPollIntervalConfig?: () => PollIntervalConfig
167
+ /**
168
+ * Max initial messages to replay on connect. REPL wrapper reads from the
169
+ * tengu_bridge_initial_history_cap GrowthBook flag. Daemon passes no
170
+ * initialMessages so this is never read. Default 200 matches the flag
171
+ * default.
172
+ */
173
+ initialHistoryCap?: number
174
+ // Same REPL-flush machinery as InitBridgeOptions — daemon omits these.
175
+ initialMessages?: Message[]
176
+ previouslyFlushedUUIDs?: Set<string>
177
+ onInboundMessage?: (msg: SDKMessage) => void
178
+ onPermissionResponse?: (response: SDKControlResponse) => void
179
+ onInterrupt?: () => void
180
+ onSetModel?: (model: string | undefined) => void
181
+ onSetMaxThinkingTokens?: (maxTokens: number | null) => void
182
+ /**
183
+ * Returns a policy verdict so this module can emit an error control_response
184
+ * without importing the policy checks itself (bootstrap-isolation constraint).
185
+ * The callback must guard `auto` (isAutoModeGateEnabled) and
186
+ * `bypassPermissions` (isBypassPermissionsModeDisabled AND
187
+ * isBypassPermissionsModeAvailable) BEFORE calling transitionPermissionMode —
188
+ * that function's internal auto-gate check is a defensive throw, not a
189
+ * graceful guard, and its side-effect order is setAutoModeActive(true) then
190
+ * throw, which corrupts the 3-way invariant documented in src/CLAUDE.md if
191
+ * the callback lets the throw escape here.
192
+ */
193
+ onSetPermissionMode?: (
194
+ mode: PermissionMode,
195
+ ) => { ok: true } | { ok: false; error: string }
196
+ onStateChange?: (state: BridgeState, detail?: string) => void
197
+ /**
198
+ * Fires on each real user message to flow through writeMessages() until
199
+ * the callback returns true (done). Mirrors remoteBridgeCore.ts's
200
+ * onUserMessage so the REPL bridge can derive a session title from early
201
+ * prompts when none was set at init time (e.g. user runs /remote-control
202
+ * on an empty conversation, then types). Tool-result wrappers, meta
203
+ * messages, and display-tag-only messages are skipped. Receives
204
+ * currentSessionId so the wrapper can PATCH the title without a closure
205
+ * dance to reach the not-yet-returned handle. The caller owns the
206
+ * derive-at-count-1-and-3 policy; the transport just keeps calling until
207
+ * told to stop. Not fired for the writeSdkMessages daemon path (daemon
208
+ * sets its own title at init). Distinct from SessionSpawnOpts's
209
+ * onFirstUserMessage (spawn-bridge, PR #21250), which stays fire-once.
210
+ */
211
+ onUserMessage?: (text: string, sessionId: string) => boolean
212
+ /** See InitBridgeOptions.perpetual. */
213
+ perpetual?: boolean
214
+ /**
215
+ * Seeds lastTransportSequenceNum — the SSE event-stream high-water mark
216
+ * that's carried across transport swaps within one process. Daemon callers
217
+ * pass the value they persisted at shutdown so the FIRST SSE connect of a
218
+ * fresh process sends from_sequence_num and the server doesn't replay full
219
+ * history. REPL callers omit (fresh session each run → 0 is correct).
220
+ */
221
+ initialSSESequenceNum?: number
222
+ }
223
+
224
+ /**
225
+ * Superset of ReplBridgeHandle. Adds getSSESequenceNum for daemon callers
226
+ * that persist the SSE seq-num across process restarts and pass it back as
227
+ * initialSSESequenceNum on the next start.
228
+ */
229
+ export type BridgeCoreHandle = ReplBridgeHandle & {
230
+ /**
231
+ * Current SSE sequence-number high-water mark. Updates as transports
232
+ * swap. Daemon callers persist this on shutdown and pass it back as
233
+ * initialSSESequenceNum on next start.
234
+ */
235
+ getSSESequenceNum(): number
236
+ }
237
+
238
+ /**
239
+ * Poll error recovery constants. When the work poll starts failing (e.g.
240
+ * server 500s), we use exponential backoff and give up after this timeout.
241
+ * This is deliberately long — the server is the authority on when a session
242
+ * is truly dead. As long as the server accepts our poll, we keep waiting
243
+ * for it to re-dispatch the work item.
244
+ */
245
+ const POLL_ERROR_INITIAL_DELAY_MS = 2_000
246
+ const POLL_ERROR_MAX_DELAY_MS = 60_000
247
+ const POLL_ERROR_GIVE_UP_MS = 15 * 60 * 1000
248
+
249
+ // Monotonically increasing counter for distinguishing init calls in logs
250
+ let initSequence = 0
251
+
252
+ /**
253
+ * Bootstrap-free core: env registration → session creation → poll loop →
254
+ * ingress WS → teardown. Reads nothing from bootstrap/state or
255
+ * sessionStorage — all context comes from params. Caller (initReplBridge
256
+ * below, or a daemon in PR 4) has already passed entitlement gates and
257
+ * gathered git/auth/title.
258
+ *
259
+ * Returns null on registration or session-creation failure.
260
+ */
261
+ export async function initBridgeCore(
262
+ params: BridgeCoreParams,
263
+ ): Promise<BridgeCoreHandle | null> {
264
+ const {
265
+ dir,
266
+ machineName,
267
+ branch,
268
+ gitRepoUrl,
269
+ title,
270
+ baseUrl,
271
+ sessionIngressUrl,
272
+ workerType,
273
+ getAccessToken,
274
+ createSession,
275
+ archiveSession,
276
+ getCurrentTitle = () => title,
277
+ toSDKMessages = () => {
278
+ throw new Error(
279
+ 'BridgeCoreParams.toSDKMessages not provided. Pass it if you use writeMessages() or initialMessages — daemon callers that only use writeSdkMessages() never hit this path.',
280
+ )
281
+ },
282
+ onAuth401,
283
+ getPollIntervalConfig = () => DEFAULT_POLL_CONFIG,
284
+ initialHistoryCap = 200,
285
+ initialMessages,
286
+ previouslyFlushedUUIDs,
287
+ onInboundMessage,
288
+ onPermissionResponse,
289
+ onInterrupt,
290
+ onSetModel,
291
+ onSetMaxThinkingTokens,
292
+ onSetPermissionMode,
293
+ onStateChange,
294
+ onUserMessage,
295
+ perpetual,
296
+ initialSSESequenceNum = 0,
297
+ } = params
298
+
299
+ const seq = ++initSequence
300
+
301
+ // bridgePointer import hoisted: perpetual mode reads it before register;
302
+ // non-perpetual writes it after session create; both use clear at teardown.
303
+ const { writeBridgePointer, clearBridgePointer, readBridgePointer } =
304
+ await import('./bridgePointer.js')
305
+
306
+ // Perpetual mode: read the crash-recovery pointer and treat it as prior
307
+ // state. The pointer is written unconditionally after session create
308
+ // (crash-recovery for all sessions); perpetual mode just skips the
309
+ // teardown clear so it survives clean exits too. Only reuse 'repl'
310
+ // pointers — a crashed standalone bridge (`claude remote-control`)
311
+ // writes source:'standalone' with a different workerType.
312
+ const rawPrior = perpetual ? await readBridgePointer(dir) : null
313
+ const prior = rawPrior?.source === 'repl' ? rawPrior : null
314
+
315
+ logForDebugging(
316
+ `[bridge:repl] initBridgeCore #${seq} starting (initialMessages=${initialMessages?.length ?? 0}${prior ? ` perpetual prior=env:${prior.environmentId}` : ''})`,
317
+ )
318
+
319
+ // 5. Register bridge environment
320
+ const rawApi = createBridgeApiClient({
321
+ baseUrl,
322
+ getAccessToken,
323
+ runnerVersion: getAppVersion(),
324
+ onDebug: logForDebugging,
325
+ onAuth401,
326
+ getTrustedDeviceToken,
327
+ })
328
+ // Ant-only: interpose so /bridge-kick can inject poll/register/heartbeat
329
+ // failures. Zero cost in external builds (rawApi passes through unchanged).
330
+ const api =
331
+ process.env.USER_TYPE === 'ant' ? wrapApiForFaultInjection(rawApi) : rawApi
332
+
333
+ const bridgeConfig: BridgeConfig = {
334
+ dir,
335
+ machineName,
336
+ branch,
337
+ gitRepoUrl,
338
+ maxSessions: 1,
339
+ spawnMode: 'single-session',
340
+ verbose: false,
341
+ sandbox: false,
342
+ bridgeId: randomUUID(),
343
+ workerType,
344
+ environmentId: randomUUID(),
345
+ reuseEnvironmentId: prior?.environmentId,
346
+ apiBaseUrl: baseUrl,
347
+ sessionIngressUrl,
348
+ }
349
+
350
+ let environmentId: string
351
+ let environmentSecret: string
352
+ try {
353
+ const reg = await api.registerBridgeEnvironment(bridgeConfig)
354
+ environmentId = reg.environment_id
355
+ environmentSecret = reg.environment_secret
356
+ } catch (err) {
357
+ logBridgeSkip(
358
+ 'registration_failed',
359
+ `[bridge:repl] Environment registration failed: ${errorMessage(err)}`,
360
+ )
361
+ // Stale pointer may be the cause (expired/deleted env) — clear it so
362
+ // the next start doesn't retry the same dead ID.
363
+ if (prior) {
364
+ await clearBridgePointer(dir)
365
+ }
366
+ onStateChange?.('failed', errorMessage(err))
367
+ return null
368
+ }
369
+
370
+ logForDebugging(`[bridge:repl] Environment registered: ${environmentId}`)
371
+ logForDiagnosticsNoPII('info', 'bridge_repl_env_registered')
372
+ logEvent('tengu_bridge_repl_env_registered', {})
373
+
374
+ /**
375
+ * Reconnect-in-place: if the just-registered environmentId matches what
376
+ * was requested, call reconnectSession to force-stop stale workers and
377
+ * re-queue the session. Used at init (perpetual mode — env is alive but
378
+ * idle after clean teardown) and in doReconnect() Strategy 1 (env lost
379
+ * then resurrected). Returns true on success; caller falls back to
380
+ * fresh session creation on false.
381
+ */
382
+ async function tryReconnectInPlace(
383
+ requestedEnvId: string,
384
+ sessionId: string,
385
+ ): Promise<boolean> {
386
+ if (environmentId !== requestedEnvId) {
387
+ logForDebugging(
388
+ `[bridge:repl] Env mismatch (requested ${requestedEnvId}, got ${environmentId}) — cannot reconnect in place`,
389
+ )
390
+ return false
391
+ }
392
+ // The pointer stores what createBridgeSession returned (session_*,
393
+ // compat/convert.go:41). /bridge/reconnect is an environments-layer
394
+ // endpoint — once the server's ccr_v2_compat_enabled gate is on it
395
+ // looks sessions up by their infra tag (cse_*) and returns "Session
396
+ // not found" for the session_* costume. We don't know the gate state
397
+ // pre-poll, so try both; the re-tag is a no-op if the ID is already
398
+ // cse_* (doReconnect Strategy 1 path — currentSessionId never mutates
399
+ // to cse_* but future-proof the check).
400
+ const infraId = toInfraSessionId(sessionId)
401
+ const candidates =
402
+ infraId === sessionId ? [sessionId] : [sessionId, infraId]
403
+ for (const id of candidates) {
404
+ try {
405
+ await api.reconnectSession(environmentId, id)
406
+ logForDebugging(
407
+ `[bridge:repl] Reconnected session ${id} in place on env ${environmentId}`,
408
+ )
409
+ return true
410
+ } catch (err) {
411
+ logForDebugging(
412
+ `[bridge:repl] reconnectSession(${id}) failed: ${errorMessage(err)}`,
413
+ )
414
+ }
415
+ }
416
+ logForDebugging(
417
+ '[bridge:repl] reconnectSession exhausted — falling through to fresh session',
418
+ )
419
+ return false
420
+ }
421
+
422
+ // Perpetual init: env is alive but has no queued work after clean
423
+ // teardown. reconnectSession re-queues it. doReconnect() has the same
424
+ // call but only fires on poll 404 (env dead);
425
+ // here the env is alive but idle.
426
+ const reusedPriorSession = prior
427
+ ? await tryReconnectInPlace(prior.environmentId, prior.sessionId)
428
+ : false
429
+ if (prior && !reusedPriorSession) {
430
+ await clearBridgePointer(dir)
431
+ }
432
+
433
+ // 6. Create session on the bridge. Initial messages are NOT included as
434
+ // session creation events because those use STREAM_ONLY persistence and
435
+ // are published before the CCR UI subscribes, so they get lost. Instead,
436
+ // initial messages are flushed via the ingress WebSocket once it connects.
437
+
438
+ // Mutable session ID — updated when the environment+session pair is
439
+ // re-created after a connection loss.
440
+ let currentSessionId: string
441
+
442
+
443
+ if (reusedPriorSession && prior) {
444
+ currentSessionId = prior.sessionId
445
+ logForDebugging(
446
+ `[bridge:repl] Perpetual session reused: ${currentSessionId}`,
447
+ )
448
+ // Server already has all initialMessages from the prior CLI run. Mark
449
+ // them as previously-flushed so the initial flush filter excludes them
450
+ // (previouslyFlushedUUIDs is a fresh Set on every CLI start). Duplicate
451
+ // UUIDs cause the server to kill the WebSocket.
452
+ if (initialMessages && previouslyFlushedUUIDs) {
453
+ for (const msg of initialMessages) {
454
+ previouslyFlushedUUIDs.add(msg.uuid)
455
+ }
456
+ }
457
+ } else {
458
+ const createdSessionId = await createSession({
459
+ environmentId,
460
+ title,
461
+ gitRepoUrl,
462
+ branch,
463
+ signal: AbortSignal.timeout(15_000),
464
+ })
465
+
466
+ if (!createdSessionId) {
467
+ logForDebugging(
468
+ '[bridge:repl] Session creation failed, deregistering environment',
469
+ )
470
+ logEvent('tengu_bridge_repl_session_failed', {})
471
+ await api.deregisterEnvironment(environmentId).catch(() => {})
472
+ onStateChange?.('failed', 'Session creation failed')
473
+ return null
474
+ }
475
+
476
+ currentSessionId = createdSessionId
477
+ logForDebugging(`[bridge:repl] Session created: ${currentSessionId}`)
478
+ }
479
+
480
+ // Crash-recovery pointer: written now so a kill -9 at any point after
481
+ // this leaves a recoverable trail. Cleared in teardown (non-perpetual)
482
+ // or left alone (perpetual mode — pointer survives clean exit too).
483
+ // `claude remote-control --continue` from the same directory will detect
484
+ // it and offer to resume.
485
+ await writeBridgePointer(dir, {
486
+ sessionId: currentSessionId,
487
+ environmentId,
488
+ source: 'repl',
489
+ })
490
+ logForDiagnosticsNoPII('info', 'bridge_repl_session_created')
491
+ logEvent('tengu_bridge_repl_started', {
492
+ has_initial_messages: !!(initialMessages && initialMessages.length > 0),
493
+ inProtectedNamespace: isInProtectedNamespace(),
494
+ })
495
+
496
+ // UUIDs of initial messages. Used for dedup in writeMessages to avoid
497
+ // re-sending messages that were already flushed on WebSocket open.
498
+ const initialMessageUUIDs = new Set<string>()
499
+ if (initialMessages) {
500
+ for (const msg of initialMessages) {
501
+ initialMessageUUIDs.add(msg.uuid)
502
+ }
503
+ }
504
+
505
+ // Bounded ring buffer of UUIDs for messages we've already sent to the
506
+ // server via the ingress WebSocket. Serves two purposes:
507
+ // 1. Echo filtering — ignore our own messages bouncing back on the WS.
508
+ // 2. Secondary dedup in writeMessages — catch race conditions where
509
+ // the hook's index-based tracking isn't sufficient.
510
+ //
511
+ // Seeded with initialMessageUUIDs so that when the server echoes back
512
+ // the initial conversation context over the ingress WebSocket, those
513
+ // messages are recognized as echoes and not re-injected into the REPL.
514
+ //
515
+ // Capacity of 2000 covers well over any realistic echo window (echoes
516
+ // arrive within milliseconds) and any messages that might be re-encountered
517
+ // after compaction. The hook's lastWrittenIndexRef is the primary dedup;
518
+ // this is a safety net.
519
+ const recentPostedUUIDs = new BoundedUUIDSet(2000)
520
+ for (const uuid of initialMessageUUIDs) {
521
+ recentPostedUUIDs.add(uuid)
522
+ }
523
+
524
+ // Bounded set of INBOUND prompt UUIDs we've already forwarded to the REPL.
525
+ // Defensive dedup for when the server re-delivers prompts (seq-num
526
+ // negotiation failure, server edge cases, transport swap races). The
527
+ // seq-num carryover below is the primary fix; this is the safety net.
528
+ const recentInboundUUIDs = new BoundedUUIDSet(2000)
529
+
530
+ // 7. Start poll loop for work items — this is what makes the session
531
+ // "live" on claude.ai. When a user types there, the backend dispatches
532
+ // a work item to our environment. We poll for it, get the ingress token,
533
+ // and connect the ingress WebSocket.
534
+ //
535
+ // The poll loop keeps running: when work arrives it connects the ingress
536
+ // WebSocket, and if the WebSocket drops unexpectedly (code != 1000) it
537
+ // resumes polling to get a fresh ingress token and reconnect.
538
+ const pollController = new AbortController()
539
+ // Adapter over either HybridTransport (v1: WS reads + POST writes to
540
+ // Session-Ingress) or SSETransport+CCRClient (v2: SSE reads + POST
541
+ // writes to CCR /worker/*). The v1/v2 choice is made in onWorkReceived:
542
+ // server-driven via secret.use_code_sessions, with CLAUDE_BRIDGE_USE_CCR_V2
543
+ // as an ant-dev override.
544
+ let transport: ReplBridgeTransport | null = null
545
+ // Bumped on every onWorkReceived. Captured in createV2ReplTransport's .then()
546
+ // closure to detect stale resolutions: if two calls race while transport is
547
+ // null, both registerWorker() (bumping server epoch), and whichever resolves
548
+ // SECOND is the correct one — but the transport !== null check gets this
549
+ // backwards (first-to-resolve installs, second discards). The generation
550
+ // counter catches it independent of transport state.
551
+ let v2Generation = 0
552
+ // SSE sequence-number high-water mark carried across transport swaps.
553
+ // Without this, each new SSETransport starts at 0, sends no
554
+ // from_sequence_num / Last-Event-ID on its first connect, and the server
555
+ // replays the entire session event history — every prompt ever sent
556
+ // re-delivered as fresh inbound messages on every onWorkReceived.
557
+ //
558
+ // Seed only when we actually reconnected the prior session. If
559
+ // `reusedPriorSession` is false we fell through to `createSession()` —
560
+ // the caller's persisted seq-num belongs to a dead session and applying
561
+ // it to the fresh stream (starting at 1) silently drops events. Same
562
+ // hazard as doReconnect Strategy 2; same fix as the reset there.
563
+ let lastTransportSequenceNum = reusedPriorSession ? initialSSESequenceNum : 0
564
+ // Track the current work ID so teardown can call stopWork
565
+ let currentWorkId: string | null = null
566
+ // Session ingress JWT for the current work item — used for heartbeat auth.
567
+ let currentIngressToken: string | null = null
568
+ // Signal to wake the at-capacity sleep early when the transport is lost,
569
+ // so the poll loop immediately switches back to fast polling for new work.
570
+ const capacityWake = createCapacityWake(pollController.signal)
571
+ const wakePollLoop = capacityWake.wake
572
+ const capacitySignal = capacityWake.signal
573
+ // Gates message writes during the initial flush to prevent ordering
574
+ // races where new messages arrive at the server interleaved with history.
575
+ const flushGate = new FlushGate<Message>()
576
+
577
+ // Latch for onUserMessage — flips true when the callback returns true
578
+ // (policy says "done deriving"). If no callback, skip scanning entirely
579
+ // (daemon path — no title derivation needed).
580
+ let userMessageCallbackDone = !onUserMessage
581
+
582
+ // Shared counter for environment re-creations, used by both
583
+ // onEnvironmentLost and the abnormal-close handler.
584
+ const MAX_ENVIRONMENT_RECREATIONS = 3
585
+ let environmentRecreations = 0
586
+ let reconnectPromise: Promise<boolean> | null = null
587
+
588
+ /**
589
+ * Recover from onEnvironmentLost (poll returned 404 — env was reaped
590
+ * server-side). Tries two strategies in order:
591
+ *
592
+ * 1. Reconnect-in-place: idempotent re-register with reuseEnvironmentId
593
+ * → if the backend returns the same env ID, call reconnectSession()
594
+ * to re-queue the existing session. currentSessionId stays the same;
595
+ * the URL on the user's phone stays valid; previouslyFlushedUUIDs is
596
+ * preserved so history isn't re-sent.
597
+ *
598
+ * 2. Fresh session fallback: if the backend returns a different env ID
599
+ * (original TTL-expired, e.g. laptop slept >4h) or reconnectSession()
600
+ * throws, archive the old session and create a new one on the
601
+ * now-registered env. Old behavior before #20460 primitives landed.
602
+ *
603
+ * Uses a promise-based reentrancy guard so concurrent callers share the
604
+ * same reconnection attempt.
605
+ */
606
+ async function reconnectEnvironmentWithSession(): Promise<boolean> {
607
+ if (reconnectPromise) {
608
+ return reconnectPromise
609
+ }
610
+ reconnectPromise = doReconnect()
611
+ try {
612
+ return await reconnectPromise
613
+ } finally {
614
+ reconnectPromise = null
615
+ }
616
+ }
617
+
618
+ async function doReconnect(): Promise<boolean> {
619
+ environmentRecreations++
620
+ // Invalidate any in-flight v2 handshake — the environment is being
621
+ // recreated, so a stale transport arriving post-reconnect would be
622
+ // pointed at a dead session.
623
+ v2Generation++
624
+ logForDebugging(
625
+ `[bridge:repl] Reconnecting after env lost (attempt ${environmentRecreations}/${MAX_ENVIRONMENT_RECREATIONS})`,
626
+ )
627
+
628
+ if (environmentRecreations > MAX_ENVIRONMENT_RECREATIONS) {
629
+ logForDebugging(
630
+ `[bridge:repl] Environment reconnect limit reached (${MAX_ENVIRONMENT_RECREATIONS}), giving up`,
631
+ )
632
+ return false
633
+ }
634
+
635
+ // Close the stale transport. Capture seq BEFORE close — if Strategy 1
636
+ // (tryReconnectInPlace) succeeds we keep the SAME session, and the
637
+ // next transport must resume where this one left off, not replay from
638
+ // the last transport-swap checkpoint.
639
+ if (transport) {
640
+ const seq = transport.getLastSequenceNum()
641
+ if (seq > lastTransportSequenceNum) {
642
+ lastTransportSequenceNum = seq
643
+ }
644
+ transport.close()
645
+ transport = null
646
+ }
647
+ // Transport is gone — wake the poll loop out of its at-capacity
648
+ // heartbeat sleep so it can fast-poll for re-dispatched work.
649
+ wakePollLoop()
650
+ // Reset flush gate so writeMessages() hits the !transport guard
651
+ // instead of silently queuing into a dead buffer.
652
+ flushGate.drop()
653
+
654
+ // Release the current work item (force=false — we may want the session
655
+ // back). Best-effort: the env is probably gone, so this likely 404s.
656
+ if (currentWorkId) {
657
+ const workIdBeingCleared = currentWorkId
658
+ await api
659
+ .stopWork(environmentId, workIdBeingCleared, false)
660
+ .catch(() => {})
661
+ // When doReconnect runs concurrently with the poll loop (ws_closed
662
+ // handler case — void-called, unlike the awaited onEnvironmentLost
663
+ // path), onWorkReceived can fire during the stopWork await and set
664
+ // a fresh currentWorkId. If it did, the poll loop has already
665
+ // recovered on its own — defer to it rather than proceeding to
666
+ // archiveSession, which would destroy the session its new
667
+ // transport is connected to.
668
+ if (currentWorkId !== workIdBeingCleared) {
669
+ logForDebugging(
670
+ '[bridge:repl] Poll loop recovered during stopWork await — deferring to it',
671
+ )
672
+ environmentRecreations = 0
673
+ return true
674
+ }
675
+ currentWorkId = null
676
+ currentIngressToken = null
677
+ }
678
+
679
+ // Bail out if teardown started while we were awaiting
680
+ if (pollController.signal.aborted) {
681
+ logForDebugging('[bridge:repl] Reconnect aborted by teardown')
682
+ return false
683
+ }
684
+
685
+ // Strategy 1: idempotent re-register with the server-issued env ID.
686
+ // If the backend resurrects the same env (fresh secret), we can
687
+ // reconnect the existing session. If it hands back a different ID, the
688
+ // original env is truly gone and we fall through to a fresh session.
689
+ const requestedEnvId = environmentId
690
+ bridgeConfig.reuseEnvironmentId = requestedEnvId
691
+ try {
692
+ const reg = await api.registerBridgeEnvironment(bridgeConfig)
693
+ environmentId = reg.environment_id
694
+ environmentSecret = reg.environment_secret
695
+ } catch (err) {
696
+ bridgeConfig.reuseEnvironmentId = undefined
697
+ logForDebugging(
698
+ `[bridge:repl] Environment re-registration failed: ${errorMessage(err)}`,
699
+ )
700
+ return false
701
+ }
702
+ // Clear before any await — a stale value would poison the next fresh
703
+ // registration if doReconnect runs again.
704
+ bridgeConfig.reuseEnvironmentId = undefined
705
+
706
+ logForDebugging(
707
+ `[bridge:repl] Re-registered: requested=${requestedEnvId} got=${environmentId}`,
708
+ )
709
+
710
+ // Bail out if teardown started while we were registering
711
+ if (pollController.signal.aborted) {
712
+ logForDebugging(
713
+ '[bridge:repl] Reconnect aborted after env registration, cleaning up',
714
+ )
715
+ await api.deregisterEnvironment(environmentId).catch(() => {})
716
+ return false
717
+ }
718
+
719
+ // Same race as above, narrower window: poll loop may have set up a
720
+ // transport during the registerBridgeEnvironment await. Bail before
721
+ // tryReconnectInPlace/archiveSession kill it server-side.
722
+ if (transport !== null) {
723
+ logForDebugging(
724
+ '[bridge:repl] Poll loop recovered during registerBridgeEnvironment await — deferring to it',
725
+ )
726
+ environmentRecreations = 0
727
+ return true
728
+ }
729
+
730
+ // Strategy 1: same helper as perpetual init. currentSessionId stays
731
+ // the same on success; URL on mobile/web stays valid;
732
+ // previouslyFlushedUUIDs preserved (no re-flush).
733
+ if (await tryReconnectInPlace(requestedEnvId, currentSessionId)) {
734
+ logEvent('tengu_bridge_repl_reconnected_in_place', {})
735
+ environmentRecreations = 0
736
+ return true
737
+ }
738
+ // Env differs → TTL-expired/reaped; or reconnect failed.
739
+ // Don't deregister — we have a fresh secret for this env either way.
740
+ if (environmentId !== requestedEnvId) {
741
+ logEvent('tengu_bridge_repl_env_expired_fresh_session', {})
742
+ }
743
+
744
+ // Strategy 2: fresh session on the now-registered environment.
745
+ // Archive the old session first — it's orphaned (bound to a dead env,
746
+ // or reconnectSession rejected it). Don't deregister the env — we just
747
+ // got a fresh secret for it and are about to use it.
748
+ await archiveSession(currentSessionId)
749
+
750
+ // Bail out if teardown started while we were archiving
751
+ if (pollController.signal.aborted) {
752
+ logForDebugging(
753
+ '[bridge:repl] Reconnect aborted after archive, cleaning up',
754
+ )
755
+ await api.deregisterEnvironment(environmentId).catch(() => {})
756
+ return false
757
+ }
758
+
759
+ // Re-read the current title in case the user renamed the session.
760
+ // REPL wrapper reads session storage; daemon wrapper returns the
761
+ // original title (nothing to refresh).
762
+ const currentTitle = getCurrentTitle()
763
+
764
+ // Create a new session on the now-registered environment
765
+ const newSessionId = await createSession({
766
+ environmentId,
767
+ title: currentTitle,
768
+ gitRepoUrl,
769
+ branch,
770
+ signal: AbortSignal.timeout(15_000),
771
+ })
772
+
773
+ if (!newSessionId) {
774
+ logForDebugging(
775
+ '[bridge:repl] Session creation failed during reconnection',
776
+ )
777
+ return false
778
+ }
779
+
780
+ // Bail out if teardown started during session creation (up to 15s)
781
+ if (pollController.signal.aborted) {
782
+ logForDebugging(
783
+ '[bridge:repl] Reconnect aborted after session creation, cleaning up',
784
+ )
785
+ await archiveSession(newSessionId)
786
+ return false
787
+ }
788
+
789
+ currentSessionId = newSessionId
790
+ // Re-publish to the PID file so peer dedup (peerRegistry.ts) picks up the
791
+ // new ID — setReplBridgeHandle only fires at init/teardown, not reconnect.
792
+ void updateSessionBridgeId(toCompatSessionId(newSessionId)).catch(() => {})
793
+ // Reset per-session transport state IMMEDIATELY after the session swap,
794
+ // before any await. If this runs after `await writeBridgePointer` below,
795
+ // there's a window where handle.bridgeSessionId already returns session B
796
+ // but getSSESequenceNum() still returns session A's seq — a daemon
797
+ // persistState() in that window writes {bridgeSessionId: B, seq: OLD_A},
798
+ // which PASSES the session-ID validation check and defeats it entirely.
799
+ //
800
+ // The SSE seq-num is scoped to the session's event stream — carrying it
801
+ // over leaves the transport's lastSequenceNum stuck high (seq only
802
+ // advances when received > last), and its next internal reconnect would
803
+ // send from_sequence_num=OLD_SEQ against a stream starting at 1 → all
804
+ // events in the gap silently dropped. Inbound UUID dedup is also
805
+ // session-scoped.
806
+ lastTransportSequenceNum = 0
807
+ recentInboundUUIDs.clear()
808
+ // Title derivation is session-scoped too: if the user typed during the
809
+ // createSession await above, the callback fired against the OLD archived
810
+ // session ID (PATCH lost) and the new session got `currentTitle` captured
811
+ // BEFORE they typed. Reset so the next prompt can re-derive. Self-
812
+ // correcting: if the caller's policy is already done (explicit title or
813
+ // count ≥ 3), it returns true on the first post-reset call and re-latches.
814
+ userMessageCallbackDone = !onUserMessage
815
+ logForDebugging(`[bridge:repl] Re-created session: ${currentSessionId}`)
816
+
817
+ // Rewrite the crash-recovery pointer with the new IDs so a crash after
818
+ // this point resumes the right session. (The reconnect-in-place path
819
+ // above doesn't touch the pointer — same session, same env.)
820
+ await writeBridgePointer(dir, {
821
+ sessionId: currentSessionId,
822
+ environmentId,
823
+ source: 'repl',
824
+ })
825
+
826
+ // Clear flushed UUIDs so initial messages are re-sent to the new session.
827
+ // UUIDs are scoped per-session on the server, so re-flushing is safe.
828
+ previouslyFlushedUUIDs?.clear()
829
+
830
+
831
+ // Reset the counter so independent reconnections hours apart don't
832
+ // exhaust the limit — it guards against rapid consecutive failures,
833
+ // not lifetime total.
834
+ environmentRecreations = 0
835
+
836
+ return true
837
+ }
838
+
839
+ // Helper: get the current OAuth access token for session ingress auth.
840
+ // Unlike the JWT path, OAuth tokens are refreshed by the standard OAuth
841
+ // flow — no proactive scheduler needed.
842
+ function getOAuthToken(): string | undefined {
843
+ return getAccessToken()
844
+ }
845
+
846
+ // Drain any messages that were queued during the initial flush.
847
+ // Called after writeBatch completes (or fails) so queued messages
848
+ // are sent in order after the historical messages.
849
+ function drainFlushGate(): void {
850
+ const msgs = flushGate.end()
851
+ if (msgs.length === 0) return
852
+ if (!transport) {
853
+ logForDebugging(
854
+ `[bridge:repl] Cannot drain ${msgs.length} pending message(s): no transport`,
855
+ )
856
+ return
857
+ }
858
+ for (const msg of msgs) {
859
+ recentPostedUUIDs.add(msg.uuid)
860
+ }
861
+ const sdkMessages = toSDKMessages(msgs)
862
+ const events = sdkMessages.map(sdkMsg => ({
863
+ ...sdkMsg,
864
+ session_id: currentSessionId,
865
+ }))
866
+ logForDebugging(
867
+ `[bridge:repl] Drained ${msgs.length} pending message(s) after flush`,
868
+ )
869
+ void transport.writeBatch(events)
870
+ }
871
+
872
+ // Teardown reference — set after definition below. All callers are async
873
+ // callbacks that run after assignment, so the reference is always valid.
874
+ let doTeardownImpl: (() => Promise<void>) | null = null
875
+ function triggerTeardown(): void {
876
+ void doTeardownImpl?.()
877
+ }
878
+
879
+ /**
880
+ * Body of the transport's setOnClose callback, hoisted to initBridgeCore
881
+ * scope so /bridge-kick can fire it directly. setOnClose wraps this with
882
+ * a stale-transport guard; debugFireClose calls it bare.
883
+ *
884
+ * With autoReconnect:true, this only fires on: clean close (1000),
885
+ * permanent server rejection (4001/1002/4003), or 10-min budget
886
+ * exhaustion. Transient drops are retried internally by the transport.
887
+ */
888
+ function handleTransportPermanentClose(closeCode: number | undefined): void {
889
+ logForDebugging(
890
+ `[bridge:repl] Transport permanently closed: code=${closeCode}`,
891
+ )
892
+ logEvent('tengu_bridge_repl_ws_closed', {
893
+ code: closeCode,
894
+ })
895
+ // Capture SSE seq high-water mark before nulling. When called from
896
+ // setOnClose the guard guarantees transport !== null; when fired from
897
+ // /bridge-kick it may already be null (e.g. fired twice) — skip.
898
+ if (transport) {
899
+ const closedSeq = transport.getLastSequenceNum()
900
+ if (closedSeq > lastTransportSequenceNum) {
901
+ lastTransportSequenceNum = closedSeq
902
+ }
903
+ transport = null
904
+ }
905
+ // Transport is gone — wake the poll loop out of its at-capacity
906
+ // heartbeat sleep so it's fast-polling by the time the reconnect
907
+ // below completes and the server re-queues work.
908
+ wakePollLoop()
909
+ // Reset flush state so writeMessages() hits the !transport guard
910
+ // (with a warning log) instead of silently queuing into a buffer
911
+ // that will never be drained. Unlike onWorkReceived (which
912
+ // preserves pending messages for the new transport), onClose is
913
+ // a permanent close — no new transport will drain these.
914
+ const dropped = flushGate.drop()
915
+ if (dropped > 0) {
916
+ logForDebugging(
917
+ `[bridge:repl] Dropping ${dropped} pending message(s) on transport close (code=${closeCode})`,
918
+ { level: 'warn' },
919
+ )
920
+ }
921
+
922
+ if (closeCode === 1000) {
923
+ // Clean close — session ended normally. Tear down the bridge.
924
+ onStateChange?.('failed', 'session ended')
925
+ pollController.abort()
926
+ triggerTeardown()
927
+ return
928
+ }
929
+
930
+ // Transport reconnect budget exhausted or permanent server
931
+ // rejection. By this point the env has usually been reaped
932
+ // server-side (BQ 2026-03-12: ~98% of ws_closed never recover
933
+ // via poll alone). stopWork(force=false) can't re-dispatch work
934
+ // from an archived env; reconnectEnvironmentWithSession can
935
+ // re-activate it via POST /bridge/reconnect, or fall through
936
+ // to a fresh session if the env is truly gone. The poll loop
937
+ // (already woken above) picks up the re-queued work once
938
+ // doReconnect completes.
939
+ onStateChange?.(
940
+ 'reconnecting',
941
+ `Remote Control connection lost (code ${closeCode})`,
942
+ )
943
+ logForDebugging(
944
+ `[bridge:repl] Transport reconnect budget exhausted (code=${closeCode}), attempting env reconnect`,
945
+ )
946
+ void reconnectEnvironmentWithSession().then(success => {
947
+ if (success) return
948
+ // doReconnect has four abort-check return-false sites for
949
+ // teardown-in-progress. Don't pollute the BQ failure signal
950
+ // or double-teardown when the user just quit.
951
+ if (pollController.signal.aborted) return
952
+ // doReconnect returns false (never throws) on genuine failure.
953
+ // The dangerous case: registerBridgeEnvironment succeeded (so
954
+ // environmentId now points at a fresh valid env) but
955
+ // createSession failed — poll loop would poll a sessionless
956
+ // env getting null work with no errors, never hitting any
957
+ // give-up path. Tear down explicitly.
958
+ logForDebugging(
959
+ '[bridge:repl] reconnectEnvironmentWithSession resolved false — tearing down',
960
+ )
961
+ logEvent('tengu_bridge_repl_reconnect_failed', {
962
+ close_code: closeCode,
963
+ })
964
+ onStateChange?.('failed', 'reconnection failed')
965
+ triggerTeardown()
966
+ })
967
+ }
968
+
969
+ // Ant-only: SIGUSR2 → force doReconnect() for manual testing. Skips the
970
+ // ~30s poll wait — fire-and-observe in the debug log immediately.
971
+ // Windows has no USR signals; `process.on` would throw there.
972
+ let sigusr2Handler: (() => void) | undefined
973
+ if (process.env.USER_TYPE === 'ant' && process.platform !== 'win32') {
974
+ sigusr2Handler = () => {
975
+ logForDebugging(
976
+ '[bridge:repl] SIGUSR2 received — forcing doReconnect() for testing',
977
+ )
978
+ void reconnectEnvironmentWithSession()
979
+ }
980
+ process.on('SIGUSR2', sigusr2Handler)
981
+ }
982
+
983
+ // Ant-only: /bridge-kick fault injection. handleTransportPermanentClose
984
+ // is defined below and assigned into this slot so the slash command can
985
+ // invoke it directly — the real setOnClose callback is buried inside
986
+ // wireTransport which is itself inside onWorkReceived.
987
+ let debugFireClose: ((code: number) => void) | null = null
988
+ if (process.env.USER_TYPE === 'ant') {
989
+ registerBridgeDebugHandle({
990
+ fireClose: code => {
991
+ if (!debugFireClose) {
992
+ logForDebugging('[bridge:debug] fireClose: no transport wired yet')
993
+ return
994
+ }
995
+ logForDebugging(`[bridge:debug] fireClose(${code}) — injecting`)
996
+ debugFireClose(code)
997
+ },
998
+ forceReconnect: () => {
999
+ logForDebugging('[bridge:debug] forceReconnect — injecting')
1000
+ void reconnectEnvironmentWithSession()
1001
+ },
1002
+ injectFault: injectBridgeFault,
1003
+ wakePollLoop,
1004
+ describe: () =>
1005
+ `env=${environmentId} session=${currentSessionId} transport=${transport?.getStateLabel() ?? 'null'} workId=${currentWorkId ?? 'null'}`,
1006
+ })
1007
+ }
1008
+
1009
+ const pollOpts = {
1010
+ api,
1011
+ getCredentials: () => ({ environmentId, environmentSecret }),
1012
+ signal: pollController.signal,
1013
+ getPollIntervalConfig,
1014
+ onStateChange,
1015
+ getWsState: () => transport?.getStateLabel() ?? 'null',
1016
+ // REPL bridge is single-session: having any transport == at capacity.
1017
+ // No need to check isConnectedStatus() — even while the transport is
1018
+ // auto-reconnecting internally (up to 10 min), poll is heartbeat-only.
1019
+ isAtCapacity: () => transport !== null,
1020
+ capacitySignal,
1021
+ onFatalError: triggerTeardown,
1022
+ getHeartbeatInfo: () => {
1023
+ if (!currentWorkId || !currentIngressToken) {
1024
+ return null
1025
+ }
1026
+ return {
1027
+ environmentId,
1028
+ workId: currentWorkId,
1029
+ sessionToken: currentIngressToken,
1030
+ }
1031
+ },
1032
+ // Work-item JWT expired (or work gone). The transport is useless —
1033
+ // SSE reconnects and CCR writes use the same stale token. Without
1034
+ // this callback the poll loop would do a 10-min at-capacity backoff,
1035
+ // during which the work lease (300s TTL) expires and the server stops
1036
+ // forwarding prompts → ~25-min dead window observed in daemon logs.
1037
+ // Kill the transport + work state so isAtCapacity()=false; the loop
1038
+ // fast-polls and picks up the server's re-dispatched work in seconds.
1039
+ onHeartbeatFatal: (err: BridgeFatalError) => {
1040
+ logForDebugging(
1041
+ `[bridge:repl] heartbeatWork fatal (status=${err.status}) — tearing down work item for fast re-dispatch`,
1042
+ )
1043
+ if (transport) {
1044
+ const seq = transport.getLastSequenceNum()
1045
+ if (seq > lastTransportSequenceNum) {
1046
+ lastTransportSequenceNum = seq
1047
+ }
1048
+ transport.close()
1049
+ transport = null
1050
+ }
1051
+ flushGate.drop()
1052
+ // force=false → server re-queues. Likely already expired, but
1053
+ // idempotent and makes re-dispatch immediate if not.
1054
+ if (currentWorkId) {
1055
+ void api
1056
+ .stopWork(environmentId, currentWorkId, false)
1057
+ .catch((e: unknown) => {
1058
+ logForDebugging(
1059
+ `[bridge:repl] stopWork after heartbeat fatal: ${errorMessage(e)}`,
1060
+ )
1061
+ })
1062
+ }
1063
+ currentWorkId = null
1064
+ currentIngressToken = null
1065
+ wakePollLoop()
1066
+ onStateChange?.(
1067
+ 'reconnecting',
1068
+ 'Work item lease expired, fetching fresh token',
1069
+ )
1070
+ },
1071
+ async onEnvironmentLost() {
1072
+ const success = await reconnectEnvironmentWithSession()
1073
+ if (!success) {
1074
+ return null
1075
+ }
1076
+ return { environmentId, environmentSecret }
1077
+ },
1078
+ onWorkReceived: (
1079
+ workSessionId: string,
1080
+ ingressToken: string,
1081
+ workId: string,
1082
+ serverUseCcrV2: boolean,
1083
+ ) => {
1084
+ // When new work arrives while a transport is already open, the
1085
+ // server has decided to re-dispatch (e.g. token rotation, server
1086
+ // restart). Close the existing transport and reconnect — discarding
1087
+ // the work causes a stuck 'reconnecting' state if the old WS dies
1088
+ // shortly after (the server won't re-dispatch a work item it
1089
+ // already delivered).
1090
+ // ingressToken (JWT) is stored for heartbeat auth (both v1 and v2).
1091
+ // Transport auth diverges — see the v1/v2 split below.
1092
+ if (transport?.isConnectedStatus()) {
1093
+ logForDebugging(
1094
+ `[bridge:repl] Work received while transport connected, replacing with fresh token (workId=${workId})`,
1095
+ )
1096
+ }
1097
+
1098
+ logForDebugging(
1099
+ `[bridge:repl] Work received: workId=${workId} workSessionId=${workSessionId} currentSessionId=${currentSessionId} match=${sameSessionId(workSessionId, currentSessionId)}`,
1100
+ )
1101
+
1102
+ // Refresh the crash-recovery pointer's mtime. Staleness checks file
1103
+ // mtime (not embedded timestamp) so this re-write bumps the clock —
1104
+ // a 5h+ session that crashes still has a fresh pointer. Fires once
1105
+ // per work dispatch (infrequent — bounded by user message rate).
1106
+ void writeBridgePointer(dir, {
1107
+ sessionId: currentSessionId,
1108
+ environmentId,
1109
+ source: 'repl',
1110
+ })
1111
+
1112
+ // Reject foreign session IDs — the server shouldn't assign sessions
1113
+ // from other environments. Since we create env+session as a pair,
1114
+ // a mismatch indicates an unexpected server-side reassignment.
1115
+ //
1116
+ // Compare by underlying UUID, not by tagged-ID prefix. When CCR
1117
+ // v2's compat layer serves the session, createBridgeSession gets
1118
+ // session_* from the v1-facing API (compat/convert.go:41) but the
1119
+ // infrastructure layer delivers cse_* in the work queue
1120
+ // (container_manager.go:129). Same UUID, different tag.
1121
+ if (!sameSessionId(workSessionId, currentSessionId)) {
1122
+ logForDebugging(
1123
+ `[bridge:repl] Rejecting foreign session: expected=${currentSessionId} got=${workSessionId}`,
1124
+ )
1125
+ return
1126
+ }
1127
+
1128
+ currentWorkId = workId
1129
+ currentIngressToken = ingressToken
1130
+
1131
+ // Server decides per-session (secret.use_code_sessions from the work
1132
+ // secret, threaded through runWorkPollLoop). The env var is an ant-dev
1133
+ // override for forcing v2 before the server flag is on for your user —
1134
+ // requires ccr_v2_compat_enabled server-side or registerWorker 404s.
1135
+ //
1136
+ // Kept separate from CLAUDE_CODE_USE_CCR_V2 (the child-SDK transport
1137
+ // selector set by sessionRunner/environment-manager) to avoid the
1138
+ // inheritance hazard in spawn mode where the parent's orchestrator
1139
+ // var would leak into a v1 child.
1140
+ const useCcrV2 =
1141
+ serverUseCcrV2 || isEnvTruthy(process.env.CLAUDE_BRIDGE_USE_CCR_V2)
1142
+
1143
+ // Auth is the one place v1 and v2 diverge hard:
1144
+ //
1145
+ // - v1 (Session-Ingress): accepts OAuth OR JWT. We prefer OAuth
1146
+ // because the standard OAuth refresh flow handles expiry — no
1147
+ // separate JWT refresh scheduler needed.
1148
+ //
1149
+ // - v2 (CCR /worker/*): REQUIRES the JWT. register_worker.go:32
1150
+ // validates the session_id claim, which OAuth tokens don't carry.
1151
+ // The JWT from the work secret has both that claim and the worker
1152
+ // role (environment_auth.py:856). JWT refresh: when it expires the
1153
+ // server re-dispatches work with a fresh one, and onWorkReceived
1154
+ // fires again. createV2ReplTransport stores it via
1155
+ // updateSessionIngressAuthToken() before touching the network.
1156
+ let v1OauthToken: string | undefined
1157
+ if (!useCcrV2) {
1158
+ v1OauthToken = getOAuthToken()
1159
+ if (!v1OauthToken) {
1160
+ logForDebugging(
1161
+ '[bridge:repl] No OAuth token available for session ingress, skipping work',
1162
+ )
1163
+ return
1164
+ }
1165
+ updateSessionIngressAuthToken(v1OauthToken)
1166
+ }
1167
+ logEvent('tengu_bridge_repl_work_received', {})
1168
+
1169
+ // Close the previous transport. Nullify BEFORE calling close() so
1170
+ // the close callback doesn't treat the programmatic close as
1171
+ // "session ended normally" and trigger a full teardown.
1172
+ if (transport) {
1173
+ const oldTransport = transport
1174
+ transport = null
1175
+ // Capture the SSE sequence high-water mark so the next transport
1176
+ // resumes the stream instead of replaying from seq 0. Use max() —
1177
+ // a transport that died early (never received any frames) would
1178
+ // otherwise reset a non-zero mark back to 0.
1179
+ const oldSeq = oldTransport.getLastSequenceNum()
1180
+ if (oldSeq > lastTransportSequenceNum) {
1181
+ lastTransportSequenceNum = oldSeq
1182
+ }
1183
+ oldTransport.close()
1184
+ }
1185
+ // Reset flush state — the old flush (if any) is no longer relevant.
1186
+ // Preserve pending messages so they're drained after the new
1187
+ // transport's flush completes (the hook has already advanced its
1188
+ // lastWrittenIndex and won't re-send them).
1189
+ flushGate.deactivate()
1190
+
1191
+ // Closure adapter over the shared handleServerControlRequest —
1192
+ // captures transport/currentSessionId so the transport.setOnData
1193
+ // callback below doesn't need to thread them through.
1194
+ const onServerControlRequest = (request: SDKControlRequest): void =>
1195
+ handleServerControlRequest(request, {
1196
+ transport,
1197
+ sessionId: currentSessionId,
1198
+ onInterrupt,
1199
+ onSetModel,
1200
+ onSetMaxThinkingTokens,
1201
+ onSetPermissionMode,
1202
+ })
1203
+
1204
+ let initialFlushDone = false
1205
+
1206
+ // Wire callbacks onto a freshly constructed transport and connect.
1207
+ // Extracted so the (sync) v1 and (async) v2 construction paths can
1208
+ // share the identical callback + flush machinery.
1209
+ const wireTransport = (newTransport: ReplBridgeTransport): void => {
1210
+ transport = newTransport
1211
+
1212
+ newTransport.setOnConnect(() => {
1213
+ // Guard: if transport was replaced by a newer onWorkReceived call
1214
+ // while the WS was connecting, ignore this stale callback.
1215
+ if (transport !== newTransport) return
1216
+
1217
+ logForDebugging('[bridge:repl] Ingress transport connected')
1218
+ logEvent('tengu_bridge_repl_ws_connected', {})
1219
+
1220
+ // Update the env var with the latest OAuth token so POST writes
1221
+ // (which read via getSessionIngressAuthToken()) use a fresh token.
1222
+ // v2 skips this — createV2ReplTransport already stored the JWT,
1223
+ // and overwriting it with OAuth would break subsequent /worker/*
1224
+ // requests (session_id claim check).
1225
+ if (!useCcrV2) {
1226
+ const freshToken = getOAuthToken()
1227
+ if (freshToken) {
1228
+ updateSessionIngressAuthToken(freshToken)
1229
+ }
1230
+ }
1231
+
1232
+ // Reset teardownStarted so future teardowns are not blocked.
1233
+ teardownStarted = false
1234
+
1235
+ // Flush initial messages only on first connect, not on every
1236
+ // WS reconnection. Re-flushing would cause duplicate messages.
1237
+ // IMPORTANT: onStateChange('connected') is deferred until the
1238
+ // flush completes. This prevents writeMessages() from sending
1239
+ // new messages that could arrive at the server interleaved with
1240
+ // the historical messages, and delays the web UI from showing
1241
+ // the session as active until history is persisted.
1242
+ if (
1243
+ !initialFlushDone &&
1244
+ initialMessages &&
1245
+ initialMessages.length > 0
1246
+ ) {
1247
+ initialFlushDone = true
1248
+
1249
+ // Cap the initial flush to the most recent N messages. The full
1250
+ // history is UI-only (model doesn't see it) and large replays cause
1251
+ // slow session-ingress persistence (each event is a threadstore write)
1252
+ // plus elevated Firestore pressure. A 0 or negative cap disables it.
1253
+ const historyCap = initialHistoryCap
1254
+ const eligibleMessages = initialMessages.filter(
1255
+ m =>
1256
+ isEligibleBridgeMessage(m) &&
1257
+ !previouslyFlushedUUIDs?.has(m.uuid),
1258
+ )
1259
+ const cappedMessages =
1260
+ historyCap > 0 && eligibleMessages.length > historyCap
1261
+ ? eligibleMessages.slice(-historyCap)
1262
+ : eligibleMessages
1263
+ if (cappedMessages.length < eligibleMessages.length) {
1264
+ logForDebugging(
1265
+ `[bridge:repl] Capped initial flush: ${eligibleMessages.length} -> ${cappedMessages.length} (cap=${historyCap})`,
1266
+ )
1267
+ logEvent('tengu_bridge_repl_history_capped', {
1268
+ eligible_count: eligibleMessages.length,
1269
+ capped_count: cappedMessages.length,
1270
+ })
1271
+ }
1272
+ const sdkMessages = toSDKMessages(cappedMessages)
1273
+ if (sdkMessages.length > 0) {
1274
+ logForDebugging(
1275
+ `[bridge:repl] Flushing ${sdkMessages.length} initial message(s) via transport`,
1276
+ )
1277
+ const events = sdkMessages.map(sdkMsg => ({
1278
+ ...sdkMsg,
1279
+ session_id: currentSessionId,
1280
+ }))
1281
+ const dropsBefore = newTransport.droppedBatchCount
1282
+ void newTransport
1283
+ .writeBatch(events)
1284
+ .then(() => {
1285
+ // If any batch was dropped during this flush (SI down for
1286
+ // maxConsecutiveFailures attempts), flush() still resolved
1287
+ // normally but the events were NOT delivered. Don't mark
1288
+ // UUIDs as flushed — keep them eligible for re-send on the
1289
+ // next onWorkReceived (JWT refresh re-dispatch, line ~1144).
1290
+ if (newTransport.droppedBatchCount > dropsBefore) {
1291
+ logForDebugging(
1292
+ `[bridge:repl] Initial flush dropped ${newTransport.droppedBatchCount - dropsBefore} batch(es) — not marking ${sdkMessages.length} UUID(s) as flushed`,
1293
+ )
1294
+ return
1295
+ }
1296
+ if (previouslyFlushedUUIDs) {
1297
+ for (const sdkMsg of sdkMessages) {
1298
+ if (sdkMsg.uuid) {
1299
+ previouslyFlushedUUIDs.add(sdkMsg.uuid)
1300
+ }
1301
+ }
1302
+ }
1303
+ })
1304
+ .catch(e =>
1305
+ logForDebugging(`[bridge:repl] Initial flush failed: ${e}`),
1306
+ )
1307
+ .finally(() => {
1308
+ // Guard: if transport was replaced during the flush,
1309
+ // don't signal connected or drain — the new transport
1310
+ // owns the lifecycle now.
1311
+ if (transport !== newTransport) return
1312
+ drainFlushGate()
1313
+ onStateChange?.('connected')
1314
+ })
1315
+ } else {
1316
+ // All initial messages were already flushed (filtered by
1317
+ // previouslyFlushedUUIDs). No flush POST needed — clear
1318
+ // the flag and signal connected immediately. This is the
1319
+ // first connect for this transport (inside !initialFlushDone),
1320
+ // so no flush POST is in-flight — the flag was set before
1321
+ // connect() and must be cleared here.
1322
+ drainFlushGate()
1323
+ onStateChange?.('connected')
1324
+ }
1325
+ } else if (!flushGate.active) {
1326
+ // No initial messages or already flushed on first connect.
1327
+ // WS auto-reconnect path — only signal connected if no flush
1328
+ // POST is in-flight. If one is, .finally() owns the lifecycle.
1329
+ onStateChange?.('connected')
1330
+ }
1331
+ })
1332
+
1333
+ newTransport.setOnData(data => {
1334
+ handleIngressMessage(
1335
+ data,
1336
+ recentPostedUUIDs,
1337
+ recentInboundUUIDs,
1338
+ onInboundMessage,
1339
+ onPermissionResponse,
1340
+ onServerControlRequest,
1341
+ )
1342
+ })
1343
+
1344
+ // Body lives at initBridgeCore scope so /bridge-kick can call it
1345
+ // directly via debugFireClose. All referenced closures (transport,
1346
+ // wakePollLoop, flushGate, reconnectEnvironmentWithSession, etc.)
1347
+ // are already at that scope. The only lexical dependency on
1348
+ // wireTransport was `newTransport.getLastSequenceNum()` — but after
1349
+ // the guard below passes we know transport === newTransport.
1350
+ debugFireClose = handleTransportPermanentClose
1351
+ newTransport.setOnClose(closeCode => {
1352
+ // Guard: if transport was replaced, ignore stale close.
1353
+ if (transport !== newTransport) return
1354
+ handleTransportPermanentClose(closeCode)
1355
+ })
1356
+
1357
+ // Start the flush gate before connect() to cover the WS handshake
1358
+ // window. Between transport assignment and setOnConnect firing,
1359
+ // writeMessages() could send messages via HTTP POST before the
1360
+ // initial flush starts. Starting the gate here ensures those
1361
+ // calls are queued. If there are no initial messages, the gate
1362
+ // stays inactive.
1363
+ if (
1364
+ !initialFlushDone &&
1365
+ initialMessages &&
1366
+ initialMessages.length > 0
1367
+ ) {
1368
+ flushGate.start()
1369
+ }
1370
+
1371
+ newTransport.connect()
1372
+ } // end wireTransport
1373
+
1374
+ // Bump unconditionally — ANY new transport (v1 or v2) invalidates an
1375
+ // in-flight v2 handshake. Also bumped in doReconnect().
1376
+ v2Generation++
1377
+
1378
+ if (useCcrV2) {
1379
+ // workSessionId is the cse_* form (infrastructure-layer ID from the
1380
+ // work queue), which is what /v1/code/sessions/{id}/worker/* wants.
1381
+ // The session_* form (currentSessionId) is NOT usable here —
1382
+ // handler/convert.go:30 validates TagCodeSession.
1383
+ const sessionUrl = buildCCRv2SdkUrl(baseUrl, workSessionId)
1384
+ const thisGen = v2Generation
1385
+ logForDebugging(
1386
+ `[bridge:repl] CCR v2: sessionUrl=${sessionUrl} session=${workSessionId} gen=${thisGen}`,
1387
+ )
1388
+ void createV2ReplTransport({
1389
+ sessionUrl,
1390
+ ingressToken,
1391
+ sessionId: workSessionId,
1392
+ initialSequenceNum: lastTransportSequenceNum,
1393
+ }).then(
1394
+ t => {
1395
+ // Teardown started while registerWorker was in flight. Teardown
1396
+ // saw transport === null and skipped close(); installing now
1397
+ // would leak CCRClient heartbeat timers and reset
1398
+ // teardownStarted via wireTransport's side effects.
1399
+ if (pollController.signal.aborted) {
1400
+ t.close()
1401
+ return
1402
+ }
1403
+ // onWorkReceived may have fired again while registerWorker()
1404
+ // was in flight (server re-dispatch with a fresh JWT). The
1405
+ // transport !== null check alone gets the race wrong when BOTH
1406
+ // attempts saw transport === null — it keeps the first resolver
1407
+ // (stale epoch) and discards the second (correct epoch). The
1408
+ // generation check catches it regardless of transport state.
1409
+ if (thisGen !== v2Generation) {
1410
+ logForDebugging(
1411
+ `[bridge:repl] CCR v2: discarding stale handshake gen=${thisGen} current=${v2Generation}`,
1412
+ )
1413
+ t.close()
1414
+ return
1415
+ }
1416
+ wireTransport(t)
1417
+ },
1418
+ (err: unknown) => {
1419
+ logForDebugging(
1420
+ `[bridge:repl] CCR v2: createV2ReplTransport failed: ${errorMessage(err)}`,
1421
+ { level: 'error' },
1422
+ )
1423
+ logEvent('tengu_bridge_repl_ccr_v2_init_failed', {})
1424
+ // If a newer attempt is in flight or already succeeded, don't
1425
+ // touch its work item — our failure is irrelevant.
1426
+ if (thisGen !== v2Generation) return
1427
+ // Release the work item so the server re-dispatches immediately
1428
+ // instead of waiting for its own timeout. currentWorkId was set
1429
+ // above; without this, the session looks stuck to the user.
1430
+ if (currentWorkId) {
1431
+ void api
1432
+ .stopWork(environmentId, currentWorkId, false)
1433
+ .catch((e: unknown) => {
1434
+ logForDebugging(
1435
+ `[bridge:repl] stopWork after v2 init failure: ${errorMessage(e)}`,
1436
+ )
1437
+ })
1438
+ currentWorkId = null
1439
+ currentIngressToken = null
1440
+ }
1441
+ wakePollLoop()
1442
+ },
1443
+ )
1444
+ } else {
1445
+ // v1: HybridTransport (WS reads + POST writes to Session-Ingress).
1446
+ // autoReconnect is true (default) — when the WS dies, the transport
1447
+ // reconnects automatically with exponential backoff. POST writes
1448
+ // continue during reconnection (they use getSessionIngressAuthToken()
1449
+ // independently of WS state). The poll loop remains as a secondary
1450
+ // fallback if the reconnect budget is exhausted (10 min).
1451
+ //
1452
+ // Auth: uses OAuth tokens directly instead of the JWT from the work
1453
+ // secret. refreshHeaders picks up the latest OAuth token on each
1454
+ // WS reconnect attempt.
1455
+ const wsUrl = buildSdkUrl(sessionIngressUrl, workSessionId)
1456
+ logForDebugging(`[bridge:repl] Ingress URL: ${wsUrl}`)
1457
+ logForDebugging(
1458
+ `[bridge:repl] Creating HybridTransport: session=${workSessionId}`,
1459
+ )
1460
+ // v1OauthToken was validated non-null above (we'd have returned early).
1461
+ const oauthToken = v1OauthToken ?? ''
1462
+ wireTransport(
1463
+ createV1ReplTransport(
1464
+ new HybridTransport(
1465
+ new URL(wsUrl),
1466
+ {
1467
+ Authorization: `Bearer ${oauthToken}`,
1468
+ 'anthropic-version': '2023-06-01',
1469
+ },
1470
+ workSessionId,
1471
+ () => ({
1472
+ Authorization: `Bearer ${getOAuthToken() ?? oauthToken}`,
1473
+ 'anthropic-version': '2023-06-01',
1474
+ }),
1475
+ // Cap retries so a persistently-failing session-ingress can't
1476
+ // pin the uploader drain loop for the lifetime of the bridge.
1477
+ // 50 attempts ≈ 20 min (15s POST timeout + 8s backoff + jitter
1478
+ // per cycle at steady state). Bridge-only — 1P keeps indefinite.
1479
+ {
1480
+ maxConsecutiveFailures: 50,
1481
+ isBridge: true,
1482
+ onBatchDropped: () => {
1483
+ onStateChange?.(
1484
+ 'reconnecting',
1485
+ 'Lost sync with Remote Control — events could not be delivered',
1486
+ )
1487
+ // SI has been down ~20 min. Wake the poll loop so that when
1488
+ // SI recovers, next poll → onWorkReceived → fresh transport
1489
+ // → initial flush succeeds → onStateChange('connected') at
1490
+ // ~line 1420. Without this, state stays 'reconnecting' even
1491
+ // after SI recovers — daemon.ts:437 denies all permissions,
1492
+ // useReplBridge.ts:311 keeps replBridgeSessionActive=false.
1493
+ // If the env was archived during the outage, poll 404 →
1494
+ // onEnvironmentLost recovery path handles it.
1495
+ wakePollLoop()
1496
+ },
1497
+ },
1498
+ ),
1499
+ ),
1500
+ )
1501
+ }
1502
+ },
1503
+ }
1504
+ void startWorkPollLoop(pollOpts)
1505
+
1506
+ // Perpetual mode: hourly mtime refresh of the crash-recovery pointer.
1507
+ // The onWorkReceived refresh only fires per user prompt — a
1508
+ // daemon idle for >4h would have a stale pointer, and the next restart
1509
+ // would clear it (readBridgePointer TTL check) → fresh session. The
1510
+ // standalone bridge (bridgeMain.ts) has an identical hourly timer.
1511
+ const pointerRefreshTimer = perpetual
1512
+ ? setInterval(() => {
1513
+ // doReconnect() reassigns currentSessionId/environmentId non-
1514
+ // atomically (env at ~:634, session at ~:719, awaits in between).
1515
+ // If this timer fires in that window, its fire-and-forget write can
1516
+ // race with (and overwrite) doReconnect's own pointer write at ~:740,
1517
+ // leaving the pointer at the now-archived old session. doReconnect
1518
+ // writes the pointer itself, so skipping here is free.
1519
+ if (reconnectPromise) return
1520
+ void writeBridgePointer(dir, {
1521
+ sessionId: currentSessionId,
1522
+ environmentId,
1523
+ source: 'repl',
1524
+ })
1525
+ }, 60 * 60_000)
1526
+ : null
1527
+ pointerRefreshTimer?.unref?.()
1528
+
1529
+ // Push a silent keep_alive frame on a fixed interval so upstream proxies
1530
+ // and the session-ingress layer don't GC an otherwise-idle remote control
1531
+ // session. The keep_alive type is filtered before reaching any client UI
1532
+ // (Query.ts drops it; web/iOS/Android never see it in their message loop).
1533
+ // Interval comes from GrowthBook (tengu_bridge_poll_interval_config
1534
+ // session_keepalive_interval_v2_ms, default 120s); 0 = disabled.
1535
+ const keepAliveIntervalMs =
1536
+ getPollIntervalConfig().session_keepalive_interval_v2_ms
1537
+ const keepAliveTimer =
1538
+ keepAliveIntervalMs > 0
1539
+ ? setInterval(() => {
1540
+ if (!transport) return
1541
+ logForDebugging('[bridge:repl] keep_alive sent')
1542
+ void transport.write({ type: 'keep_alive' }).catch((err: unknown) => {
1543
+ logForDebugging(
1544
+ `[bridge:repl] keep_alive write failed: ${errorMessage(err)}`,
1545
+ )
1546
+ })
1547
+ }, keepAliveIntervalMs)
1548
+ : null
1549
+ keepAliveTimer?.unref?.()
1550
+
1551
+ // Shared teardown sequence used by both cleanup registration and
1552
+ // the explicit teardown() method on the returned handle.
1553
+ let teardownStarted = false
1554
+ doTeardownImpl = async (): Promise<void> => {
1555
+ if (teardownStarted) {
1556
+ logForDebugging(
1557
+ `[bridge:repl] Teardown already in progress, skipping duplicate call env=${environmentId} session=${currentSessionId}`,
1558
+ )
1559
+ return
1560
+ }
1561
+ teardownStarted = true
1562
+ const teardownStart = Date.now()
1563
+ logForDebugging(
1564
+ `[bridge:repl] Teardown starting: env=${environmentId} session=${currentSessionId} workId=${currentWorkId ?? 'none'} transportState=${transport?.getStateLabel() ?? 'null'}`,
1565
+ )
1566
+
1567
+ if (pointerRefreshTimer !== null) {
1568
+ clearInterval(pointerRefreshTimer)
1569
+ }
1570
+ if (keepAliveTimer !== null) {
1571
+ clearInterval(keepAliveTimer)
1572
+ }
1573
+ if (sigusr2Handler) {
1574
+ process.off('SIGUSR2', sigusr2Handler)
1575
+ }
1576
+ if (process.env.USER_TYPE === 'ant') {
1577
+ clearBridgeDebugHandle()
1578
+ debugFireClose = null
1579
+ }
1580
+ pollController.abort()
1581
+ logForDebugging('[bridge:repl] Teardown: poll loop aborted')
1582
+
1583
+ // Capture the live transport's seq BEFORE close() — close() is sync
1584
+ // (just aborts the SSE fetch) and does NOT invoke onClose, so the
1585
+ // setOnClose capture path never runs for explicit teardown.
1586
+ // Without this, getSSESequenceNum() after teardown returns the stale
1587
+ // lastTransportSequenceNum (captured at the last transport swap), and
1588
+ // daemon callers persisting that value lose all events since then.
1589
+ if (transport) {
1590
+ const finalSeq = transport.getLastSequenceNum()
1591
+ if (finalSeq > lastTransportSequenceNum) {
1592
+ lastTransportSequenceNum = finalSeq
1593
+ }
1594
+ }
1595
+
1596
+ if (perpetual) {
1597
+ // Perpetual teardown is LOCAL-ONLY — do not send result, do not call
1598
+ // stopWork, do not close the transport. All of those signal the
1599
+ // server (and any mobile/attach subscribers) that the session is
1600
+ // ending. Instead: stop polling, let the socket die with the
1601
+ // process; the backend times the work-item lease back to pending on
1602
+ // its own (TTL 300s). Next daemon start reads the pointer and
1603
+ // reconnectSession re-queues work.
1604
+ transport = null
1605
+ flushGate.drop()
1606
+ // Refresh the pointer mtime so that sessions lasting longer than
1607
+ // BRIDGE_POINTER_TTL_MS (4h) don't appear stale on next start.
1608
+ await writeBridgePointer(dir, {
1609
+ sessionId: currentSessionId,
1610
+ environmentId,
1611
+ source: 'repl',
1612
+ })
1613
+ logForDebugging(
1614
+ `[bridge:repl] Teardown (perpetual): leaving env=${environmentId} session=${currentSessionId} alive on server, duration=${Date.now() - teardownStart}ms`,
1615
+ )
1616
+ return
1617
+ }
1618
+
1619
+ // Fire the result message, then archive, THEN close. transport.write()
1620
+ // only enqueues (SerialBatchEventUploader resolves on buffer-add); the
1621
+ // stopWork/archive latency (~200-500ms) is the drain window for the
1622
+ // result POST. Closing BEFORE archive meant relying on HybridTransport's
1623
+ // void-ed 3s grace period, which nothing awaits — forceExit can kill the
1624
+ // socket mid-POST. Same reorder as remoteBridgeCore.ts teardown (#22803).
1625
+ const teardownTransport = transport
1626
+ transport = null
1627
+ flushGate.drop()
1628
+ if (teardownTransport) {
1629
+ void teardownTransport.write(makeResultMessage(currentSessionId))
1630
+ }
1631
+
1632
+ const stopWorkP = currentWorkId
1633
+ ? api
1634
+ .stopWork(environmentId, currentWorkId, true)
1635
+ .then(() => {
1636
+ logForDebugging('[bridge:repl] Teardown: stopWork completed')
1637
+ })
1638
+ .catch((err: unknown) => {
1639
+ logForDebugging(
1640
+ `[bridge:repl] Teardown stopWork failed: ${errorMessage(err)}`,
1641
+ )
1642
+ })
1643
+ : Promise.resolve()
1644
+
1645
+ // Run stopWork and archiveSession in parallel. gracefulShutdown.ts:407
1646
+ // races runCleanupFunctions() against 2s (NOT the 5s outer failsafe),
1647
+ // so archive is capped at 1.5s at the injection site to stay under budget.
1648
+ // archiveSession is contractually no-throw; the injected implementations
1649
+ // log their own success/failure internally.
1650
+ await Promise.all([stopWorkP, archiveSession(currentSessionId)])
1651
+
1652
+ teardownTransport?.close()
1653
+ logForDebugging('[bridge:repl] Teardown: transport closed')
1654
+
1655
+ await api.deregisterEnvironment(environmentId).catch((err: unknown) => {
1656
+ logForDebugging(
1657
+ `[bridge:repl] Teardown deregister failed: ${errorMessage(err)}`,
1658
+ )
1659
+ })
1660
+
1661
+ // Clear the crash-recovery pointer — explicit disconnect or clean REPL
1662
+ // exit means the user is done with this session. Crash/kill-9 never
1663
+ // reaches this line, leaving the pointer for next-launch recovery.
1664
+ await clearBridgePointer(dir)
1665
+
1666
+ logForDebugging(
1667
+ `[bridge:repl] Teardown complete: env=${environmentId} duration=${Date.now() - teardownStart}ms`,
1668
+ )
1669
+ }
1670
+
1671
+ // 8. Register cleanup for graceful shutdown
1672
+ const unregister = registerCleanup(() => doTeardownImpl?.())
1673
+
1674
+ logForDebugging(
1675
+ `[bridge:repl] Ready: env=${environmentId} session=${currentSessionId}`,
1676
+ )
1677
+ onStateChange?.('ready')
1678
+
1679
+ return {
1680
+ get bridgeSessionId() {
1681
+ return currentSessionId
1682
+ },
1683
+ get environmentId() {
1684
+ return environmentId
1685
+ },
1686
+ getSSESequenceNum() {
1687
+ // lastTransportSequenceNum only updates when a transport is CLOSED
1688
+ // (captured at swap/onClose). During normal operation the CURRENT
1689
+ // transport's live seq isn't reflected there. Merge both so callers
1690
+ // (e.g. daemon persistState()) get the actual high-water mark.
1691
+ const live = transport?.getLastSequenceNum() ?? 0
1692
+ return Math.max(lastTransportSequenceNum, live)
1693
+ },
1694
+ sessionIngressUrl,
1695
+ writeMessages(messages) {
1696
+ // Filter to user/assistant messages that haven't already been sent.
1697
+ // Two layers of dedup:
1698
+ // - initialMessageUUIDs: messages sent as session creation events
1699
+ // - recentPostedUUIDs: messages recently sent via POST
1700
+ const filtered = messages.filter(
1701
+ m =>
1702
+ isEligibleBridgeMessage(m) &&
1703
+ !initialMessageUUIDs.has(m.uuid) &&
1704
+ !recentPostedUUIDs.has(m.uuid),
1705
+ )
1706
+ if (filtered.length === 0) return
1707
+
1708
+ // Fire onUserMessage for title derivation. Scan before the flushGate
1709
+ // check — prompts are title-worthy even if they queue behind the
1710
+ // initial history flush. Keeps calling on every title-worthy message
1711
+ // until the callback returns true; the caller owns the policy.
1712
+ if (!userMessageCallbackDone) {
1713
+ for (const m of filtered) {
1714
+ const text = extractTitleText(m)
1715
+ if (text !== undefined && onUserMessage?.(text, currentSessionId)) {
1716
+ userMessageCallbackDone = true
1717
+ break
1718
+ }
1719
+ }
1720
+ }
1721
+
1722
+ // Queue messages while the initial flush is in progress to prevent
1723
+ // them from arriving at the server interleaved with history.
1724
+ if (flushGate.enqueue(...filtered)) {
1725
+ logForDebugging(
1726
+ `[bridge:repl] Queued ${filtered.length} message(s) during initial flush`,
1727
+ )
1728
+ return
1729
+ }
1730
+
1731
+ if (!transport) {
1732
+ const types = filtered.map(m => m.type).join(',')
1733
+ logForDebugging(
1734
+ `[bridge:repl] Transport not configured, dropping ${filtered.length} message(s) [${types}] for session=${currentSessionId}`,
1735
+ { level: 'warn' },
1736
+ )
1737
+ return
1738
+ }
1739
+
1740
+ // Track in the bounded ring buffer for echo filtering and dedup.
1741
+ for (const msg of filtered) {
1742
+ recentPostedUUIDs.add(msg.uuid)
1743
+ }
1744
+
1745
+ logForDebugging(
1746
+ `[bridge:repl] Sending ${filtered.length} message(s) via transport`,
1747
+ )
1748
+
1749
+ // Convert to SDK format and send via HTTP POST (HybridTransport).
1750
+ // The web UI receives them via the subscribe WebSocket.
1751
+ const sdkMessages = toSDKMessages(filtered)
1752
+ const events = sdkMessages.map(sdkMsg => ({
1753
+ ...sdkMsg,
1754
+ session_id: currentSessionId,
1755
+ }))
1756
+ void transport.writeBatch(events)
1757
+ },
1758
+ writeSdkMessages(messages) {
1759
+ // Daemon path: query() already yields SDKMessage, skip conversion.
1760
+ // Still run echo dedup (server bounces writes back on the WS).
1761
+ // No initialMessageUUIDs filter — daemon has no initial messages.
1762
+ // No flushGate — daemon never starts it (no initial flush).
1763
+ const filtered = messages.filter(
1764
+ m => !m.uuid || !recentPostedUUIDs.has(m.uuid),
1765
+ )
1766
+ if (filtered.length === 0) return
1767
+ if (!transport) {
1768
+ logForDebugging(
1769
+ `[bridge:repl] Transport not configured, dropping ${filtered.length} SDK message(s) for session=${currentSessionId}`,
1770
+ { level: 'warn' },
1771
+ )
1772
+ return
1773
+ }
1774
+ for (const msg of filtered) {
1775
+ if (msg.uuid) recentPostedUUIDs.add(msg.uuid)
1776
+ }
1777
+ const events = filtered.map(m => ({ ...m, session_id: currentSessionId }))
1778
+ void transport.writeBatch(events)
1779
+ },
1780
+ sendControlRequest(request: SDKControlRequest) {
1781
+ if (!transport) {
1782
+ logForDebugging(
1783
+ '[bridge:repl] Transport not configured, skipping control_request',
1784
+ )
1785
+ return
1786
+ }
1787
+ const event = { ...request, session_id: currentSessionId }
1788
+ void transport.write(event)
1789
+ logForDebugging(
1790
+ `[bridge:repl] Sent control_request request_id=${request.request_id}`,
1791
+ )
1792
+ },
1793
+ sendControlResponse(response: SDKControlResponse) {
1794
+ if (!transport) {
1795
+ logForDebugging(
1796
+ '[bridge:repl] Transport not configured, skipping control_response',
1797
+ )
1798
+ return
1799
+ }
1800
+ const event = { ...response, session_id: currentSessionId }
1801
+ void transport.write(event)
1802
+ logForDebugging('[bridge:repl] Sent control_response')
1803
+ },
1804
+ sendControlCancelRequest(requestId: string) {
1805
+ if (!transport) {
1806
+ logForDebugging(
1807
+ '[bridge:repl] Transport not configured, skipping control_cancel_request',
1808
+ )
1809
+ return
1810
+ }
1811
+ const event = {
1812
+ type: 'control_cancel_request' as const,
1813
+ request_id: requestId,
1814
+ session_id: currentSessionId,
1815
+ }
1816
+ void transport.write(event)
1817
+ logForDebugging(
1818
+ `[bridge:repl] Sent control_cancel_request request_id=${requestId}`,
1819
+ )
1820
+ },
1821
+ sendResult() {
1822
+ if (!transport) {
1823
+ logForDebugging(
1824
+ `[bridge:repl] sendResult: skipping, transport not configured session=${currentSessionId}`,
1825
+ )
1826
+ return
1827
+ }
1828
+ void transport.write(makeResultMessage(currentSessionId))
1829
+ logForDebugging(
1830
+ `[bridge:repl] Sent result for session=${currentSessionId}`,
1831
+ )
1832
+ },
1833
+ async teardown() {
1834
+ unregister()
1835
+ await doTeardownImpl?.()
1836
+ logForDebugging('[bridge:repl] Torn down')
1837
+ logEvent('tengu_bridge_repl_teardown', {})
1838
+ },
1839
+ }
1840
+ }
1841
+
1842
+ /**
1843
+ * Persistent poll loop for work items. Runs in the background for the
1844
+ * lifetime of the bridge connection.
1845
+ *
1846
+ * When a work item arrives, acknowledges it and calls onWorkReceived
1847
+ * with the session ID and ingress token (which connects the ingress
1848
+ * WebSocket). Then continues polling — the server will dispatch a new
1849
+ * work item if the ingress WebSocket drops, allowing automatic
1850
+ * reconnection without tearing down the bridge.
1851
+ */
1852
+ async function startWorkPollLoop({
1853
+ api,
1854
+ getCredentials,
1855
+ signal,
1856
+ onStateChange,
1857
+ onWorkReceived,
1858
+ onEnvironmentLost,
1859
+ getWsState,
1860
+ isAtCapacity,
1861
+ capacitySignal,
1862
+ onFatalError,
1863
+ getPollIntervalConfig = () => DEFAULT_POLL_CONFIG,
1864
+ getHeartbeatInfo,
1865
+ onHeartbeatFatal,
1866
+ }: {
1867
+ api: BridgeApiClient
1868
+ getCredentials: () => { environmentId: string; environmentSecret: string }
1869
+ signal: AbortSignal
1870
+ onStateChange?: (state: BridgeState, detail?: string) => void
1871
+ onWorkReceived: (
1872
+ sessionId: string,
1873
+ ingressToken: string,
1874
+ workId: string,
1875
+ useCodeSessions: boolean,
1876
+ ) => void
1877
+ /** Called when the environment has been deleted. Returns new credentials or null. */
1878
+ onEnvironmentLost?: () => Promise<{
1879
+ environmentId: string
1880
+ environmentSecret: string
1881
+ } | null>
1882
+ /** Returns the current WebSocket readyState label for diagnostic logging. */
1883
+ getWsState?: () => string
1884
+ /**
1885
+ * Returns true when the caller cannot accept new work (transport already
1886
+ * connected). When true, the loop polls at the configured at-capacity
1887
+ * interval as a heartbeat only. Server-side BRIDGE_LAST_POLL_TTL is
1888
+ * 4 hours — anything shorter than that is sufficient for liveness.
1889
+ */
1890
+ isAtCapacity?: () => boolean
1891
+ /**
1892
+ * Produces a signal that aborts when capacity frees up (transport lost),
1893
+ * merged with the loop signal. Used to interrupt the at-capacity sleep
1894
+ * so recovery polling starts immediately.
1895
+ */
1896
+ capacitySignal?: () => CapacitySignal
1897
+ /** Called on unrecoverable errors (e.g. server-side expiry) to trigger full teardown. */
1898
+ onFatalError?: () => void
1899
+ /** Poll interval config getter — defaults to DEFAULT_POLL_CONFIG. */
1900
+ getPollIntervalConfig?: () => PollIntervalConfig
1901
+ /**
1902
+ * Returns the current work ID and session ingress token for heartbeat.
1903
+ * When null, heartbeat is not possible (no active work item).
1904
+ */
1905
+ getHeartbeatInfo?: () => {
1906
+ environmentId: string
1907
+ workId: string
1908
+ sessionToken: string
1909
+ } | null
1910
+ /**
1911
+ * Called when heartbeatWork throws BridgeFatalError (401/403/404/410 —
1912
+ * JWT expired or work item gone). Caller should tear down the transport
1913
+ * + work state so isAtCapacity() flips to false and the loop fast-polls
1914
+ * for the server's re-dispatched work item. When provided, the loop
1915
+ * SKIPS the at-capacity backoff sleep (which would otherwise cause a
1916
+ * ~10-minute dead window before recovery). When omitted, falls back to
1917
+ * the backoff sleep to avoid a tight poll+heartbeat loop.
1918
+ */
1919
+ onHeartbeatFatal?: (err: BridgeFatalError) => void
1920
+ }): Promise<void> {
1921
+ const MAX_ENVIRONMENT_RECREATIONS = 3
1922
+
1923
+ logForDebugging(
1924
+ `[bridge:repl] Starting work poll loop for env=${getCredentials().environmentId}`,
1925
+ )
1926
+
1927
+ let consecutiveErrors = 0
1928
+ let firstErrorTime: number | null = null
1929
+ let lastPollErrorTime: number | null = null
1930
+ let environmentRecreations = 0
1931
+ // Set when the at-capacity sleep overruns its deadline by a large margin
1932
+ // (process suspension). Consumed at the top of the next iteration to
1933
+ // force one fast-poll cycle — isAtCapacity() is `transport !== null`,
1934
+ // which stays true while the transport auto-reconnects, so the poll
1935
+ // loop would otherwise go straight back to a 10-minute sleep on a
1936
+ // transport that may be pointed at a dead socket.
1937
+ let suspensionDetected = false
1938
+
1939
+ while (!signal.aborted) {
1940
+ // Capture credentials outside try so the catch block can detect
1941
+ // whether a concurrent reconnection replaced the environment.
1942
+ const { environmentId: envId, environmentSecret: envSecret } =
1943
+ getCredentials()
1944
+ const pollConfig = getPollIntervalConfig()
1945
+ try {
1946
+ const work = await api.pollForWork(
1947
+ envId,
1948
+ envSecret,
1949
+ signal,
1950
+ pollConfig.reclaim_older_than_ms,
1951
+ )
1952
+
1953
+ // A successful poll proves the env is genuinely healthy — reset the
1954
+ // env-loss counter so events hours apart each start fresh. Outside
1955
+ // the state-change guard below because onEnvLost's success path
1956
+ // already emits 'ready'; emitting again here would be a duplicate.
1957
+ // (onEnvLost returning creds does NOT reset this — that would break
1958
+ // oscillation protection when the new env immediately dies.)
1959
+ environmentRecreations = 0
1960
+
1961
+ // Reset error tracking on successful poll
1962
+ if (consecutiveErrors > 0) {
1963
+ logForDebugging(
1964
+ `[bridge:repl] Poll recovered after ${consecutiveErrors} consecutive error(s)`,
1965
+ )
1966
+ consecutiveErrors = 0
1967
+ firstErrorTime = null
1968
+ lastPollErrorTime = null
1969
+ onStateChange?.('ready')
1970
+ }
1971
+
1972
+ if (!work) {
1973
+ // Read-and-clear: after a detected suspension, skip the at-capacity
1974
+ // branch exactly once. The pollForWork above already refreshed the
1975
+ // server's BRIDGE_LAST_POLL_TTL; this fast cycle gives any
1976
+ // re-dispatched work item a chance to land before we go back under.
1977
+ const skipAtCapacityOnce = suspensionDetected
1978
+ suspensionDetected = false
1979
+ if (isAtCapacity?.() && capacitySignal && !skipAtCapacityOnce) {
1980
+ const atCapMs = pollConfig.poll_interval_ms_at_capacity
1981
+ // Heartbeat loops WITHOUT polling. When at-capacity polling is also
1982
+ // enabled (atCapMs > 0), the loop tracks a deadline and breaks out
1983
+ // to poll at that interval — heartbeat and poll compose instead of
1984
+ // one suppressing the other. Breaks out when:
1985
+ // - Poll deadline reached (atCapMs > 0 only)
1986
+ // - Auth fails (JWT expired → poll refreshes tokens)
1987
+ // - Capacity wake fires (transport lost → poll for new work)
1988
+ // - Heartbeat config disabled (GrowthBook update)
1989
+ // - Loop aborted (shutdown)
1990
+ if (
1991
+ pollConfig.non_exclusive_heartbeat_interval_ms > 0 &&
1992
+ getHeartbeatInfo
1993
+ ) {
1994
+ logEvent('tengu_bridge_heartbeat_mode_entered', {
1995
+ heartbeat_interval_ms:
1996
+ pollConfig.non_exclusive_heartbeat_interval_ms,
1997
+ })
1998
+ // Deadline computed once at entry — GB updates to atCapMs don't
1999
+ // shift an in-flight deadline (next entry picks up the new value).
2000
+ const pollDeadline = atCapMs > 0 ? Date.now() + atCapMs : null
2001
+ let needsBackoff = false
2002
+ let hbCycles = 0
2003
+ while (
2004
+ !signal.aborted &&
2005
+ isAtCapacity() &&
2006
+ (pollDeadline === null || Date.now() < pollDeadline)
2007
+ ) {
2008
+ const hbConfig = getPollIntervalConfig()
2009
+ if (hbConfig.non_exclusive_heartbeat_interval_ms <= 0) break
2010
+
2011
+ const info = getHeartbeatInfo()
2012
+ if (!info) break
2013
+
2014
+ // Capture capacity signal BEFORE the async heartbeat call so
2015
+ // a transport loss during the HTTP request is caught by the
2016
+ // subsequent sleep.
2017
+ const cap = capacitySignal()
2018
+
2019
+ try {
2020
+ await api.heartbeatWork(
2021
+ info.environmentId,
2022
+ info.workId,
2023
+ info.sessionToken,
2024
+ )
2025
+ } catch (err) {
2026
+ logForDebugging(
2027
+ `[bridge:repl:heartbeat] Failed: ${errorMessage(err)}`,
2028
+ )
2029
+ if (err instanceof BridgeFatalError) {
2030
+ cap.cleanup()
2031
+ logEvent('tengu_bridge_heartbeat_error', {
2032
+ status:
2033
+ err.status as unknown as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2034
+ error_type: (err.status === 401 || err.status === 403
2035
+ ? 'auth_failed'
2036
+ : 'fatal') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2037
+ })
2038
+ // JWT expired (401/403) or work item gone (404/410).
2039
+ // Either way the current transport is dead — SSE
2040
+ // reconnects and CCR writes will fail on the same
2041
+ // stale token. If the caller gave us a recovery hook,
2042
+ // tear down work state and skip backoff: isAtCapacity()
2043
+ // flips to false, next outer-loop iteration fast-polls
2044
+ // for the server's re-dispatched work item. Without
2045
+ // the hook, backoff to avoid tight poll+heartbeat loop.
2046
+ if (onHeartbeatFatal) {
2047
+ onHeartbeatFatal(err)
2048
+ logForDebugging(
2049
+ `[bridge:repl:heartbeat] Fatal (status=${err.status}), work state cleared — fast-polling for re-dispatch`,
2050
+ )
2051
+ } else {
2052
+ needsBackoff = true
2053
+ }
2054
+ break
2055
+ }
2056
+ }
2057
+
2058
+ hbCycles++
2059
+ await sleep(
2060
+ hbConfig.non_exclusive_heartbeat_interval_ms,
2061
+ cap.signal,
2062
+ )
2063
+ cap.cleanup()
2064
+ }
2065
+
2066
+ const exitReason = needsBackoff
2067
+ ? 'error'
2068
+ : signal.aborted
2069
+ ? 'shutdown'
2070
+ : !isAtCapacity()
2071
+ ? 'capacity_changed'
2072
+ : pollDeadline !== null && Date.now() >= pollDeadline
2073
+ ? 'poll_due'
2074
+ : 'config_disabled'
2075
+ logEvent('tengu_bridge_heartbeat_mode_exited', {
2076
+ reason:
2077
+ exitReason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2078
+ heartbeat_cycles: hbCycles,
2079
+ })
2080
+
2081
+ // On auth_failed or fatal, backoff before polling to avoid a
2082
+ // tight poll+heartbeat loop. Fall through to the shared sleep
2083
+ // below — it's the same capacitySignal-wrapped sleep the legacy
2084
+ // path uses, and both need the suspension-overrun check.
2085
+ if (!needsBackoff) {
2086
+ if (exitReason === 'poll_due') {
2087
+ // bridgeApi throttles empty-poll logs (EMPTY_POLL_LOG_INTERVAL=100)
2088
+ // so the once-per-10min poll_due poll is invisible at counter=2.
2089
+ // Log it here so verification runs see both endpoints in the debug log.
2090
+ logForDebugging(
2091
+ `[bridge:repl] Heartbeat poll_due after ${hbCycles} cycles — falling through to pollForWork`,
2092
+ )
2093
+ }
2094
+ continue
2095
+ }
2096
+ }
2097
+ // At-capacity sleep — reached by both the legacy path (heartbeat
2098
+ // disabled) and the heartbeat-backoff path (needsBackoff=true).
2099
+ // Merged so the suspension detector covers both; previously the
2100
+ // backoff path had no overrun check and could go straight back
2101
+ // under for 10 min after a laptop wake. Use atCapMs when enabled,
2102
+ // else the heartbeat interval as a floor (guaranteed > 0 on the
2103
+ // backoff path) so heartbeat-only configs don't tight-loop.
2104
+ const sleepMs =
2105
+ atCapMs > 0
2106
+ ? atCapMs
2107
+ : pollConfig.non_exclusive_heartbeat_interval_ms
2108
+ if (sleepMs > 0) {
2109
+ const cap = capacitySignal()
2110
+ const sleepStart = Date.now()
2111
+ await sleep(sleepMs, cap.signal)
2112
+ cap.cleanup()
2113
+ // Process-suspension detector. A setTimeout overshooting its
2114
+ // deadline by 60s means the process was suspended (laptop lid,
2115
+ // SIGSTOP, VM pause) — even a pathological GC pause is seconds,
2116
+ // not minutes. Early aborts (wakePollLoop → cap.signal) produce
2117
+ // overrun < 0 and fall through. Note: this only catches sleeps
2118
+ // that outlast their deadline; WebSocketTransport's ping
2119
+ // interval (10s granularity) is the primary detector for shorter
2120
+ // suspensions. This is the backstop for when that detector isn't
2121
+ // running (transport mid-reconnect, interval stopped).
2122
+ const overrun = Date.now() - sleepStart - sleepMs
2123
+ if (overrun > 60_000) {
2124
+ logForDebugging(
2125
+ `[bridge:repl] At-capacity sleep overran by ${Math.round(overrun / 1000)}s — process suspension detected, forcing one fast-poll cycle`,
2126
+ )
2127
+ logEvent('tengu_bridge_repl_suspension_detected', {
2128
+ overrun_ms: overrun,
2129
+ })
2130
+ suspensionDetected = true
2131
+ }
2132
+ }
2133
+ } else {
2134
+ await sleep(pollConfig.poll_interval_ms_not_at_capacity, signal)
2135
+ }
2136
+ continue
2137
+ }
2138
+
2139
+ // Decode before type dispatch — need the JWT for the explicit ack.
2140
+ let secret
2141
+ try {
2142
+ secret = decodeWorkSecret(work.secret)
2143
+ } catch (err) {
2144
+ logForDebugging(
2145
+ `[bridge:repl] Failed to decode work secret: ${errorMessage(err)}`,
2146
+ )
2147
+ logEvent('tengu_bridge_repl_work_secret_failed', {})
2148
+ // Can't ack (needs the JWT we failed to decode). stopWork uses OAuth.
2149
+ // Prevents XAUTOCLAIM re-delivering this poisoned item every cycle.
2150
+ await api.stopWork(envId, work.id, false).catch(() => {})
2151
+ continue
2152
+ }
2153
+
2154
+ // Explicitly acknowledge to prevent redelivery. Non-fatal on failure:
2155
+ // server re-delivers, and the onWorkReceived callback handles dedup.
2156
+ logForDebugging(`[bridge:repl] Acknowledging workId=${work.id}`)
2157
+ try {
2158
+ await api.acknowledgeWork(envId, work.id, secret.session_ingress_token)
2159
+ } catch (err) {
2160
+ logForDebugging(
2161
+ `[bridge:repl] Acknowledge failed workId=${work.id}: ${errorMessage(err)}`,
2162
+ )
2163
+ }
2164
+
2165
+ if (work.data.type === 'healthcheck') {
2166
+ logForDebugging('[bridge:repl] Healthcheck received')
2167
+ continue
2168
+ }
2169
+
2170
+ if (work.data.type === 'session') {
2171
+ const workSessionId = work.data.id
2172
+ try {
2173
+ validateBridgeId(workSessionId, 'session_id')
2174
+ } catch {
2175
+ logForDebugging(
2176
+ `[bridge:repl] Invalid session_id in work: ${workSessionId}`,
2177
+ )
2178
+ continue
2179
+ }
2180
+
2181
+ onWorkReceived(
2182
+ workSessionId,
2183
+ secret.session_ingress_token,
2184
+ work.id,
2185
+ secret.use_code_sessions === true,
2186
+ )
2187
+ logForDebugging('[bridge:repl] Work accepted, continuing poll loop')
2188
+ }
2189
+ } catch (err) {
2190
+ if (signal.aborted) break
2191
+
2192
+ // Detect permanent "environment deleted" error — no amount of
2193
+ // retrying will recover. Re-register a new environment instead.
2194
+ // Checked BEFORE the generic BridgeFatalError bail. pollForWork uses
2195
+ // validateStatus: s => s < 500, so 404 is always wrapped into a
2196
+ // BridgeFatalError by handleErrorStatus() — never an axios-shaped
2197
+ // error. The poll endpoint's only path param is the env ID; 404
2198
+ // unambiguously means env-gone (no-work is a 200 with null body).
2199
+ // The server sends error.type='not_found_error' (standard Anthropic
2200
+ // API shape), not a bridge-specific string — but status===404 is
2201
+ // the real signal and survives body-shape changes.
2202
+ if (
2203
+ err instanceof BridgeFatalError &&
2204
+ err.status === 404 &&
2205
+ onEnvironmentLost
2206
+ ) {
2207
+ // If credentials have already been refreshed by a concurrent
2208
+ // reconnection (e.g. WS close handler), the stale poll's error
2209
+ // is expected — skip onEnvironmentLost and retry with fresh creds.
2210
+ const currentEnvId = getCredentials().environmentId
2211
+ if (envId !== currentEnvId) {
2212
+ logForDebugging(
2213
+ `[bridge:repl] Stale poll error for old env=${envId}, current env=${currentEnvId} — skipping onEnvironmentLost`,
2214
+ )
2215
+ consecutiveErrors = 0
2216
+ firstErrorTime = null
2217
+ continue
2218
+ }
2219
+
2220
+ environmentRecreations++
2221
+ logForDebugging(
2222
+ `[bridge:repl] Environment deleted, attempting re-registration (attempt ${environmentRecreations}/${MAX_ENVIRONMENT_RECREATIONS})`,
2223
+ )
2224
+ logEvent('tengu_bridge_repl_env_lost', {
2225
+ attempt: environmentRecreations,
2226
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2227
+
2228
+ if (environmentRecreations > MAX_ENVIRONMENT_RECREATIONS) {
2229
+ logForDebugging(
2230
+ `[bridge:repl] Environment re-registration limit reached (${MAX_ENVIRONMENT_RECREATIONS}), giving up`,
2231
+ )
2232
+ onStateChange?.(
2233
+ 'failed',
2234
+ 'Environment deleted and re-registration limit reached',
2235
+ )
2236
+ onFatalError?.()
2237
+ break
2238
+ }
2239
+
2240
+ onStateChange?.('reconnecting', 'environment lost, recreating session')
2241
+ const newCreds = await onEnvironmentLost()
2242
+ // doReconnect() makes several sequential network calls (1-5s).
2243
+ // If the user triggered teardown during that window, its internal
2244
+ // abort checks return false — but we need to re-check here to
2245
+ // avoid emitting a spurious 'failed' + onFatalError() during
2246
+ // graceful shutdown.
2247
+ if (signal.aborted) break
2248
+ if (newCreds) {
2249
+ // Credentials are updated in the outer scope via
2250
+ // reconnectEnvironmentWithSession — getCredentials() will
2251
+ // return the fresh values on the next poll iteration.
2252
+ // Do NOT reset environmentRecreations here — onEnvLost returning
2253
+ // creds only proves we tried to fix it, not that the env is
2254
+ // healthy. A successful poll (above) is the reset point; if the
2255
+ // new env immediately dies again we still want the limit to fire.
2256
+ consecutiveErrors = 0
2257
+ firstErrorTime = null
2258
+ onStateChange?.('ready')
2259
+ logForDebugging(
2260
+ `[bridge:repl] Re-registered environment: ${newCreds.environmentId}`,
2261
+ )
2262
+ continue
2263
+ }
2264
+
2265
+ onStateChange?.(
2266
+ 'failed',
2267
+ 'Environment deleted and re-registration failed',
2268
+ )
2269
+ onFatalError?.()
2270
+ break
2271
+ }
2272
+
2273
+ // Fatal errors (401/403/404/410) — no point retrying
2274
+ if (err instanceof BridgeFatalError) {
2275
+ const isExpiry = isExpiredErrorType(err.errorType)
2276
+ const isSuppressible = isSuppressible403(err)
2277
+ logForDebugging(
2278
+ `[bridge:repl] Fatal poll error: ${err.message} (status=${err.status}, type=${err.errorType ?? 'unknown'})${isSuppressible ? ' (suppressed)' : ''}`,
2279
+ )
2280
+ logEvent('tengu_bridge_repl_fatal_error', {
2281
+ status: err.status,
2282
+ error_type:
2283
+ err.errorType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
2284
+ })
2285
+ logForDiagnosticsNoPII(
2286
+ isExpiry ? 'info' : 'error',
2287
+ 'bridge_repl_fatal_error',
2288
+ { status: err.status, error_type: err.errorType },
2289
+ )
2290
+ // Cosmetic 403 errors (e.g., external_poll_sessions scope,
2291
+ // environments:manage permission) — suppress user-visible error
2292
+ // but always trigger teardown so cleanup runs.
2293
+ if (!isSuppressible) {
2294
+ onStateChange?.(
2295
+ 'failed',
2296
+ isExpiry
2297
+ ? 'session expired · /remote-control to reconnect'
2298
+ : err.message,
2299
+ )
2300
+ }
2301
+ // Always trigger teardown — matches bridgeMain.ts where fatalExit=true
2302
+ // is unconditional and post-loop cleanup always runs.
2303
+ onFatalError?.()
2304
+ break
2305
+ }
2306
+
2307
+ const now = Date.now()
2308
+
2309
+ // Detect system sleep/wake: if the gap since the last poll error
2310
+ // greatly exceeds the max backoff delay, the machine likely slept.
2311
+ // Reset error tracking so we retry with a fresh budget instead of
2312
+ // immediately giving up.
2313
+ if (
2314
+ lastPollErrorTime !== null &&
2315
+ now - lastPollErrorTime > POLL_ERROR_MAX_DELAY_MS * 2
2316
+ ) {
2317
+ logForDebugging(
2318
+ `[bridge:repl] Detected system sleep (${Math.round((now - lastPollErrorTime) / 1000)}s gap), resetting poll error budget`,
2319
+ )
2320
+ logForDiagnosticsNoPII('info', 'bridge_repl_poll_sleep_detected', {
2321
+ gapMs: now - lastPollErrorTime,
2322
+ })
2323
+ consecutiveErrors = 0
2324
+ firstErrorTime = null
2325
+ }
2326
+ lastPollErrorTime = now
2327
+
2328
+ consecutiveErrors++
2329
+ if (firstErrorTime === null) {
2330
+ firstErrorTime = now
2331
+ }
2332
+ const elapsed = now - firstErrorTime
2333
+ const httpStatus = extractHttpStatus(err)
2334
+ const errMsg = describeAxiosError(err)
2335
+ const wsLabel = getWsState?.() ?? 'unknown'
2336
+
2337
+ logForDebugging(
2338
+ `[bridge:repl] Poll error (attempt ${consecutiveErrors}, elapsed ${Math.round(elapsed / 1000)}s, ws=${wsLabel}): ${errMsg}`,
2339
+ )
2340
+ logEvent('tengu_bridge_repl_poll_error', {
2341
+ status: httpStatus,
2342
+ consecutiveErrors,
2343
+ elapsedMs: elapsed,
2344
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2345
+
2346
+ // Only transition to 'reconnecting' on the first error — stay
2347
+ // there until a successful poll (avoid flickering the UI state).
2348
+ if (consecutiveErrors === 1) {
2349
+ onStateChange?.('reconnecting', errMsg)
2350
+ }
2351
+
2352
+ // Give up after continuous failures
2353
+ if (elapsed >= POLL_ERROR_GIVE_UP_MS) {
2354
+ logForDebugging(
2355
+ `[bridge:repl] Poll failures exceeded ${POLL_ERROR_GIVE_UP_MS / 1000}s (${consecutiveErrors} errors), giving up`,
2356
+ )
2357
+ logForDiagnosticsNoPII('info', 'bridge_repl_poll_give_up')
2358
+ logEvent('tengu_bridge_repl_poll_give_up', {
2359
+ consecutiveErrors,
2360
+ elapsedMs: elapsed,
2361
+ lastStatus: httpStatus,
2362
+ } as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS)
2363
+ onStateChange?.('failed', 'connection to server lost')
2364
+ break
2365
+ }
2366
+
2367
+ // Exponential backoff: 2s → 4s → 8s → 16s → 32s → 60s (cap)
2368
+ const backoff = Math.min(
2369
+ POLL_ERROR_INITIAL_DELAY_MS * 2 ** (consecutiveErrors - 1),
2370
+ POLL_ERROR_MAX_DELAY_MS,
2371
+ )
2372
+ // The poll_due heartbeat-loop exit leaves a healthy lease exposed to
2373
+ // this backoff path. Heartbeat before each sleep so /poll outages
2374
+ // (the VerifyEnvironmentSecretAuth DB path heartbeat was introduced to
2375
+ // avoid) don't kill the 300s lease TTL.
2376
+ if (getPollIntervalConfig().non_exclusive_heartbeat_interval_ms > 0) {
2377
+ const info = getHeartbeatInfo?.()
2378
+ if (info) {
2379
+ try {
2380
+ await api.heartbeatWork(
2381
+ info.environmentId,
2382
+ info.workId,
2383
+ info.sessionToken,
2384
+ )
2385
+ } catch {
2386
+ // Best-effort — if heartbeat also fails the lease dies, same as
2387
+ // pre-poll_due behavior (where the only heartbeat-loop exits were
2388
+ // ones where the lease was already dying).
2389
+ }
2390
+ }
2391
+ }
2392
+ await sleep(backoff, signal)
2393
+ }
2394
+ }
2395
+
2396
+ logForDebugging(
2397
+ `[bridge:repl] Work poll loop ended (aborted=${signal.aborted}) env=${getCredentials().environmentId}`,
2398
+ )
2399
+ }
2400
+
2401
+ // Exported for testing only
2402
+ export {
2403
+ startWorkPollLoop as _startWorkPollLoopForTesting,
2404
+ POLL_ERROR_INITIAL_DELAY_MS as _POLL_ERROR_INITIAL_DELAY_MS_ForTesting,
2405
+ POLL_ERROR_MAX_DELAY_MS as _POLL_ERROR_MAX_DELAY_MS_ForTesting,
2406
+ POLL_ERROR_GIVE_UP_MS as _POLL_ERROR_GIVE_UP_MS_ForTesting,
2407
+ }