/ hooks / useReplBridge.tsx
useReplBridge.tsx
  1  import { feature } from 'bun:bundle';
  2  import React, { useCallback, useEffect, useRef } from 'react';
  3  import { setMainLoopModelOverride } from '../bootstrap/state.js';
  4  import { type BridgePermissionCallbacks, type BridgePermissionResponse, isBridgePermissionResponse } from '../bridge/bridgePermissionCallbacks.js';
  5  import { buildBridgeConnectUrl } from '../bridge/bridgeStatusUtil.js';
  6  import { extractInboundMessageFields } from '../bridge/inboundMessages.js';
  7  import type { BridgeState, ReplBridgeHandle } from '../bridge/replBridge.js';
  8  import { setReplBridgeHandle } from '../bridge/replBridgeHandle.js';
  9  import type { Command } from '../commands.js';
 10  import { getSlashCommandToolSkills, isBridgeSafeCommand } from '../commands.js';
 11  import { getRemoteSessionUrl } from '../constants/product.js';
 12  import { useNotifications } from '../context/notifications.js';
 13  import type { PermissionMode, SDKMessage } from '../entrypoints/agentSdkTypes.js';
 14  import type { SDKControlResponse } from '../entrypoints/sdk/controlTypes.js';
 15  import { Text } from '../ink.js';
 16  import { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js';
 17  import { useAppState, useAppStateStore, useSetAppState } from '../state/AppState.js';
 18  import type { Message } from '../types/message.js';
 19  import { getCwd } from '../utils/cwd.js';
 20  import { logForDebugging } from '../utils/debug.js';
 21  import { errorMessage } from '../utils/errors.js';
 22  import { enqueue } from '../utils/messageQueueManager.js';
 23  import { buildSystemInitMessage } from '../utils/messages/systemInit.js';
 24  import { createBridgeStatusMessage, createSystemMessage } from '../utils/messages.js';
 25  import { getAutoModeUnavailableNotification, getAutoModeUnavailableReason, isAutoModeGateEnabled, isBypassPermissionsModeDisabled, transitionPermissionMode } from '../utils/permissions/permissionSetup.js';
 26  import { getLeaderToolUseConfirmQueue } from '../utils/swarm/leaderPermissionBridge.js';
 27  
 28  /** How long after a failure before replBridgeEnabled is auto-cleared (stops retries). */
 29  export const BRIDGE_FAILURE_DISMISS_MS = 10_000;
 30  
 31  /**
 32   * Max consecutive initReplBridge failures before the hook stops re-attempting
 33   * for the session lifetime. Guards against paths that flip replBridgeEnabled
 34   * back on after auto-disable (settings sync, /remote-control, config tool)
 35   * when the underlying OAuth is unrecoverable — each re-attempt is another
 36   * guaranteed 401 against POST /v1/environments/bridge. Datadog 2026-03-08:
 37   * top stuck client generated 2,879 × 401/day alone (17% of all 401s on the
 38   * route).
 39   */
 40  const MAX_CONSECUTIVE_INIT_FAILURES = 3;
 41  
 42  /**
 43   * Hook that initializes an always-on bridge connection in the background
 44   * and writes new user/assistant messages to the bridge session.
 45   *
 46   * Silently skips if bridge is not enabled or user is not OAuth-authenticated.
 47   *
 48   * Watches AppState.replBridgeEnabled — when toggled off (via /config or footer),
 49   * the bridge is torn down. When toggled back on, it re-initializes.
 50   *
 51   * Inbound messages from claude.ai are injected into the REPL via queuedCommands.
 52   */
 53  export function useReplBridge(messages: Message[], setMessages: (action: React.SetStateAction<Message[]>) => void, abortControllerRef: React.RefObject<AbortController | null>, commands: readonly Command[], mainLoopModel: string): {
 54    sendBridgeResult: () => void;
 55  } {
 56    const handleRef = useRef<ReplBridgeHandle | null>(null);
 57    const teardownPromiseRef = useRef<Promise<void> | undefined>(undefined);
 58    const lastWrittenIndexRef = useRef(0);
 59    // Tracks UUIDs already flushed as initial messages. Persists across
 60    // bridge reconnections so Bridge #2+ only sends new messages — sending
 61    // duplicate UUIDs causes the server to kill the WebSocket.
 62    const flushedUUIDsRef = useRef(new Set<string>());
 63    const failureTimeoutRef = useRef<ReturnType<typeof setTimeout> | undefined>(undefined);
 64    // Persists across effect re-runs (unlike the effect's local state). Reset
 65    // only on successful init. Hits MAX_CONSECUTIVE_INIT_FAILURES → fuse blown
 66    // for the session, regardless of replBridgeEnabled re-toggling.
 67    const consecutiveFailuresRef = useRef(0);
 68    const setAppState = useSetAppState();
 69    const commandsRef = useRef(commands);
 70    commandsRef.current = commands;
 71    const mainLoopModelRef = useRef(mainLoopModel);
 72    mainLoopModelRef.current = mainLoopModel;
 73    const messagesRef = useRef(messages);
 74    messagesRef.current = messages;
 75    const store = useAppStateStore();
 76    const {
 77      addNotification
 78    } = useNotifications();
 79    const replBridgeEnabled = feature('BRIDGE_MODE') ?
 80    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 81    useAppState(s => s.replBridgeEnabled) : false;
 82    const replBridgeConnected = feature('BRIDGE_MODE') ?
 83    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 84    useAppState(s_0 => s_0.replBridgeConnected) : false;
 85    const replBridgeOutboundOnly = feature('BRIDGE_MODE') ?
 86    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 87    useAppState(s_1 => s_1.replBridgeOutboundOnly) : false;
 88    const replBridgeInitialName = feature('BRIDGE_MODE') ?
 89    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 90    useAppState(s_2 => s_2.replBridgeInitialName) : undefined;
 91  
 92    // Initialize/teardown bridge when enabled state changes.
 93    // Passes current messages as initialMessages so the remote session
 94    // starts with the existing conversation context (e.g. from /bridge).
 95    useEffect(() => {
 96      // feature() check must use positive pattern for dead code elimination —
 97      // negative pattern (if (!feature(...)) return) does NOT eliminate
 98      // dynamic imports below.
 99      if (feature('BRIDGE_MODE')) {
100        if (!replBridgeEnabled) return;
101        const outboundOnly = replBridgeOutboundOnly;
102        function notifyBridgeFailed(detail?: string): void {
103          if (outboundOnly) return;
104          addNotification({
105            key: 'bridge-failed',
106            jsx: <>
107                <Text color="error">Remote Control failed</Text>
108                {detail && <Text dimColor> · {detail}</Text>}
109              </>,
110            priority: 'immediate'
111          });
112        }
113        if (consecutiveFailuresRef.current >= MAX_CONSECUTIVE_INIT_FAILURES) {
114          logForDebugging(`[bridge:repl] Hook: ${consecutiveFailuresRef.current} consecutive init failures, not retrying this session`);
115          // Clear replBridgeEnabled so /remote-control doesn't mistakenly show
116          // BridgeDisconnectDialog for a bridge that never connected.
117          const fuseHint = 'disabled after repeated failures · restart to retry';
118          notifyBridgeFailed(fuseHint);
119          setAppState(prev => {
120            if (prev.replBridgeError === fuseHint && !prev.replBridgeEnabled) return prev;
121            return {
122              ...prev,
123              replBridgeError: fuseHint,
124              replBridgeEnabled: false
125            };
126          });
127          return;
128        }
129        let cancelled = false;
130        // Capture messages.length now so we don't re-send initial messages
131        // through writeMessages after the bridge connects.
132        const initialMessageCount = messages.length;
133        void (async () => {
134          try {
135            // Wait for any in-progress teardown to complete before registering
136            // a new environment. Without this, the deregister HTTP call from
137            // the previous teardown races with the new register call, and the
138            // server may tear down the freshly-created environment.
139            if (teardownPromiseRef.current) {
140              logForDebugging('[bridge:repl] Hook: waiting for previous teardown to complete before re-init');
141              await teardownPromiseRef.current;
142              teardownPromiseRef.current = undefined;
143              logForDebugging('[bridge:repl] Hook: previous teardown complete, proceeding with re-init');
144            }
145            if (cancelled) return;
146  
147            // Dynamic import so the module is tree-shaken in external builds
148            const {
149              initReplBridge
150            } = await import('../bridge/initReplBridge.js');
151            const {
152              shouldShowAppUpgradeMessage
153            } = await import('../bridge/envLessBridgeConfig.js');
154  
155            // Assistant mode: perpetual bridge session — claude.ai shows one
156            // continuous conversation across CLI restarts instead of a new
157            // session per invocation. initBridgeCore reads bridge-pointer.json
158            // (the same crash-recovery file #20735 added) and reuses its
159            // {environmentId, sessionId} via reuseEnvironmentId +
160            // api.reconnectSession(). Teardown skips archive/deregister/
161            // pointer-clear so the session survives clean exits, not just
162            // crashes. Non-assistant bridges clear the pointer on teardown
163            // (crash-recovery only).
164            let perpetual = false;
165            if (feature('KAIROS')) {
166              const {
167                isAssistantMode
168              } = await import('../assistant/index.js');
169              perpetual = isAssistantMode();
170            }
171  
172            // When a user message arrives from claude.ai, inject it into the REPL.
173            // Preserves the original UUID so that when the message is forwarded
174            // back to CCR, it matches the original — avoiding duplicate messages.
175            //
176            // Async because file_attachments (if present) need a network fetch +
177            // disk write before we enqueue with the @path prefix. Caller doesn't
178            // await — messages with attachments just land in the queue slightly
179            // later, which is fine (web messages aren't rapid-fire).
180            async function handleInboundMessage(msg: SDKMessage): Promise<void> {
181              try {
182                const fields = extractInboundMessageFields(msg);
183                if (!fields) return;
184                const {
185                  uuid
186                } = fields;
187  
188                // Dynamic import keeps the bridge code out of non-BRIDGE_MODE builds.
189                const {
190                  resolveAndPrepend
191                } = await import('../bridge/inboundAttachments.js');
192                let sanitized = fields.content;
193                if (feature('KAIROS_GITHUB_WEBHOOKS')) {
194                  /* eslint-disable @typescript-eslint/no-require-imports */
195                  const {
196                    sanitizeInboundWebhookContent
197                  } = require('../bridge/webhookSanitizer.js') as typeof import('../bridge/webhookSanitizer.js');
198                  /* eslint-enable @typescript-eslint/no-require-imports */
199                  sanitized = sanitizeInboundWebhookContent(fields.content);
200                }
201                const content = await resolveAndPrepend(msg, sanitized);
202                const preview = typeof content === 'string' ? content.slice(0, 80) : `[${content.length} content blocks]`;
203                logForDebugging(`[bridge:repl] Injecting inbound user message: ${preview}${uuid ? ` uuid=${uuid}` : ''}`);
204                enqueue({
205                  value: content,
206                  mode: 'prompt' as const,
207                  uuid,
208                  // skipSlashCommands stays true as defense-in-depth —
209                  // processUserInputBase overrides it internally when bridgeOrigin
210                  // is set AND the resolved command passes isBridgeSafeCommand.
211                  // This keeps exit-word suppression and immediate-command blocks
212                  // intact for any code path that checks skipSlashCommands directly.
213                  skipSlashCommands: true,
214                  bridgeOrigin: true
215                });
216              } catch (e) {
217                logForDebugging(`[bridge:repl] handleInboundMessage failed: ${e}`, {
218                  level: 'error'
219                });
220              }
221            }
222  
223            // State change callback — maps bridge lifecycle events to AppState.
224            function handleStateChange(state: BridgeState, detail_0?: string): void {
225              if (cancelled) return;
226              if (outboundOnly) {
227                logForDebugging(`[bridge:repl] Mirror state=${state}${detail_0 ? ` detail=${detail_0}` : ''}`);
228                // Sync replBridgeConnected so the forwarding effect starts/stops
229                // writing as the transport comes up or dies.
230                if (state === 'failed') {
231                  setAppState(prev_3 => {
232                    if (!prev_3.replBridgeConnected) return prev_3;
233                    return {
234                      ...prev_3,
235                      replBridgeConnected: false
236                    };
237                  });
238                } else if (state === 'ready' || state === 'connected') {
239                  setAppState(prev_4 => {
240                    if (prev_4.replBridgeConnected) return prev_4;
241                    return {
242                      ...prev_4,
243                      replBridgeConnected: true
244                    };
245                  });
246                }
247                return;
248              }
249              const handle = handleRef.current;
250              switch (state) {
251                case 'ready':
252                  setAppState(prev_9 => {
253                    const connectUrl = handle && handle.environmentId !== '' ? buildBridgeConnectUrl(handle.environmentId, handle.sessionIngressUrl) : prev_9.replBridgeConnectUrl;
254                    const sessionUrl = handle ? getRemoteSessionUrl(handle.bridgeSessionId, handle.sessionIngressUrl) : prev_9.replBridgeSessionUrl;
255                    const envId = handle?.environmentId;
256                    const sessionId = handle?.bridgeSessionId;
257                    if (prev_9.replBridgeConnected && !prev_9.replBridgeSessionActive && !prev_9.replBridgeReconnecting && prev_9.replBridgeConnectUrl === connectUrl && prev_9.replBridgeSessionUrl === sessionUrl && prev_9.replBridgeEnvironmentId === envId && prev_9.replBridgeSessionId === sessionId) {
258                      return prev_9;
259                    }
260                    return {
261                      ...prev_9,
262                      replBridgeConnected: true,
263                      replBridgeSessionActive: false,
264                      replBridgeReconnecting: false,
265                      replBridgeConnectUrl: connectUrl,
266                      replBridgeSessionUrl: sessionUrl,
267                      replBridgeEnvironmentId: envId,
268                      replBridgeSessionId: sessionId,
269                      replBridgeError: undefined
270                    };
271                  });
272                  break;
273                case 'connected':
274                  {
275                    setAppState(prev_8 => {
276                      if (prev_8.replBridgeSessionActive) return prev_8;
277                      return {
278                        ...prev_8,
279                        replBridgeConnected: true,
280                        replBridgeSessionActive: true,
281                        replBridgeReconnecting: false,
282                        replBridgeError: undefined
283                      };
284                    });
285                    // Send system/init so remote clients (web/iOS/Android) get
286                    // session metadata. REPL uses query() directly — never hits
287                    // QueryEngine's SDKMessage layer — so this is the only path
288                    // to put system/init on the REPL-bridge wire. Skills load is
289                    // async (memoized, cheap after REPL startup); fire-and-forget
290                    // so the connected-state transition isn't blocked.
291                    if (getFeatureValue_CACHED_MAY_BE_STALE('tengu_bridge_system_init', false)) {
292                      void (async () => {
293                        try {
294                          const skills = await getSlashCommandToolSkills(getCwd());
295                          if (cancelled) return;
296                          const state_0 = store.getState();
297                          handleRef.current?.writeSdkMessages([buildSystemInitMessage({
298                            // tools/mcpClients/plugins redacted for REPL-bridge:
299                            // MCP-prefixed tool names and server names leak which
300                            // integrations the user has wired up; plugin paths leak
301                            // raw filesystem paths (username, project structure).
302                            // CCR v2 persists SDK messages to Spanner — users who
303                            // tap "Connect from phone" may not expect these on
304                            // Anthropic's servers. QueryEngine (SDK) still emits
305                            // full lists — SDK consumers expect full telemetry.
306                            tools: [],
307                            mcpClients: [],
308                            model: mainLoopModelRef.current,
309                            permissionMode: state_0.toolPermissionContext.mode as PermissionMode,
310                            // TODO: avoid the cast
311                            // Remote clients can only invoke bridge-safe commands —
312                            // advertising unsafe ones (local-jsx, unallowed local)
313                            // would let mobile/web attempt them and hit errors.
314                            commands: commandsRef.current.filter(isBridgeSafeCommand),
315                            agents: state_0.agentDefinitions.activeAgents,
316                            skills,
317                            plugins: [],
318                            fastMode: state_0.fastMode
319                          })]);
320                        } catch (err_0) {
321                          logForDebugging(`[bridge:repl] Failed to send system/init: ${errorMessage(err_0)}`, {
322                            level: 'error'
323                          });
324                        }
325                      })();
326                    }
327                    break;
328                  }
329                case 'reconnecting':
330                  setAppState(prev_7 => {
331                    if (prev_7.replBridgeReconnecting) return prev_7;
332                    return {
333                      ...prev_7,
334                      replBridgeReconnecting: true,
335                      replBridgeSessionActive: false
336                    };
337                  });
338                  break;
339                case 'failed':
340                  // Clear any previous failure dismiss timer
341                  clearTimeout(failureTimeoutRef.current);
342                  notifyBridgeFailed(detail_0);
343                  setAppState(prev_5 => ({
344                    ...prev_5,
345                    replBridgeError: detail_0,
346                    replBridgeReconnecting: false,
347                    replBridgeSessionActive: false,
348                    replBridgeConnected: false
349                  }));
350                  // Auto-disable after timeout so the hook stops retrying.
351                  failureTimeoutRef.current = setTimeout(() => {
352                    if (cancelled) return;
353                    failureTimeoutRef.current = undefined;
354                    setAppState(prev_6 => {
355                      if (!prev_6.replBridgeError) return prev_6;
356                      return {
357                        ...prev_6,
358                        replBridgeEnabled: false,
359                        replBridgeError: undefined
360                      };
361                    });
362                  }, BRIDGE_FAILURE_DISMISS_MS);
363                  break;
364              }
365            }
366  
367            // Map of pending bridge permission response handlers, keyed by request_id.
368            // Each entry is an onResponse handler waiting for CCR to reply.
369            const pendingPermissionHandlers = new Map<string, (response: BridgePermissionResponse) => void>();
370  
371            // Dispatch incoming control_response messages to registered handlers
372            function handlePermissionResponse(msg_0: SDKControlResponse): void {
373              const requestId = msg_0.response?.request_id;
374              if (!requestId) return;
375              const handler = pendingPermissionHandlers.get(requestId);
376              if (!handler) {
377                logForDebugging(`[bridge:repl] No handler for control_response request_id=${requestId}`);
378                return;
379              }
380              pendingPermissionHandlers.delete(requestId);
381              // Extract the permission decision from the control_response payload
382              const inner = msg_0.response;
383              if (inner.subtype === 'success' && inner.response && isBridgePermissionResponse(inner.response)) {
384                handler(inner.response);
385              }
386            }
387            const handle_0 = await initReplBridge({
388              outboundOnly,
389              tags: outboundOnly ? ['ccr-mirror'] : undefined,
390              onInboundMessage: handleInboundMessage,
391              onPermissionResponse: handlePermissionResponse,
392              onInterrupt() {
393                abortControllerRef.current?.abort();
394              },
395              onSetModel(model) {
396                const resolved = model === 'default' ? null : model ?? null;
397                setMainLoopModelOverride(resolved);
398                setAppState(prev_10 => {
399                  if (prev_10.mainLoopModelForSession === resolved) return prev_10;
400                  return {
401                    ...prev_10,
402                    mainLoopModelForSession: resolved
403                  };
404                });
405              },
406              onSetMaxThinkingTokens(maxTokens) {
407                const enabled = maxTokens !== null;
408                setAppState(prev_11 => {
409                  if (prev_11.thinkingEnabled === enabled) return prev_11;
410                  return {
411                    ...prev_11,
412                    thinkingEnabled: enabled
413                  };
414                });
415              },
416              onSetPermissionMode(mode) {
417                // Policy guards MUST fire before transitionPermissionMode —
418                // its internal auto-gate check is a defensive throw (with a
419                // setAutoModeActive(true) side-effect BEFORE the throw) rather
420                // than a graceful reject. Letting that throw escape would:
421                // (1) leave STATE.autoModeActive=true while the mode is
422                //     unchanged (3-way invariant violation per src/CLAUDE.md)
423                // (2) fail to send a control_response → server kills WS
424                // These mirror print.ts handleSetPermissionMode; the bridge
425                // can't import the checks directly (bootstrap-isolation), so
426                // it relies on this verdict to emit the error response.
427                if (mode === 'bypassPermissions') {
428                  if (isBypassPermissionsModeDisabled()) {
429                    return {
430                      ok: false,
431                      error: 'Cannot set permission mode to bypassPermissions because it is disabled by settings or configuration'
432                    };
433                  }
434                  if (!store.getState().toolPermissionContext.isBypassPermissionsModeAvailable) {
435                    return {
436                      ok: false,
437                      error: 'Cannot set permission mode to bypassPermissions because the session was not launched with --dangerously-skip-permissions'
438                    };
439                  }
440                }
441                if (feature('TRANSCRIPT_CLASSIFIER') && mode === 'auto' && !isAutoModeGateEnabled()) {
442                  const reason = getAutoModeUnavailableReason();
443                  return {
444                    ok: false,
445                    error: reason ? `Cannot set permission mode to auto: ${getAutoModeUnavailableNotification(reason)}` : 'Cannot set permission mode to auto'
446                  };
447                }
448                // Guards passed — apply via the centralized transition so
449                // prePlanMode stashing and auto-mode state sync all fire.
450                setAppState(prev_12 => {
451                  const current = prev_12.toolPermissionContext.mode;
452                  if (current === mode) return prev_12;
453                  const next = transitionPermissionMode(current, mode, prev_12.toolPermissionContext);
454                  return {
455                    ...prev_12,
456                    toolPermissionContext: {
457                      ...next,
458                      mode
459                    }
460                  };
461                });
462                // Recheck queued permission prompts now that mode changed.
463                setImmediate(() => {
464                  getLeaderToolUseConfirmQueue()?.(currentQueue => {
465                    currentQueue.forEach(item => {
466                      void item.recheckPermission();
467                    });
468                    return currentQueue;
469                  });
470                });
471                return {
472                  ok: true
473                };
474              },
475              onStateChange: handleStateChange,
476              initialMessages: messages.length > 0 ? messages : undefined,
477              getMessages: () => messagesRef.current,
478              previouslyFlushedUUIDs: flushedUUIDsRef.current,
479              initialName: replBridgeInitialName,
480              perpetual
481            });
482            if (cancelled) {
483              // Effect was cancelled while initReplBridge was in flight.
484              // Tear down the handle to avoid leaking resources (poll loop,
485              // WebSocket, registered environment, cleanup callback).
486              logForDebugging(`[bridge:repl] Hook: init cancelled during flight, tearing down${handle_0 ? ` env=${handle_0.environmentId}` : ''}`);
487              if (handle_0) {
488                void handle_0.teardown();
489              }
490              return;
491            }
492            if (!handle_0) {
493              // initReplBridge returned null — a precondition failed. For most
494              // cases (no_oauth, policy_denied, etc.) onStateChange('failed')
495              // already fired with a specific hint. The GrowthBook-gate-off case
496              // is intentionally silent — not a failure, just not rolled out.
497              consecutiveFailuresRef.current++;
498              logForDebugging(`[bridge:repl] Init returned null (precondition or session creation failed); consecutive failures: ${consecutiveFailuresRef.current}`);
499              clearTimeout(failureTimeoutRef.current);
500              setAppState(prev_13 => ({
501                ...prev_13,
502                replBridgeError: prev_13.replBridgeError ?? 'check debug logs for details'
503              }));
504              failureTimeoutRef.current = setTimeout(() => {
505                if (cancelled) return;
506                failureTimeoutRef.current = undefined;
507                setAppState(prev_14 => {
508                  if (!prev_14.replBridgeError) return prev_14;
509                  return {
510                    ...prev_14,
511                    replBridgeEnabled: false,
512                    replBridgeError: undefined
513                  };
514                });
515              }, BRIDGE_FAILURE_DISMISS_MS);
516              return;
517            }
518            handleRef.current = handle_0;
519            setReplBridgeHandle(handle_0);
520            consecutiveFailuresRef.current = 0;
521            // Skip initial messages in the forwarding effect — they were
522            // already loaded as session events during creation.
523            lastWrittenIndexRef.current = initialMessageCount;
524            if (outboundOnly) {
525              setAppState(prev_15 => {
526                if (prev_15.replBridgeConnected && prev_15.replBridgeSessionId === handle_0.bridgeSessionId) return prev_15;
527                return {
528                  ...prev_15,
529                  replBridgeConnected: true,
530                  replBridgeSessionId: handle_0.bridgeSessionId,
531                  replBridgeSessionUrl: undefined,
532                  replBridgeConnectUrl: undefined,
533                  replBridgeError: undefined
534                };
535              });
536              logForDebugging(`[bridge:repl] Mirror initialized, session=${handle_0.bridgeSessionId}`);
537            } else {
538              // Build bridge permission callbacks so the interactive permission
539              // handler can race bridge responses against local user interaction.
540              const permissionCallbacks: BridgePermissionCallbacks = {
541                sendRequest(requestId_0, toolName, input, toolUseId, description, permissionSuggestions, blockedPath) {
542                  handle_0.sendControlRequest({
543                    type: 'control_request',
544                    request_id: requestId_0,
545                    request: {
546                      subtype: 'can_use_tool',
547                      tool_name: toolName,
548                      input,
549                      tool_use_id: toolUseId,
550                      description,
551                      ...(permissionSuggestions ? {
552                        permission_suggestions: permissionSuggestions
553                      } : {}),
554                      ...(blockedPath ? {
555                        blocked_path: blockedPath
556                      } : {})
557                    }
558                  });
559                },
560                sendResponse(requestId_1, response) {
561                  const payload: Record<string, unknown> = {
562                    ...response
563                  };
564                  handle_0.sendControlResponse({
565                    type: 'control_response',
566                    response: {
567                      subtype: 'success',
568                      request_id: requestId_1,
569                      response: payload
570                    }
571                  });
572                },
573                cancelRequest(requestId_2) {
574                  handle_0.sendControlCancelRequest(requestId_2);
575                },
576                onResponse(requestId_3, handler_0) {
577                  pendingPermissionHandlers.set(requestId_3, handler_0);
578                  return () => {
579                    pendingPermissionHandlers.delete(requestId_3);
580                  };
581                }
582              };
583              setAppState(prev_16 => ({
584                ...prev_16,
585                replBridgePermissionCallbacks: permissionCallbacks
586              }));
587              const url = getRemoteSessionUrl(handle_0.bridgeSessionId, handle_0.sessionIngressUrl);
588              // environmentId === '' signals the v2 env-less path. buildBridgeConnectUrl
589              // builds an env-specific connect URL, which doesn't exist without an env.
590              const hasEnv = handle_0.environmentId !== '';
591              const connectUrl_0 = hasEnv ? buildBridgeConnectUrl(handle_0.environmentId, handle_0.sessionIngressUrl) : undefined;
592              setAppState(prev_17 => {
593                if (prev_17.replBridgeConnected && prev_17.replBridgeSessionUrl === url) {
594                  return prev_17;
595                }
596                return {
597                  ...prev_17,
598                  replBridgeConnected: true,
599                  replBridgeSessionUrl: url,
600                  replBridgeConnectUrl: connectUrl_0 ?? prev_17.replBridgeConnectUrl,
601                  replBridgeEnvironmentId: handle_0.environmentId,
602                  replBridgeSessionId: handle_0.bridgeSessionId,
603                  replBridgeError: undefined
604                };
605              });
606  
607              // Show bridge status with URL in the transcript. perpetual (KAIROS
608              // assistant mode) falls back to v1 at initReplBridge.ts — skip the
609              // v2-only upgrade nudge for them. Own try/catch so a cosmetic
610              // GrowthBook hiccup doesn't hit the outer init-failure handler.
611              const upgradeNudge = !perpetual ? await shouldShowAppUpgradeMessage().catch(() => false) : false;
612              if (cancelled) return;
613              setMessages(prev_18 => [...prev_18, createBridgeStatusMessage(url, upgradeNudge ? 'Please upgrade to the latest version of the Claude mobile app to see your Remote Control sessions.' : undefined)]);
614              logForDebugging(`[bridge:repl] Hook initialized, session=${handle_0.bridgeSessionId}`);
615            }
616          } catch (err) {
617            // Never crash the REPL — surface the error in the UI.
618            // Check cancelled first (symmetry with the !handle path at line ~386):
619            // if initReplBridge threw during rapid toggle-off (in-flight network
620            // error), don't count that toward the fuse or spam a stale error
621            // into the UI. Also fixes pre-existing spurious setAppState/
622            // setMessages on cancelled throws.
623            if (cancelled) return;
624            consecutiveFailuresRef.current++;
625            const errMsg = errorMessage(err);
626            logForDebugging(`[bridge:repl] Init failed: ${errMsg}; consecutive failures: ${consecutiveFailuresRef.current}`);
627            clearTimeout(failureTimeoutRef.current);
628            notifyBridgeFailed(errMsg);
629            setAppState(prev_0 => ({
630              ...prev_0,
631              replBridgeError: errMsg
632            }));
633            failureTimeoutRef.current = setTimeout(() => {
634              if (cancelled) return;
635              failureTimeoutRef.current = undefined;
636              setAppState(prev_1 => {
637                if (!prev_1.replBridgeError) return prev_1;
638                return {
639                  ...prev_1,
640                  replBridgeEnabled: false,
641                  replBridgeError: undefined
642                };
643              });
644            }, BRIDGE_FAILURE_DISMISS_MS);
645            if (!outboundOnly) {
646              setMessages(prev_2 => [...prev_2, createSystemMessage(`Remote Control failed to connect: ${errMsg}`, 'warning')]);
647            }
648          }
649        })();
650        return () => {
651          cancelled = true;
652          clearTimeout(failureTimeoutRef.current);
653          failureTimeoutRef.current = undefined;
654          if (handleRef.current) {
655            logForDebugging(`[bridge:repl] Hook cleanup: starting teardown for env=${handleRef.current.environmentId} session=${handleRef.current.bridgeSessionId}`);
656            teardownPromiseRef.current = handleRef.current.teardown();
657            handleRef.current = null;
658            setReplBridgeHandle(null);
659          }
660          setAppState(prev_19 => {
661            if (!prev_19.replBridgeConnected && !prev_19.replBridgeSessionActive && !prev_19.replBridgeError) {
662              return prev_19;
663            }
664            return {
665              ...prev_19,
666              replBridgeConnected: false,
667              replBridgeSessionActive: false,
668              replBridgeReconnecting: false,
669              replBridgeConnectUrl: undefined,
670              replBridgeSessionUrl: undefined,
671              replBridgeEnvironmentId: undefined,
672              replBridgeSessionId: undefined,
673              replBridgeError: undefined,
674              replBridgePermissionCallbacks: undefined
675            };
676          });
677          lastWrittenIndexRef.current = 0;
678        };
679      }
680    }, [replBridgeEnabled, replBridgeOutboundOnly, setAppState, setMessages, addNotification]);
681  
682    // Write new messages as they appear.
683    // Also re-runs when replBridgeConnected changes (bridge finishes init),
684    // so any messages that arrived before the bridge was ready get written.
685    useEffect(() => {
686      // Positive feature() guard — see first useEffect comment
687      if (feature('BRIDGE_MODE')) {
688        if (!replBridgeConnected) return;
689        const handle_1 = handleRef.current;
690        if (!handle_1) return;
691  
692        // Clamp the index in case messages were compacted (array shortened).
693        // After compaction the ref could exceed messages.length, and without
694        // clamping no new messages would be forwarded.
695        if (lastWrittenIndexRef.current > messages.length) {
696          logForDebugging(`[bridge:repl] Compaction detected: lastWrittenIndex=${lastWrittenIndexRef.current} > messages.length=${messages.length}, clamping`);
697        }
698        const startIndex = Math.min(lastWrittenIndexRef.current, messages.length);
699  
700        // Collect new messages since last write
701        const newMessages: Message[] = [];
702        for (let i = startIndex; i < messages.length; i++) {
703          const msg_1 = messages[i];
704          if (msg_1 && (msg_1.type === 'user' || msg_1.type === 'assistant' || msg_1.type === 'system' && msg_1.subtype === 'local_command')) {
705            newMessages.push(msg_1);
706          }
707        }
708        lastWrittenIndexRef.current = messages.length;
709        if (newMessages.length > 0) {
710          handle_1.writeMessages(newMessages);
711        }
712      }
713    }, [messages, replBridgeConnected]);
714    const sendBridgeResult = useCallback(() => {
715      if (feature('BRIDGE_MODE')) {
716        handleRef.current?.sendResult();
717      }
718    }, []);
719    return {
720      sendBridgeResult
721    };
722  }
723  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","React","useCallback","useEffect","useRef","setMainLoopModelOverride","BridgePermissionCallbacks","BridgePermissionResponse","isBridgePermissionResponse","buildBridgeConnectUrl","extractInboundMessageFields","BridgeState","ReplBridgeHandle","setReplBridgeHandle","Command","getSlashCommandToolSkills","isBridgeSafeCommand","getRemoteSessionUrl","useNotifications","PermissionMode","SDKMessage","SDKControlResponse","Text","getFeatureValue_CACHED_MAY_BE_STALE","useAppState","useAppStateStore","useSetAppState","Message","getCwd","logForDebugging","errorMessage","enqueue","buildSystemInitMessage","createBridgeStatusMessage","createSystemMessage","getAutoModeUnavailableNotification","getAutoModeUnavailableReason","isAutoModeGateEnabled","isBypassPermissionsModeDisabled","transitionPermissionMode","getLeaderToolUseConfirmQueue","BRIDGE_FAILURE_DISMISS_MS","MAX_CONSECUTIVE_INIT_FAILURES","useReplBridge","messages","setMessages","action","SetStateAction","abortControllerRef","RefObject","AbortController","commands","mainLoopModel","sendBridgeResult","handleRef","teardownPromiseRef","Promise","undefined","lastWrittenIndexRef","flushedUUIDsRef","Set","failureTimeoutRef","ReturnType","setTimeout","consecutiveFailuresRef","setAppState","commandsRef","current","mainLoopModelRef","messagesRef","store","addNotification","replBridgeEnabled","s","replBridgeConnected","replBridgeOutboundOnly","replBridgeInitialName","outboundOnly","notifyBridgeFailed","detail","key","jsx","priority","fuseHint","prev","replBridgeError","cancelled","initialMessageCount","length","initReplBridge","shouldShowAppUpgradeMessage","perpetual","isAssistantMode","handleInboundMessage","msg","fields","uuid","resolveAndPrepend","sanitized","content","sanitizeInboundWebhookContent","require","preview","slice","value","mode","const","skipSlashCommands","bridgeOrigin","e","level","handleStateChange","state","handle","connectUrl","environmentId","sessionIngressUrl","replBridgeConnectUrl","sessionUrl","bridgeSessionId","replBridgeSessionUrl","envId","sessionId","replBridgeSessionActive","replBridgeReconnecting","replBridgeEnvironmentId","replBridgeSessionId","skills","getState","writeSdkMessages","tools","mcpClients","model","permissionMode","toolPermissionContext","filter","agents","agentDefinitions","activeAgents","plugins","fastMode","err","clearTimeout","pendingPermissionHandlers","Map","response","handlePermissionResponse","requestId","request_id","handler","get","delete","inner","subtype","tags","onInboundMessage","onPermissionResponse","onInterrupt","abort","onSetModel","resolved","mainLoopModelForSession","onSetMaxThinkingTokens","maxTokens","enabled","thinkingEnabled","onSetPermissionMode","ok","error","isBypassPermissionsModeAvailable","reason","next","setImmediate","currentQueue","forEach","item","recheckPermission","onStateChange","initialMessages","getMessages","previouslyFlushedUUIDs","initialName","teardown","permissionCallbacks","sendRequest","toolName","input","toolUseId","description","permissionSuggestions","blockedPath","sendControlRequest","type","request","tool_name","tool_use_id","permission_suggestions","blocked_path","sendResponse","payload","Record","sendControlResponse","cancelRequest","sendControlCancelRequest","onResponse","set","replBridgePermissionCallbacks","url","hasEnv","upgradeNudge","catch","errMsg","startIndex","Math","min","newMessages","i","push","writeMessages","sendResult"],"sources":["useReplBridge.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport React, { useCallback, useEffect, useRef } from 'react'\nimport { setMainLoopModelOverride } from '../bootstrap/state.js'\nimport {\n  type BridgePermissionCallbacks,\n  type BridgePermissionResponse,\n  isBridgePermissionResponse,\n} from '../bridge/bridgePermissionCallbacks.js'\nimport { buildBridgeConnectUrl } from '../bridge/bridgeStatusUtil.js'\nimport { extractInboundMessageFields } from '../bridge/inboundMessages.js'\nimport type { BridgeState, ReplBridgeHandle } from '../bridge/replBridge.js'\nimport { setReplBridgeHandle } from '../bridge/replBridgeHandle.js'\nimport type { Command } from '../commands.js'\nimport { getSlashCommandToolSkills, isBridgeSafeCommand } from '../commands.js'\nimport { getRemoteSessionUrl } from '../constants/product.js'\nimport { useNotifications } from '../context/notifications.js'\nimport type {\n  PermissionMode,\n  SDKMessage,\n} from '../entrypoints/agentSdkTypes.js'\nimport type { SDKControlResponse } from '../entrypoints/sdk/controlTypes.js'\nimport { Text } from '../ink.js'\nimport { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js'\nimport {\n  useAppState,\n  useAppStateStore,\n  useSetAppState,\n} from '../state/AppState.js'\nimport type { Message } from '../types/message.js'\nimport { getCwd } from '../utils/cwd.js'\nimport { logForDebugging } from '../utils/debug.js'\nimport { errorMessage } from '../utils/errors.js'\nimport { enqueue } from '../utils/messageQueueManager.js'\nimport { buildSystemInitMessage } from '../utils/messages/systemInit.js'\nimport {\n  createBridgeStatusMessage,\n  createSystemMessage,\n} from '../utils/messages.js'\nimport {\n  getAutoModeUnavailableNotification,\n  getAutoModeUnavailableReason,\n  isAutoModeGateEnabled,\n  isBypassPermissionsModeDisabled,\n  transitionPermissionMode,\n} from '../utils/permissions/permissionSetup.js'\nimport { getLeaderToolUseConfirmQueue } from '../utils/swarm/leaderPermissionBridge.js'\n\n/** How long after a failure before replBridgeEnabled is auto-cleared (stops retries). */\nexport const BRIDGE_FAILURE_DISMISS_MS = 10_000\n\n/**\n * Max consecutive initReplBridge failures before the hook stops re-attempting\n * for the session lifetime. Guards against paths that flip replBridgeEnabled\n * back on after auto-disable (settings sync, /remote-control, config tool)\n * when the underlying OAuth is unrecoverable — each re-attempt is another\n * guaranteed 401 against POST /v1/environments/bridge. Datadog 2026-03-08:\n * top stuck client generated 2,879 × 401/day alone (17% of all 401s on the\n * route).\n */\nconst MAX_CONSECUTIVE_INIT_FAILURES = 3\n\n/**\n * Hook that initializes an always-on bridge connection in the background\n * and writes new user/assistant messages to the bridge session.\n *\n * Silently skips if bridge is not enabled or user is not OAuth-authenticated.\n *\n * Watches AppState.replBridgeEnabled — when toggled off (via /config or footer),\n * the bridge is torn down. When toggled back on, it re-initializes.\n *\n * Inbound messages from claude.ai are injected into the REPL via queuedCommands.\n */\nexport function useReplBridge(\n  messages: Message[],\n  setMessages: (action: React.SetStateAction<Message[]>) => void,\n  abortControllerRef: React.RefObject<AbortController | null>,\n  commands: readonly Command[],\n  mainLoopModel: string,\n): { sendBridgeResult: () => void } {\n  const handleRef = useRef<ReplBridgeHandle | null>(null)\n  const teardownPromiseRef = useRef<Promise<void> | undefined>(undefined)\n  const lastWrittenIndexRef = useRef(0)\n  // Tracks UUIDs already flushed as initial messages. Persists across\n  // bridge reconnections so Bridge #2+ only sends new messages — sending\n  // duplicate UUIDs causes the server to kill the WebSocket.\n  const flushedUUIDsRef = useRef(new Set<string>())\n  const failureTimeoutRef = useRef<ReturnType<typeof setTimeout> | undefined>(\n    undefined,\n  )\n  // Persists across effect re-runs (unlike the effect's local state). Reset\n  // only on successful init. Hits MAX_CONSECUTIVE_INIT_FAILURES → fuse blown\n  // for the session, regardless of replBridgeEnabled re-toggling.\n  const consecutiveFailuresRef = useRef(0)\n  const setAppState = useSetAppState()\n  const commandsRef = useRef(commands)\n  commandsRef.current = commands\n  const mainLoopModelRef = useRef(mainLoopModel)\n  mainLoopModelRef.current = mainLoopModel\n  const messagesRef = useRef(messages)\n  messagesRef.current = messages\n  const store = useAppStateStore()\n  const { addNotification } = useNotifications()\n  const replBridgeEnabled = feature('BRIDGE_MODE')\n    ? // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant\n      useAppState(s => s.replBridgeEnabled)\n    : false\n  const replBridgeConnected = feature('BRIDGE_MODE')\n    ? // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant\n      useAppState(s => s.replBridgeConnected)\n    : false\n  const replBridgeOutboundOnly = feature('BRIDGE_MODE')\n    ? // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant\n      useAppState(s => s.replBridgeOutboundOnly)\n    : false\n  const replBridgeInitialName = feature('BRIDGE_MODE')\n    ? // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant\n      useAppState(s => s.replBridgeInitialName)\n    : undefined\n\n  // Initialize/teardown bridge when enabled state changes.\n  // Passes current messages as initialMessages so the remote session\n  // starts with the existing conversation context (e.g. from /bridge).\n  useEffect(() => {\n    // feature() check must use positive pattern for dead code elimination —\n    // negative pattern (if (!feature(...)) return) does NOT eliminate\n    // dynamic imports below.\n    if (feature('BRIDGE_MODE')) {\n      if (!replBridgeEnabled) return\n\n      const outboundOnly = replBridgeOutboundOnly\n      function notifyBridgeFailed(detail?: string): void {\n        if (outboundOnly) return\n        addNotification({\n          key: 'bridge-failed',\n          jsx: (\n            <>\n              <Text color=\"error\">Remote Control failed</Text>\n              {detail && <Text dimColor> · {detail}</Text>}\n            </>\n          ),\n          priority: 'immediate',\n        })\n      }\n\n      if (consecutiveFailuresRef.current >= MAX_CONSECUTIVE_INIT_FAILURES) {\n        logForDebugging(\n          `[bridge:repl] Hook: ${consecutiveFailuresRef.current} consecutive init failures, not retrying this session`,\n        )\n        // Clear replBridgeEnabled so /remote-control doesn't mistakenly show\n        // BridgeDisconnectDialog for a bridge that never connected.\n        const fuseHint = 'disabled after repeated failures · restart to retry'\n        notifyBridgeFailed(fuseHint)\n        setAppState(prev => {\n          if (prev.replBridgeError === fuseHint && !prev.replBridgeEnabled)\n            return prev\n          return {\n            ...prev,\n            replBridgeError: fuseHint,\n            replBridgeEnabled: false,\n          }\n        })\n        return\n      }\n\n      let cancelled = false\n      // Capture messages.length now so we don't re-send initial messages\n      // through writeMessages after the bridge connects.\n      const initialMessageCount = messages.length\n\n      void (async () => {\n        try {\n          // Wait for any in-progress teardown to complete before registering\n          // a new environment. Without this, the deregister HTTP call from\n          // the previous teardown races with the new register call, and the\n          // server may tear down the freshly-created environment.\n          if (teardownPromiseRef.current) {\n            logForDebugging(\n              '[bridge:repl] Hook: waiting for previous teardown to complete before re-init',\n            )\n            await teardownPromiseRef.current\n            teardownPromiseRef.current = undefined\n            logForDebugging(\n              '[bridge:repl] Hook: previous teardown complete, proceeding with re-init',\n            )\n          }\n          if (cancelled) return\n\n          // Dynamic import so the module is tree-shaken in external builds\n          const { initReplBridge } = await import('../bridge/initReplBridge.js')\n          const { shouldShowAppUpgradeMessage } = await import(\n            '../bridge/envLessBridgeConfig.js'\n          )\n\n          // Assistant mode: perpetual bridge session — claude.ai shows one\n          // continuous conversation across CLI restarts instead of a new\n          // session per invocation. initBridgeCore reads bridge-pointer.json\n          // (the same crash-recovery file #20735 added) and reuses its\n          // {environmentId, sessionId} via reuseEnvironmentId +\n          // api.reconnectSession(). Teardown skips archive/deregister/\n          // pointer-clear so the session survives clean exits, not just\n          // crashes. Non-assistant bridges clear the pointer on teardown\n          // (crash-recovery only).\n          let perpetual = false\n          if (feature('KAIROS')) {\n            const { isAssistantMode } = await import('../assistant/index.js')\n            perpetual = isAssistantMode()\n          }\n\n          // When a user message arrives from claude.ai, inject it into the REPL.\n          // Preserves the original UUID so that when the message is forwarded\n          // back to CCR, it matches the original — avoiding duplicate messages.\n          //\n          // Async because file_attachments (if present) need a network fetch +\n          // disk write before we enqueue with the @path prefix. Caller doesn't\n          // await — messages with attachments just land in the queue slightly\n          // later, which is fine (web messages aren't rapid-fire).\n          async function handleInboundMessage(msg: SDKMessage): Promise<void> {\n            try {\n              const fields = extractInboundMessageFields(msg)\n              if (!fields) return\n\n              const { uuid } = fields\n\n              // Dynamic import keeps the bridge code out of non-BRIDGE_MODE builds.\n              const { resolveAndPrepend } = await import(\n                '../bridge/inboundAttachments.js'\n              )\n              let sanitized = fields.content\n              if (feature('KAIROS_GITHUB_WEBHOOKS')) {\n                /* eslint-disable @typescript-eslint/no-require-imports */\n                const { sanitizeInboundWebhookContent } =\n                  require('../bridge/webhookSanitizer.js') as typeof import('../bridge/webhookSanitizer.js')\n                /* eslint-enable @typescript-eslint/no-require-imports */\n                sanitized = sanitizeInboundWebhookContent(fields.content)\n              }\n              const content = await resolveAndPrepend(msg, sanitized)\n\n              const preview =\n                typeof content === 'string'\n                  ? content.slice(0, 80)\n                  : `[${content.length} content blocks]`\n              logForDebugging(\n                `[bridge:repl] Injecting inbound user message: ${preview}${uuid ? ` uuid=${uuid}` : ''}`,\n              )\n              enqueue({\n                value: content,\n                mode: 'prompt' as const,\n                uuid,\n                // skipSlashCommands stays true as defense-in-depth —\n                // processUserInputBase overrides it internally when bridgeOrigin\n                // is set AND the resolved command passes isBridgeSafeCommand.\n                // This keeps exit-word suppression and immediate-command blocks\n                // intact for any code path that checks skipSlashCommands directly.\n                skipSlashCommands: true,\n                bridgeOrigin: true,\n              })\n            } catch (e) {\n              logForDebugging(\n                `[bridge:repl] handleInboundMessage failed: ${e}`,\n                { level: 'error' },\n              )\n            }\n          }\n\n          // State change callback — maps bridge lifecycle events to AppState.\n          function handleStateChange(\n            state: BridgeState,\n            detail?: string,\n          ): void {\n            if (cancelled) return\n            if (outboundOnly) {\n              logForDebugging(\n                `[bridge:repl] Mirror state=${state}${detail ? ` detail=${detail}` : ''}`,\n              )\n              // Sync replBridgeConnected so the forwarding effect starts/stops\n              // writing as the transport comes up or dies.\n              if (state === 'failed') {\n                setAppState(prev => {\n                  if (!prev.replBridgeConnected) return prev\n                  return { ...prev, replBridgeConnected: false }\n                })\n              } else if (state === 'ready' || state === 'connected') {\n                setAppState(prev => {\n                  if (prev.replBridgeConnected) return prev\n                  return { ...prev, replBridgeConnected: true }\n                })\n              }\n              return\n            }\n            const handle = handleRef.current\n            switch (state) {\n              case 'ready':\n                setAppState(prev => {\n                  const connectUrl =\n                    handle && handle.environmentId !== ''\n                      ? buildBridgeConnectUrl(\n                          handle.environmentId,\n                          handle.sessionIngressUrl,\n                        )\n                      : prev.replBridgeConnectUrl\n                  const sessionUrl = handle\n                    ? getRemoteSessionUrl(\n                        handle.bridgeSessionId,\n                        handle.sessionIngressUrl,\n                      )\n                    : prev.replBridgeSessionUrl\n                  const envId = handle?.environmentId\n                  const sessionId = handle?.bridgeSessionId\n                  if (\n                    prev.replBridgeConnected &&\n                    !prev.replBridgeSessionActive &&\n                    !prev.replBridgeReconnecting &&\n                    prev.replBridgeConnectUrl === connectUrl &&\n                    prev.replBridgeSessionUrl === sessionUrl &&\n                    prev.replBridgeEnvironmentId === envId &&\n                    prev.replBridgeSessionId === sessionId\n                  ) {\n                    return prev\n                  }\n                  return {\n                    ...prev,\n                    replBridgeConnected: true,\n                    replBridgeSessionActive: false,\n                    replBridgeReconnecting: false,\n                    replBridgeConnectUrl: connectUrl,\n                    replBridgeSessionUrl: sessionUrl,\n                    replBridgeEnvironmentId: envId,\n                    replBridgeSessionId: sessionId,\n                    replBridgeError: undefined,\n                  }\n                })\n                break\n              case 'connected': {\n                setAppState(prev => {\n                  if (prev.replBridgeSessionActive) return prev\n                  return {\n                    ...prev,\n                    replBridgeConnected: true,\n                    replBridgeSessionActive: true,\n                    replBridgeReconnecting: false,\n                    replBridgeError: undefined,\n                  }\n                })\n                // Send system/init so remote clients (web/iOS/Android) get\n                // session metadata. REPL uses query() directly — never hits\n                // QueryEngine's SDKMessage layer — so this is the only path\n                // to put system/init on the REPL-bridge wire. Skills load is\n                // async (memoized, cheap after REPL startup); fire-and-forget\n                // so the connected-state transition isn't blocked.\n                if (\n                  getFeatureValue_CACHED_MAY_BE_STALE(\n                    'tengu_bridge_system_init',\n                    false,\n                  )\n                ) {\n                  void (async () => {\n                    try {\n                      const skills = await getSlashCommandToolSkills(getCwd())\n                      if (cancelled) return\n                      const state = store.getState()\n                      handleRef.current?.writeSdkMessages([\n                        buildSystemInitMessage({\n                          // tools/mcpClients/plugins redacted for REPL-bridge:\n                          // MCP-prefixed tool names and server names leak which\n                          // integrations the user has wired up; plugin paths leak\n                          // raw filesystem paths (username, project structure).\n                          // CCR v2 persists SDK messages to Spanner — users who\n                          // tap \"Connect from phone\" may not expect these on\n                          // Anthropic's servers. QueryEngine (SDK) still emits\n                          // full lists — SDK consumers expect full telemetry.\n                          tools: [],\n                          mcpClients: [],\n                          model: mainLoopModelRef.current,\n                          permissionMode: state.toolPermissionContext\n                            .mode as PermissionMode, // TODO: avoid the cast\n                          // Remote clients can only invoke bridge-safe commands —\n                          // advertising unsafe ones (local-jsx, unallowed local)\n                          // would let mobile/web attempt them and hit errors.\n                          commands:\n                            commandsRef.current.filter(isBridgeSafeCommand),\n                          agents: state.agentDefinitions.activeAgents,\n                          skills,\n                          plugins: [],\n                          fastMode: state.fastMode,\n                        }),\n                      ])\n                    } catch (err) {\n                      logForDebugging(\n                        `[bridge:repl] Failed to send system/init: ${errorMessage(err)}`,\n                        { level: 'error' },\n                      )\n                    }\n                  })()\n                }\n                break\n              }\n              case 'reconnecting':\n                setAppState(prev => {\n                  if (prev.replBridgeReconnecting) return prev\n                  return {\n                    ...prev,\n                    replBridgeReconnecting: true,\n                    replBridgeSessionActive: false,\n                  }\n                })\n                break\n              case 'failed':\n                // Clear any previous failure dismiss timer\n                clearTimeout(failureTimeoutRef.current)\n                notifyBridgeFailed(detail)\n                setAppState(prev => ({\n                  ...prev,\n                  replBridgeError: detail,\n                  replBridgeReconnecting: false,\n                  replBridgeSessionActive: false,\n                  replBridgeConnected: false,\n                }))\n                // Auto-disable after timeout so the hook stops retrying.\n                failureTimeoutRef.current = setTimeout(() => {\n                  if (cancelled) return\n                  failureTimeoutRef.current = undefined\n                  setAppState(prev => {\n                    if (!prev.replBridgeError) return prev\n                    return {\n                      ...prev,\n                      replBridgeEnabled: false,\n                      replBridgeError: undefined,\n                    }\n                  })\n                }, BRIDGE_FAILURE_DISMISS_MS)\n                break\n            }\n          }\n\n          // Map of pending bridge permission response handlers, keyed by request_id.\n          // Each entry is an onResponse handler waiting for CCR to reply.\n          const pendingPermissionHandlers = new Map<\n            string,\n            (response: BridgePermissionResponse) => void\n          >()\n\n          // Dispatch incoming control_response messages to registered handlers\n          function handlePermissionResponse(msg: SDKControlResponse): void {\n            const requestId = msg.response?.request_id\n            if (!requestId) return\n            const handler = pendingPermissionHandlers.get(requestId)\n            if (!handler) {\n              logForDebugging(\n                `[bridge:repl] No handler for control_response request_id=${requestId}`,\n              )\n              return\n            }\n            pendingPermissionHandlers.delete(requestId)\n            // Extract the permission decision from the control_response payload\n            const inner = msg.response\n            if (\n              inner.subtype === 'success' &&\n              inner.response &&\n              isBridgePermissionResponse(inner.response)\n            ) {\n              handler(inner.response)\n            }\n          }\n\n          const handle = await initReplBridge({\n            outboundOnly,\n            tags: outboundOnly ? ['ccr-mirror'] : undefined,\n            onInboundMessage: handleInboundMessage,\n            onPermissionResponse: handlePermissionResponse,\n            onInterrupt() {\n              abortControllerRef.current?.abort()\n            },\n            onSetModel(model) {\n              const resolved = model === 'default' ? null : (model ?? null)\n              setMainLoopModelOverride(resolved)\n              setAppState(prev => {\n                if (prev.mainLoopModelForSession === resolved) return prev\n                return { ...prev, mainLoopModelForSession: resolved }\n              })\n            },\n            onSetMaxThinkingTokens(maxTokens) {\n              const enabled = maxTokens !== null\n              setAppState(prev => {\n                if (prev.thinkingEnabled === enabled) return prev\n                return { ...prev, thinkingEnabled: enabled }\n              })\n            },\n            onSetPermissionMode(mode) {\n              // Policy guards MUST fire before transitionPermissionMode —\n              // its internal auto-gate check is a defensive throw (with a\n              // setAutoModeActive(true) side-effect BEFORE the throw) rather\n              // than a graceful reject. Letting that throw escape would:\n              // (1) leave STATE.autoModeActive=true while the mode is\n              //     unchanged (3-way invariant violation per src/CLAUDE.md)\n              // (2) fail to send a control_response → server kills WS\n              // These mirror print.ts handleSetPermissionMode; the bridge\n              // can't import the checks directly (bootstrap-isolation), so\n              // it relies on this verdict to emit the error response.\n              if (mode === 'bypassPermissions') {\n                if (isBypassPermissionsModeDisabled()) {\n                  return {\n                    ok: false,\n                    error:\n                      'Cannot set permission mode to bypassPermissions because it is disabled by settings or configuration',\n                  }\n                }\n                if (\n                  !store.getState().toolPermissionContext\n                    .isBypassPermissionsModeAvailable\n                ) {\n                  return {\n                    ok: false,\n                    error:\n                      'Cannot set permission mode to bypassPermissions because the session was not launched with --dangerously-skip-permissions',\n                  }\n                }\n              }\n              if (\n                feature('TRANSCRIPT_CLASSIFIER') &&\n                mode === 'auto' &&\n                !isAutoModeGateEnabled()\n              ) {\n                const reason = getAutoModeUnavailableReason()\n                return {\n                  ok: false,\n                  error: reason\n                    ? `Cannot set permission mode to auto: ${getAutoModeUnavailableNotification(reason)}`\n                    : 'Cannot set permission mode to auto',\n                }\n              }\n              // Guards passed — apply via the centralized transition so\n              // prePlanMode stashing and auto-mode state sync all fire.\n              setAppState(prev => {\n                const current = prev.toolPermissionContext.mode\n                if (current === mode) return prev\n                const next = transitionPermissionMode(\n                  current,\n                  mode,\n                  prev.toolPermissionContext,\n                )\n                return {\n                  ...prev,\n                  toolPermissionContext: { ...next, mode },\n                }\n              })\n              // Recheck queued permission prompts now that mode changed.\n              setImmediate(() => {\n                getLeaderToolUseConfirmQueue()?.(currentQueue => {\n                  currentQueue.forEach(item => {\n                    void item.recheckPermission()\n                  })\n                  return currentQueue\n                })\n              })\n              return { ok: true }\n            },\n            onStateChange: handleStateChange,\n            initialMessages: messages.length > 0 ? messages : undefined,\n            getMessages: () => messagesRef.current,\n            previouslyFlushedUUIDs: flushedUUIDsRef.current,\n            initialName: replBridgeInitialName,\n            perpetual,\n          })\n          if (cancelled) {\n            // Effect was cancelled while initReplBridge was in flight.\n            // Tear down the handle to avoid leaking resources (poll loop,\n            // WebSocket, registered environment, cleanup callback).\n            logForDebugging(\n              `[bridge:repl] Hook: init cancelled during flight, tearing down${handle ? ` env=${handle.environmentId}` : ''}`,\n            )\n            if (handle) {\n              void handle.teardown()\n            }\n            return\n          }\n          if (!handle) {\n            // initReplBridge returned null — a precondition failed. For most\n            // cases (no_oauth, policy_denied, etc.) onStateChange('failed')\n            // already fired with a specific hint. The GrowthBook-gate-off case\n            // is intentionally silent — not a failure, just not rolled out.\n            consecutiveFailuresRef.current++\n            logForDebugging(\n              `[bridge:repl] Init returned null (precondition or session creation failed); consecutive failures: ${consecutiveFailuresRef.current}`,\n            )\n            clearTimeout(failureTimeoutRef.current)\n            setAppState(prev => ({\n              ...prev,\n              replBridgeError:\n                prev.replBridgeError ?? 'check debug logs for details',\n            }))\n            failureTimeoutRef.current = setTimeout(() => {\n              if (cancelled) return\n              failureTimeoutRef.current = undefined\n              setAppState(prev => {\n                if (!prev.replBridgeError) return prev\n                return {\n                  ...prev,\n                  replBridgeEnabled: false,\n                  replBridgeError: undefined,\n                }\n              })\n            }, BRIDGE_FAILURE_DISMISS_MS)\n            return\n          }\n          handleRef.current = handle\n          setReplBridgeHandle(handle)\n          consecutiveFailuresRef.current = 0\n          // Skip initial messages in the forwarding effect — they were\n          // already loaded as session events during creation.\n          lastWrittenIndexRef.current = initialMessageCount\n\n          if (outboundOnly) {\n            setAppState(prev => {\n              if (\n                prev.replBridgeConnected &&\n                prev.replBridgeSessionId === handle.bridgeSessionId\n              )\n                return prev\n              return {\n                ...prev,\n                replBridgeConnected: true,\n                replBridgeSessionId: handle.bridgeSessionId,\n                replBridgeSessionUrl: undefined,\n                replBridgeConnectUrl: undefined,\n                replBridgeError: undefined,\n              }\n            })\n            logForDebugging(\n              `[bridge:repl] Mirror initialized, session=${handle.bridgeSessionId}`,\n            )\n          } else {\n            // Build bridge permission callbacks so the interactive permission\n            // handler can race bridge responses against local user interaction.\n            const permissionCallbacks: BridgePermissionCallbacks = {\n              sendRequest(\n                requestId,\n                toolName,\n                input,\n                toolUseId,\n                description,\n                permissionSuggestions,\n                blockedPath,\n              ) {\n                handle.sendControlRequest({\n                  type: 'control_request',\n                  request_id: requestId,\n                  request: {\n                    subtype: 'can_use_tool',\n                    tool_name: toolName,\n                    input,\n                    tool_use_id: toolUseId,\n                    description,\n                    ...(permissionSuggestions\n                      ? { permission_suggestions: permissionSuggestions }\n                      : {}),\n                    ...(blockedPath ? { blocked_path: blockedPath } : {}),\n                  },\n                })\n              },\n              sendResponse(requestId, response) {\n                const payload: Record<string, unknown> = { ...response }\n                handle.sendControlResponse({\n                  type: 'control_response',\n                  response: {\n                    subtype: 'success',\n                    request_id: requestId,\n                    response: payload,\n                  },\n                })\n              },\n              cancelRequest(requestId) {\n                handle.sendControlCancelRequest(requestId)\n              },\n              onResponse(requestId, handler) {\n                pendingPermissionHandlers.set(requestId, handler)\n                return () => {\n                  pendingPermissionHandlers.delete(requestId)\n                }\n              },\n            }\n            setAppState(prev => ({\n              ...prev,\n              replBridgePermissionCallbacks: permissionCallbacks,\n            }))\n            const url = getRemoteSessionUrl(\n              handle.bridgeSessionId,\n              handle.sessionIngressUrl,\n            )\n            // environmentId === '' signals the v2 env-less path. buildBridgeConnectUrl\n            // builds an env-specific connect URL, which doesn't exist without an env.\n            const hasEnv = handle.environmentId !== ''\n            const connectUrl = hasEnv\n              ? buildBridgeConnectUrl(\n                  handle.environmentId,\n                  handle.sessionIngressUrl,\n                )\n              : undefined\n            setAppState(prev => {\n              if (\n                prev.replBridgeConnected &&\n                prev.replBridgeSessionUrl === url\n              ) {\n                return prev\n              }\n              return {\n                ...prev,\n                replBridgeConnected: true,\n                replBridgeSessionUrl: url,\n                replBridgeConnectUrl: connectUrl ?? prev.replBridgeConnectUrl,\n                replBridgeEnvironmentId: handle.environmentId,\n                replBridgeSessionId: handle.bridgeSessionId,\n                replBridgeError: undefined,\n              }\n            })\n\n            // Show bridge status with URL in the transcript. perpetual (KAIROS\n            // assistant mode) falls back to v1 at initReplBridge.ts — skip the\n            // v2-only upgrade nudge for them. Own try/catch so a cosmetic\n            // GrowthBook hiccup doesn't hit the outer init-failure handler.\n            const upgradeNudge = !perpetual\n              ? await shouldShowAppUpgradeMessage().catch(() => false)\n              : false\n            if (cancelled) return\n            setMessages(prev => [\n              ...prev,\n              createBridgeStatusMessage(\n                url,\n                upgradeNudge\n                  ? 'Please upgrade to the latest version of the Claude mobile app to see your Remote Control sessions.'\n                  : undefined,\n              ),\n            ])\n\n            logForDebugging(\n              `[bridge:repl] Hook initialized, session=${handle.bridgeSessionId}`,\n            )\n          }\n        } catch (err) {\n          // Never crash the REPL — surface the error in the UI.\n          // Check cancelled first (symmetry with the !handle path at line ~386):\n          // if initReplBridge threw during rapid toggle-off (in-flight network\n          // error), don't count that toward the fuse or spam a stale error\n          // into the UI. Also fixes pre-existing spurious setAppState/\n          // setMessages on cancelled throws.\n          if (cancelled) return\n          consecutiveFailuresRef.current++\n          const errMsg = errorMessage(err)\n          logForDebugging(\n            `[bridge:repl] Init failed: ${errMsg}; consecutive failures: ${consecutiveFailuresRef.current}`,\n          )\n          clearTimeout(failureTimeoutRef.current)\n          notifyBridgeFailed(errMsg)\n          setAppState(prev => ({\n            ...prev,\n            replBridgeError: errMsg,\n          }))\n          failureTimeoutRef.current = setTimeout(() => {\n            if (cancelled) return\n            failureTimeoutRef.current = undefined\n            setAppState(prev => {\n              if (!prev.replBridgeError) return prev\n              return {\n                ...prev,\n                replBridgeEnabled: false,\n                replBridgeError: undefined,\n              }\n            })\n          }, BRIDGE_FAILURE_DISMISS_MS)\n          if (!outboundOnly) {\n            setMessages(prev => [\n              ...prev,\n              createSystemMessage(\n                `Remote Control failed to connect: ${errMsg}`,\n                'warning',\n              ),\n            ])\n          }\n        }\n      })()\n\n      return () => {\n        cancelled = true\n        clearTimeout(failureTimeoutRef.current)\n        failureTimeoutRef.current = undefined\n        if (handleRef.current) {\n          logForDebugging(\n            `[bridge:repl] Hook cleanup: starting teardown for env=${handleRef.current.environmentId} session=${handleRef.current.bridgeSessionId}`,\n          )\n          teardownPromiseRef.current = handleRef.current.teardown()\n          handleRef.current = null\n          setReplBridgeHandle(null)\n        }\n        setAppState(prev => {\n          if (\n            !prev.replBridgeConnected &&\n            !prev.replBridgeSessionActive &&\n            !prev.replBridgeError\n          ) {\n            return prev\n          }\n          return {\n            ...prev,\n            replBridgeConnected: false,\n            replBridgeSessionActive: false,\n            replBridgeReconnecting: false,\n            replBridgeConnectUrl: undefined,\n            replBridgeSessionUrl: undefined,\n            replBridgeEnvironmentId: undefined,\n            replBridgeSessionId: undefined,\n            replBridgeError: undefined,\n            replBridgePermissionCallbacks: undefined,\n          }\n        })\n        lastWrittenIndexRef.current = 0\n      }\n    }\n  }, [\n    replBridgeEnabled,\n    replBridgeOutboundOnly,\n    setAppState,\n    setMessages,\n    addNotification,\n  ])\n\n  // Write new messages as they appear.\n  // Also re-runs when replBridgeConnected changes (bridge finishes init),\n  // so any messages that arrived before the bridge was ready get written.\n  useEffect(() => {\n    // Positive feature() guard — see first useEffect comment\n    if (feature('BRIDGE_MODE')) {\n      if (!replBridgeConnected) return\n\n      const handle = handleRef.current\n      if (!handle) return\n\n      // Clamp the index in case messages were compacted (array shortened).\n      // After compaction the ref could exceed messages.length, and without\n      // clamping no new messages would be forwarded.\n      if (lastWrittenIndexRef.current > messages.length) {\n        logForDebugging(\n          `[bridge:repl] Compaction detected: lastWrittenIndex=${lastWrittenIndexRef.current} > messages.length=${messages.length}, clamping`,\n        )\n      }\n      const startIndex = Math.min(lastWrittenIndexRef.current, messages.length)\n\n      // Collect new messages since last write\n      const newMessages: Message[] = []\n      for (let i = startIndex; i < messages.length; i++) {\n        const msg = messages[i]\n        if (\n          msg &&\n          (msg.type === 'user' ||\n            msg.type === 'assistant' ||\n            (msg.type === 'system' && msg.subtype === 'local_command'))\n        ) {\n          newMessages.push(msg)\n        }\n      }\n      lastWrittenIndexRef.current = messages.length\n\n      if (newMessages.length > 0) {\n        handle.writeMessages(newMessages)\n      }\n    }\n  }, [messages, replBridgeConnected])\n\n  const sendBridgeResult = useCallback(() => {\n    if (feature('BRIDGE_MODE')) {\n      handleRef.current?.sendResult()\n    }\n  }, [])\n\n  return { sendBridgeResult }\n}\n"],"mappings":"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","ignoreList":[]}