@xelauvas/xela-cli 0.1.0

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