/ tools / BashTool / BashTool.tsx
BashTool.tsx
   1  import { feature } from 'bun:bundle';
   2  import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs';
   3  import { copyFile, stat as fsStat, truncate as fsTruncate, link } from 'fs/promises';
   4  import * as React from 'react';
   5  import type { CanUseToolFn } from 'src/hooks/useCanUseTool.js';
   6  import type { AppState } from 'src/state/AppState.js';
   7  import { z } from 'zod/v4';
   8  import { getKairosActive } from '../../bootstrap/state.js';
   9  import { TOOL_SUMMARY_MAX_LENGTH } from '../../constants/toolLimits.js';
  10  import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js';
  11  import { notifyVscodeFileUpdated } from '../../services/mcp/vscodeSdkMcp.js';
  12  import type { SetToolJSXFn, ToolCallProgress, ToolUseContext, ValidationResult } from '../../Tool.js';
  13  import { buildTool, type ToolDef } from '../../Tool.js';
  14  import { backgroundExistingForegroundTask, markTaskNotified, registerForeground, spawnShellTask, unregisterForeground } from '../../tasks/LocalShellTask/LocalShellTask.js';
  15  import type { AgentId } from '../../types/ids.js';
  16  import type { AssistantMessage } from '../../types/message.js';
  17  import { parseForSecurity } from '../../utils/bash/ast.js';
  18  import { splitCommand_DEPRECATED, splitCommandWithOperators } from '../../utils/bash/commands.js';
  19  import { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js';
  20  import { detectCodeIndexingFromCommand } from '../../utils/codeIndexing.js';
  21  import { isEnvTruthy } from '../../utils/envUtils.js';
  22  import { isENOENT, ShellError } from '../../utils/errors.js';
  23  import { detectFileEncoding, detectLineEndings, getFileModificationTime, writeTextContent } from '../../utils/file.js';
  24  import { fileHistoryEnabled, fileHistoryTrackEdit } from '../../utils/fileHistory.js';
  25  import { truncate } from '../../utils/format.js';
  26  import { getFsImplementation } from '../../utils/fsOperations.js';
  27  import { lazySchema } from '../../utils/lazySchema.js';
  28  import { expandPath } from '../../utils/path.js';
  29  import type { PermissionResult } from '../../utils/permissions/PermissionResult.js';
  30  import { maybeRecordPluginHint } from '../../utils/plugins/hintRecommendation.js';
  31  import { exec } from '../../utils/Shell.js';
  32  import type { ExecResult } from '../../utils/ShellCommand.js';
  33  import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js';
  34  import { semanticBoolean } from '../../utils/semanticBoolean.js';
  35  import { semanticNumber } from '../../utils/semanticNumber.js';
  36  import { EndTruncatingAccumulator } from '../../utils/stringUtils.js';
  37  import { getTaskOutputPath } from '../../utils/task/diskOutput.js';
  38  import { TaskOutput } from '../../utils/task/TaskOutput.js';
  39  import { isOutputLineTruncated } from '../../utils/terminal.js';
  40  import { buildLargeToolResultMessage, ensureToolResultsDir, generatePreview, getToolResultPath, PREVIEW_SIZE_BYTES } from '../../utils/toolResultStorage.js';
  41  import { userFacingName as fileEditUserFacingName } from '../FileEditTool/UI.js';
  42  import { trackGitOperations } from '../shared/gitOperationTracking.js';
  43  import { bashToolHasPermission, commandHasAnyCd, matchWildcardPattern, permissionRuleExtractPrefix } from './bashPermissions.js';
  44  import { interpretCommandResult } from './commandSemantics.js';
  45  import { getDefaultTimeoutMs, getMaxTimeoutMs, getSimplePrompt } from './prompt.js';
  46  import { checkReadOnlyConstraints } from './readOnlyValidation.js';
  47  import { parseSedEditCommand } from './sedEditParser.js';
  48  import { shouldUseSandbox } from './shouldUseSandbox.js';
  49  import { BASH_TOOL_NAME } from './toolName.js';
  50  import { BackgroundHint, renderToolResultMessage, renderToolUseErrorMessage, renderToolUseMessage, renderToolUseProgressMessage, renderToolUseQueuedMessage } from './UI.js';
  51  import { buildImageToolResult, isImageOutput, resetCwdIfOutsideProject, resizeShellImageOutput, stdErrAppendShellResetMessage, stripEmptyLines } from './utils.js';
  52  const EOL = '\n';
  53  
  54  // Progress display constants
  55  const PROGRESS_THRESHOLD_MS = 2000; // Show progress after 2 seconds
  56  // In assistant mode, blocking bash auto-backgrounds after this many ms in the main agent
  57  const ASSISTANT_BLOCKING_BUDGET_MS = 15_000;
  58  
  59  // Search commands for collapsible display (grep, find, etc.)
  60  const BASH_SEARCH_COMMANDS = new Set(['find', 'grep', 'rg', 'ag', 'ack', 'locate', 'which', 'whereis']);
  61  
  62  // Read/view commands for collapsible display (cat, head, etc.)
  63  const BASH_READ_COMMANDS = new Set(['cat', 'head', 'tail', 'less', 'more',
  64  // Analysis commands
  65  'wc', 'stat', 'file', 'strings',
  66  // Data processing — commonly used to parse/transform file content in pipes
  67  'jq', 'awk', 'cut', 'sort', 'uniq', 'tr']);
  68  
  69  // Directory-listing commands for collapsible display (ls, tree, du).
  70  // Split from BASH_READ_COMMANDS so the summary says "Listed N directories"
  71  // instead of the misleading "Read N files".
  72  const BASH_LIST_COMMANDS = new Set(['ls', 'tree', 'du']);
  73  
  74  // Commands that are semantic-neutral in any position — pure output/status commands
  75  // that don't change the read/search nature of the overall pipeline.
  76  // e.g. `ls dir && echo "---" && ls dir2` is still a read-only compound command.
  77  const BASH_SEMANTIC_NEUTRAL_COMMANDS = new Set(['echo', 'printf', 'true', 'false', ':' // bash no-op
  78  ]);
  79  
  80  // Commands that typically produce no stdout on success
  81  const BASH_SILENT_COMMANDS = new Set(['mv', 'cp', 'rm', 'mkdir', 'rmdir', 'chmod', 'chown', 'chgrp', 'touch', 'ln', 'cd', 'export', 'unset', 'wait']);
  82  
  83  /**
  84   * Checks if a bash command is a search or read operation.
  85   * Used to determine if the command should be collapsed in the UI.
  86   * Returns an object indicating whether it's a search or read operation.
  87   *
  88   * For pipelines (e.g., `cat file | bq`), ALL parts must be search/read commands
  89   * for the whole command to be considered collapsible.
  90   *
  91   * Semantic-neutral commands (echo, printf, true, false, :) are skipped in any
  92   * position, as they're pure output/status commands that don't affect the read/search
  93   * nature of the pipeline (e.g. `ls dir && echo "---" && ls dir2` is still a read).
  94   */
  95  export function isSearchOrReadBashCommand(command: string): {
  96    isSearch: boolean;
  97    isRead: boolean;
  98    isList: boolean;
  99  } {
 100    let partsWithOperators: string[];
 101    try {
 102      partsWithOperators = splitCommandWithOperators(command);
 103    } catch {
 104      // If we can't parse the command due to malformed syntax,
 105      // it's not a search/read command
 106      return {
 107        isSearch: false,
 108        isRead: false,
 109        isList: false
 110      };
 111    }
 112    if (partsWithOperators.length === 0) {
 113      return {
 114        isSearch: false,
 115        isRead: false,
 116        isList: false
 117      };
 118    }
 119    let hasSearch = false;
 120    let hasRead = false;
 121    let hasList = false;
 122    let hasNonNeutralCommand = false;
 123    let skipNextAsRedirectTarget = false;
 124    for (const part of partsWithOperators) {
 125      if (skipNextAsRedirectTarget) {
 126        skipNextAsRedirectTarget = false;
 127        continue;
 128      }
 129      if (part === '>' || part === '>>' || part === '>&') {
 130        skipNextAsRedirectTarget = true;
 131        continue;
 132      }
 133      if (part === '||' || part === '&&' || part === '|' || part === ';') {
 134        continue;
 135      }
 136      const baseCommand = part.trim().split(/\s+/)[0];
 137      if (!baseCommand) {
 138        continue;
 139      }
 140      if (BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {
 141        continue;
 142      }
 143      hasNonNeutralCommand = true;
 144      const isPartSearch = BASH_SEARCH_COMMANDS.has(baseCommand);
 145      const isPartRead = BASH_READ_COMMANDS.has(baseCommand);
 146      const isPartList = BASH_LIST_COMMANDS.has(baseCommand);
 147      if (!isPartSearch && !isPartRead && !isPartList) {
 148        return {
 149          isSearch: false,
 150          isRead: false,
 151          isList: false
 152        };
 153      }
 154      if (isPartSearch) hasSearch = true;
 155      if (isPartRead) hasRead = true;
 156      if (isPartList) hasList = true;
 157    }
 158  
 159    // Only neutral commands (e.g., just "echo foo") -- not collapsible
 160    if (!hasNonNeutralCommand) {
 161      return {
 162        isSearch: false,
 163        isRead: false,
 164        isList: false
 165      };
 166    }
 167    return {
 168      isSearch: hasSearch,
 169      isRead: hasRead,
 170      isList: hasList
 171    };
 172  }
 173  
 174  /**
 175   * Checks if a bash command is expected to produce no stdout on success.
 176   * Used to show "Done" instead of "(No output)" in the UI.
 177   */
 178  function isSilentBashCommand(command: string): boolean {
 179    let partsWithOperators: string[];
 180    try {
 181      partsWithOperators = splitCommandWithOperators(command);
 182    } catch {
 183      return false;
 184    }
 185    if (partsWithOperators.length === 0) {
 186      return false;
 187    }
 188    let hasNonFallbackCommand = false;
 189    let lastOperator: string | null = null;
 190    let skipNextAsRedirectTarget = false;
 191    for (const part of partsWithOperators) {
 192      if (skipNextAsRedirectTarget) {
 193        skipNextAsRedirectTarget = false;
 194        continue;
 195      }
 196      if (part === '>' || part === '>>' || part === '>&') {
 197        skipNextAsRedirectTarget = true;
 198        continue;
 199      }
 200      if (part === '||' || part === '&&' || part === '|' || part === ';') {
 201        lastOperator = part;
 202        continue;
 203      }
 204      const baseCommand = part.trim().split(/\s+/)[0];
 205      if (!baseCommand) {
 206        continue;
 207      }
 208      if (lastOperator === '||' && BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {
 209        continue;
 210      }
 211      hasNonFallbackCommand = true;
 212      if (!BASH_SILENT_COMMANDS.has(baseCommand)) {
 213        return false;
 214      }
 215    }
 216    return hasNonFallbackCommand;
 217  }
 218  
 219  // Commands that should not be auto-backgrounded
 220  const DISALLOWED_AUTO_BACKGROUND_COMMANDS = ['sleep' // Sleep should run in foreground unless explicitly backgrounded by user
 221  ];
 222  
 223  // Check if background tasks are disabled at module load time
 224  const isBackgroundTasksDisabled =
 225  // eslint-disable-next-line custom-rules/no-process-env-top-level -- Intentional: schema must be defined at module load
 226  isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_BACKGROUND_TASKS);
 227  const fullInputSchema = lazySchema(() => z.strictObject({
 228    command: z.string().describe('The command to execute'),
 229    timeout: semanticNumber(z.number().optional()).describe(`Optional timeout in milliseconds (max ${getMaxTimeoutMs()})`),
 230    description: z.string().optional().describe(`Clear, concise description of what this command does in active voice. Never use words like "complex" or "risk" in the description - just describe what it does.
 231  
 232  For simple commands (git, npm, standard CLI tools), keep it brief (5-10 words):
 233  - ls → "List files in current directory"
 234  - git status → "Show working tree status"
 235  - npm install → "Install package dependencies"
 236  
 237  For commands that are harder to parse at a glance (piped commands, obscure flags, etc.), add enough context to clarify what it does:
 238  - find . -name "*.tmp" -exec rm {} \\; → "Find and delete all .tmp files recursively"
 239  - git reset --hard origin/main → "Discard all local changes and match remote main"
 240  - curl -s url | jq '.data[]' → "Fetch JSON from URL and extract data array elements"`),
 241    run_in_background: semanticBoolean(z.boolean().optional()).describe(`Set to true to run this command in the background. Use Read to read the output later.`),
 242    dangerouslyDisableSandbox: semanticBoolean(z.boolean().optional()).describe('Set this to true to dangerously override sandbox mode and run commands without sandboxing.'),
 243    _simulatedSedEdit: z.object({
 244      filePath: z.string(),
 245      newContent: z.string()
 246    }).optional().describe('Internal: pre-computed sed edit result from preview')
 247  }));
 248  
 249  // Always omit _simulatedSedEdit from the model-facing schema. It is an internal-only
 250  // field set by SedEditPermissionRequest after the user approves a sed edit preview.
 251  // Exposing it in the schema would let the model bypass permission checks and the
 252  // sandbox by pairing an innocuous command with an arbitrary file write.
 253  // Also conditionally remove run_in_background when background tasks are disabled.
 254  const inputSchema = lazySchema(() => isBackgroundTasksDisabled ? fullInputSchema().omit({
 255    run_in_background: true,
 256    _simulatedSedEdit: true
 257  }) : fullInputSchema().omit({
 258    _simulatedSedEdit: true
 259  }));
 260  type InputSchema = ReturnType<typeof inputSchema>;
 261  
 262  // Use fullInputSchema for the type to always include run_in_background
 263  // (even when it's omitted from the schema, the code needs to handle it)
 264  export type BashToolInput = z.infer<ReturnType<typeof fullInputSchema>>;
 265  const COMMON_BACKGROUND_COMMANDS = ['npm', 'yarn', 'pnpm', 'node', 'python', 'python3', 'go', 'cargo', 'make', 'docker', 'terraform', 'webpack', 'vite', 'jest', 'pytest', 'curl', 'wget', 'build', 'test', 'serve', 'watch', 'dev'] as const;
 266  function getCommandTypeForLogging(command: string): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {
 267    const parts = splitCommand_DEPRECATED(command);
 268    if (parts.length === 0) return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
 269  
 270    // Check each part of the command to see if any match common background commands
 271    for (const part of parts) {
 272      const baseCommand = part.split(' ')[0] || '';
 273      if (COMMON_BACKGROUND_COMMANDS.includes(baseCommand as (typeof COMMON_BACKGROUND_COMMANDS)[number])) {
 274        return baseCommand as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
 275      }
 276    }
 277    return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
 278  }
 279  const outputSchema = lazySchema(() => z.object({
 280    stdout: z.string().describe('The standard output of the command'),
 281    stderr: z.string().describe('The standard error output of the command'),
 282    rawOutputPath: z.string().optional().describe('Path to raw output file for large MCP tool outputs'),
 283    interrupted: z.boolean().describe('Whether the command was interrupted'),
 284    isImage: z.boolean().optional().describe('Flag to indicate if stdout contains image data'),
 285    backgroundTaskId: z.string().optional().describe('ID of the background task if command is running in background'),
 286    backgroundedByUser: z.boolean().optional().describe('True if the user manually backgrounded the command with Ctrl+B'),
 287    assistantAutoBackgrounded: z.boolean().optional().describe('True if assistant-mode auto-backgrounded a long-running blocking command'),
 288    dangerouslyDisableSandbox: z.boolean().optional().describe('Flag to indicate if sandbox mode was overridden'),
 289    returnCodeInterpretation: z.string().optional().describe('Semantic interpretation for non-error exit codes with special meaning'),
 290    noOutputExpected: z.boolean().optional().describe('Whether the command is expected to produce no output on success'),
 291    structuredContent: z.array(z.any()).optional().describe('Structured content blocks'),
 292    persistedOutputPath: z.string().optional().describe('Path to the persisted full output in tool-results dir (set when output is too large for inline)'),
 293    persistedOutputSize: z.number().optional().describe('Total size of the output in bytes (set when output is too large for inline)')
 294  }));
 295  type OutputSchema = ReturnType<typeof outputSchema>;
 296  export type Out = z.infer<OutputSchema>;
 297  
 298  // Re-export BashProgress from centralized types to break import cycles
 299  export type { BashProgress } from '../../types/tools.js';
 300  import type { BashProgress } from '../../types/tools.js';
 301  
 302  /**
 303   * Checks if a command is allowed to be automatically backgrounded
 304   * @param command The command to check
 305   * @returns false for commands that should not be auto-backgrounded (like sleep)
 306   */
 307  function isAutobackgroundingAllowed(command: string): boolean {
 308    const parts = splitCommand_DEPRECATED(command);
 309    if (parts.length === 0) return true;
 310  
 311    // Get the first part which should be the base command
 312    const baseCommand = parts[0]?.trim();
 313    if (!baseCommand) return true;
 314    return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(baseCommand);
 315  }
 316  
 317  /**
 318   * Detect standalone or leading `sleep N` patterns that should use Monitor
 319   * instead. Catches `sleep 5`, `sleep 5 && check`, `sleep 5; check` — but
 320   * not sleep inside pipelines, subshells, or scripts (those are fine).
 321   */
 322  export function detectBlockedSleepPattern(command: string): string | null {
 323    const parts = splitCommand_DEPRECATED(command);
 324    if (parts.length === 0) return null;
 325    const first = parts[0]?.trim() ?? '';
 326    // Bare `sleep N` or `sleep N.N` as the first subcommand.
 327    // Float durations (sleep 0.5) are allowed — those are legit pacing, not polls.
 328    const m = /^sleep\s+(\d+)\s*$/.exec(first);
 329    if (!m) return null;
 330    const secs = parseInt(m[1]!, 10);
 331    if (secs < 2) return null; // sub-2s sleeps are fine (rate limiting, pacing)
 332  
 333    // `sleep N` alone → "what are you waiting for?"
 334    // `sleep N && check` → "use Monitor { command: check }"
 335    const rest = parts.slice(1).join(' ').trim();
 336    return rest ? `sleep ${secs} followed by: ${rest}` : `standalone sleep ${secs}`;
 337  }
 338  
 339  /**
 340   * Checks if a command contains tools that shouldn't run in sandbox
 341   * This includes:
 342   * - Dynamic config-based disabled commands and substrings (tengu_sandbox_disabled_commands)
 343   * - User-configured commands from settings.json (sandbox.excludedCommands)
 344   *
 345   * User-configured commands support the same pattern syntax as permission rules:
 346   * - Exact matches: "npm run lint"
 347   * - Prefix patterns: "npm run test:*"
 348   */
 349  
 350  type SimulatedSedEditResult = {
 351    data: Out;
 352  };
 353  type SimulatedSedEditContext = Pick<ToolUseContext, 'readFileState' | 'updateFileHistoryState'>;
 354  
 355  /**
 356   * Applies a simulated sed edit directly instead of running sed.
 357   * This is used by the permission dialog to ensure what the user previews
 358   * is exactly what gets written to the file.
 359   */
 360  async function applySedEdit(simulatedEdit: {
 361    filePath: string;
 362    newContent: string;
 363  }, toolUseContext: SimulatedSedEditContext, parentMessage?: AssistantMessage): Promise<SimulatedSedEditResult> {
 364    const {
 365      filePath,
 366      newContent
 367    } = simulatedEdit;
 368    const absoluteFilePath = expandPath(filePath);
 369    const fs = getFsImplementation();
 370  
 371    // Read original content for VS Code notification
 372    const encoding = detectFileEncoding(absoluteFilePath);
 373    let originalContent: string;
 374    try {
 375      originalContent = await fs.readFile(absoluteFilePath, {
 376        encoding
 377      });
 378    } catch (e) {
 379      if (isENOENT(e)) {
 380        return {
 381          data: {
 382            stdout: '',
 383            stderr: `sed: ${filePath}: No such file or directory\nExit code 1`,
 384            interrupted: false
 385          }
 386        };
 387      }
 388      throw e;
 389    }
 390  
 391    // Track file history before making changes (for undo support)
 392    if (fileHistoryEnabled() && parentMessage) {
 393      await fileHistoryTrackEdit(toolUseContext.updateFileHistoryState, absoluteFilePath, parentMessage.uuid);
 394    }
 395  
 396    // Detect line endings and write new content
 397    const endings = detectLineEndings(absoluteFilePath);
 398    writeTextContent(absoluteFilePath, newContent, encoding, endings);
 399  
 400    // Notify VS Code about the file change
 401    notifyVscodeFileUpdated(absoluteFilePath, originalContent, newContent);
 402  
 403    // Update read timestamp to invalidate stale writes
 404    toolUseContext.readFileState.set(absoluteFilePath, {
 405      content: newContent,
 406      timestamp: getFileModificationTime(absoluteFilePath),
 407      offset: undefined,
 408      limit: undefined
 409    });
 410  
 411    // Return success result matching sed output format (sed produces no output on success)
 412    return {
 413      data: {
 414        stdout: '',
 415        stderr: '',
 416        interrupted: false
 417      }
 418    };
 419  }
 420  export const BashTool = buildTool({
 421    name: BASH_TOOL_NAME,
 422    searchHint: 'execute shell commands',
 423    // 30K chars - tool result persistence threshold
 424    maxResultSizeChars: 30_000,
 425    strict: true,
 426    async description({
 427      description
 428    }) {
 429      return description || 'Run shell command';
 430    },
 431    async prompt() {
 432      return getSimplePrompt();
 433    },
 434    isConcurrencySafe(input) {
 435      return this.isReadOnly?.(input) ?? false;
 436    },
 437    isReadOnly(input) {
 438      const compoundCommandHasCd = commandHasAnyCd(input.command);
 439      const result = checkReadOnlyConstraints(input, compoundCommandHasCd);
 440      return result.behavior === 'allow';
 441    },
 442    toAutoClassifierInput(input) {
 443      return input.command;
 444    },
 445    async preparePermissionMatcher({
 446      command
 447    }) {
 448      // Hook `if` filtering is "no match → skip hook" (deny-like semantics), so
 449      // compound commands must fire the hook if ANY subcommand matches. Without
 450      // splitting, `ls && git push` would bypass a `Bash(git *)` security hook.
 451      const parsed = await parseForSecurity(command);
 452      if (parsed.kind !== 'simple') {
 453        // parse-unavailable / too-complex: fail safe by running the hook.
 454        return () => true;
 455      }
 456      // Match on argv (strips leading VAR=val) so `FOO=bar git push` still
 457      // matches `Bash(git *)`.
 458      const subcommands = parsed.commands.map(c => c.argv.join(' '));
 459      return pattern => {
 460        const prefix = permissionRuleExtractPrefix(pattern);
 461        return subcommands.some(cmd => {
 462          if (prefix !== null) {
 463            return cmd === prefix || cmd.startsWith(`${prefix} `);
 464          }
 465          return matchWildcardPattern(pattern, cmd);
 466        });
 467      };
 468    },
 469    isSearchOrReadCommand(input) {
 470      const parsed = inputSchema().safeParse(input);
 471      if (!parsed.success) return {
 472        isSearch: false,
 473        isRead: false,
 474        isList: false
 475      };
 476      return isSearchOrReadBashCommand(parsed.data.command);
 477    },
 478    get inputSchema(): InputSchema {
 479      return inputSchema();
 480    },
 481    get outputSchema(): OutputSchema {
 482      return outputSchema();
 483    },
 484    userFacingName(input) {
 485      if (!input) {
 486        return 'Bash';
 487      }
 488      // Render sed in-place edits as file edits
 489      if (input.command) {
 490        const sedInfo = parseSedEditCommand(input.command);
 491        if (sedInfo) {
 492          return fileEditUserFacingName({
 493            file_path: sedInfo.filePath,
 494            old_string: 'x'
 495          });
 496        }
 497      }
 498      // Env var FIRST: shouldUseSandbox → splitCommand_DEPRECATED → shell-quote's
 499      // `new RegExp` per call. userFacingName runs per-render for every bash
 500      // message in history; with ~50 msgs + one slow-to-tokenize command, this
 501      // exceeds the shimmer tick → transition abort → infinite retry (#21605).
 502      return isEnvTruthy(process.env.CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR) && shouldUseSandbox(input) ? 'SandboxedBash' : 'Bash';
 503    },
 504    getToolUseSummary(input) {
 505      if (!input?.command) {
 506        return null;
 507      }
 508      const {
 509        command,
 510        description
 511      } = input;
 512      if (description) {
 513        return description;
 514      }
 515      return truncate(command, TOOL_SUMMARY_MAX_LENGTH);
 516    },
 517    getActivityDescription(input) {
 518      if (!input?.command) {
 519        return 'Running command';
 520      }
 521      const desc = input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH);
 522      return `Running ${desc}`;
 523    },
 524    async validateInput(input: BashToolInput): Promise<ValidationResult> {
 525      if (feature('MONITOR_TOOL') && !isBackgroundTasksDisabled && !input.run_in_background) {
 526        const sleepPattern = detectBlockedSleepPattern(input.command);
 527        if (sleepPattern !== null) {
 528          return {
 529            result: false,
 530            message: `Blocked: ${sleepPattern}. Run blocking commands in the background with run_in_background: true — you'll get a completion notification when done. For streaming events (watching logs, polling APIs), use the Monitor tool. If you genuinely need a delay (rate limiting, deliberate pacing), keep it under 2 seconds.`,
 531            errorCode: 10
 532          };
 533        }
 534      }
 535      return {
 536        result: true
 537      };
 538    },
 539    async checkPermissions(input, context): Promise<PermissionResult> {
 540      return bashToolHasPermission(input, context);
 541    },
 542    renderToolUseMessage,
 543    renderToolUseProgressMessage,
 544    renderToolUseQueuedMessage,
 545    renderToolResultMessage,
 546    // BashToolResultMessage shows <OutputLine content={stdout}> + stderr.
 547    // UI never shows persistedOutputPath wrapper, backgroundInfo — those are
 548    // model-facing (mapToolResult... below).
 549    extractSearchText({
 550      stdout,
 551      stderr
 552    }) {
 553      return stderr ? `${stdout}\n${stderr}` : stdout;
 554    },
 555    mapToolResultToToolResultBlockParam({
 556      interrupted,
 557      stdout,
 558      stderr,
 559      isImage,
 560      backgroundTaskId,
 561      backgroundedByUser,
 562      assistantAutoBackgrounded,
 563      structuredContent,
 564      persistedOutputPath,
 565      persistedOutputSize
 566    }, toolUseID): ToolResultBlockParam {
 567      // Handle structured content
 568      if (structuredContent && structuredContent.length > 0) {
 569        return {
 570          tool_use_id: toolUseID,
 571          type: 'tool_result',
 572          content: structuredContent
 573        };
 574      }
 575  
 576      // For image data, format as image content block for Claude
 577      if (isImage) {
 578        const block = buildImageToolResult(stdout, toolUseID);
 579        if (block) return block;
 580      }
 581      let processedStdout = stdout;
 582      if (stdout) {
 583        // Replace any leading newlines or lines with only whitespace
 584        processedStdout = stdout.replace(/^(\s*\n)+/, '');
 585        // Still trim the end as before
 586        processedStdout = processedStdout.trimEnd();
 587      }
 588  
 589      // For large output that was persisted to disk, build <persisted-output>
 590      // message for the model. The UI never sees this — it uses data.stdout.
 591      if (persistedOutputPath) {
 592        const preview = generatePreview(processedStdout, PREVIEW_SIZE_BYTES);
 593        processedStdout = buildLargeToolResultMessage({
 594          filepath: persistedOutputPath,
 595          originalSize: persistedOutputSize ?? 0,
 596          isJson: false,
 597          preview: preview.preview,
 598          hasMore: preview.hasMore
 599        });
 600      }
 601      let errorMessage = stderr.trim();
 602      if (interrupted) {
 603        if (stderr) errorMessage += EOL;
 604        errorMessage += '<error>Command was aborted before completion</error>';
 605      }
 606      let backgroundInfo = '';
 607      if (backgroundTaskId) {
 608        const outputPath = getTaskOutputPath(backgroundTaskId);
 609        if (assistantAutoBackgrounded) {
 610          backgroundInfo = `Command exceeded the assistant-mode blocking budget (${ASSISTANT_BLOCKING_BUDGET_MS / 1000}s) and was moved to the background with ID: ${backgroundTaskId}. It is still running — you will be notified when it completes. Output is being written to: ${outputPath}. In assistant mode, delegate long-running work to a subagent or use run_in_background to keep this conversation responsive.`;
 611        } else if (backgroundedByUser) {
 612          backgroundInfo = `Command was manually backgrounded by user with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`;
 613        } else {
 614          backgroundInfo = `Command running in background with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`;
 615        }
 616      }
 617      return {
 618        tool_use_id: toolUseID,
 619        type: 'tool_result',
 620        content: [processedStdout, errorMessage, backgroundInfo].filter(Boolean).join('\n'),
 621        is_error: interrupted
 622      };
 623    },
 624    async call(input: BashToolInput, toolUseContext, _canUseTool?: CanUseToolFn, parentMessage?: AssistantMessage, onProgress?: ToolCallProgress<BashProgress>) {
 625      // Handle simulated sed edit - apply directly instead of running sed
 626      // This ensures what the user previewed is exactly what gets written
 627      if (input._simulatedSedEdit) {
 628        return applySedEdit(input._simulatedSedEdit, toolUseContext, parentMessage);
 629      }
 630      const {
 631        abortController,
 632        getAppState,
 633        setAppState,
 634        setToolJSX
 635      } = toolUseContext;
 636      const stdoutAccumulator = new EndTruncatingAccumulator();
 637      let stderrForShellReset = '';
 638      let interpretationResult: ReturnType<typeof interpretCommandResult> | undefined;
 639      let progressCounter = 0;
 640      let wasInterrupted = false;
 641      let result: ExecResult;
 642      const isMainThread = !toolUseContext.agentId;
 643      const preventCwdChanges = !isMainThread;
 644      try {
 645        // Use the new async generator version of runShellCommand
 646        const commandGenerator = runShellCommand({
 647          input,
 648          abortController,
 649          // Use the always-shared task channel so async agents' background
 650          // bash tasks are actually registered (and killable on agent exit).
 651          setAppState: toolUseContext.setAppStateForTasks ?? setAppState,
 652          setToolJSX,
 653          preventCwdChanges,
 654          isMainThread,
 655          toolUseId: toolUseContext.toolUseId,
 656          agentId: toolUseContext.agentId
 657        });
 658  
 659        // Consume the generator and capture the return value
 660        let generatorResult;
 661        do {
 662          generatorResult = await commandGenerator.next();
 663          if (!generatorResult.done && onProgress) {
 664            const progress = generatorResult.value;
 665            onProgress({
 666              toolUseID: `bash-progress-${progressCounter++}`,
 667              data: {
 668                type: 'bash_progress',
 669                output: progress.output,
 670                fullOutput: progress.fullOutput,
 671                elapsedTimeSeconds: progress.elapsedTimeSeconds,
 672                totalLines: progress.totalLines,
 673                totalBytes: progress.totalBytes,
 674                taskId: progress.taskId,
 675                timeoutMs: progress.timeoutMs
 676              }
 677            });
 678          }
 679        } while (!generatorResult.done);
 680  
 681        // Get the final result from the generator's return value
 682        result = generatorResult.value;
 683        trackGitOperations(input.command, result.code, result.stdout);
 684        const isInterrupt = result.interrupted && abortController.signal.reason === 'interrupt';
 685  
 686        // stderr is interleaved in stdout (merged fd) — result.stdout has both
 687        stdoutAccumulator.append((result.stdout || '').trimEnd() + EOL);
 688  
 689        // Interpret the command result using semantic rules
 690        interpretationResult = interpretCommandResult(input.command, result.code, result.stdout || '', '');
 691  
 692        // Check for git index.lock error (stderr is in stdout now)
 693        if (result.stdout && result.stdout.includes(".git/index.lock': File exists")) {
 694          logEvent('tengu_git_index_lock_error', {});
 695        }
 696        if (interpretationResult.isError && !isInterrupt) {
 697          // Only add exit code if it's actually an error
 698          if (result.code !== 0) {
 699            stdoutAccumulator.append(`Exit code ${result.code}`);
 700          }
 701        }
 702        if (!preventCwdChanges) {
 703          const appState = getAppState();
 704          if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {
 705            stderrForShellReset = stdErrAppendShellResetMessage('');
 706          }
 707        }
 708  
 709        // Annotate output with sandbox violations if any (stderr is in stdout)
 710        const outputWithSbFailures = SandboxManager.annotateStderrWithSandboxFailures(input.command, result.stdout || '');
 711        if (result.preSpawnError) {
 712          throw new Error(result.preSpawnError);
 713        }
 714        if (interpretationResult.isError && !isInterrupt) {
 715          // stderr is merged into stdout (merged fd); outputWithSbFailures
 716          // already has the full output. Pass '' for stdout to avoid
 717          // duplication in getErrorParts() and processBashCommand.
 718          throw new ShellError('', outputWithSbFailures, result.code, result.interrupted);
 719        }
 720        wasInterrupted = result.interrupted;
 721      } finally {
 722        if (setToolJSX) setToolJSX(null);
 723      }
 724  
 725      // Get final string from accumulator
 726      const stdout = stdoutAccumulator.toString();
 727  
 728      // Large output: the file on disk has more than getMaxOutputLength() bytes.
 729      // stdout already contains the first chunk (from getStdout()). Copy the
 730      // output file to the tool-results dir so the model can read it via
 731      // FileRead. If > 64 MB, truncate after copying.
 732      const MAX_PERSISTED_SIZE = 64 * 1024 * 1024;
 733      let persistedOutputPath: string | undefined;
 734      let persistedOutputSize: number | undefined;
 735      if (result.outputFilePath && result.outputTaskId) {
 736        try {
 737          const fileStat = await fsStat(result.outputFilePath);
 738          persistedOutputSize = fileStat.size;
 739          await ensureToolResultsDir();
 740          const dest = getToolResultPath(result.outputTaskId, false);
 741          if (fileStat.size > MAX_PERSISTED_SIZE) {
 742            await fsTruncate(result.outputFilePath, MAX_PERSISTED_SIZE);
 743          }
 744          try {
 745            await link(result.outputFilePath, dest);
 746          } catch {
 747            await copyFile(result.outputFilePath, dest);
 748          }
 749          persistedOutputPath = dest;
 750        } catch {
 751          // File may already be gone — stdout preview is sufficient
 752        }
 753      }
 754      const commandType = input.command.split(' ')[0];
 755      logEvent('tengu_bash_tool_command_executed', {
 756        command_type: commandType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
 757        stdout_length: stdout.length,
 758        stderr_length: 0,
 759        exit_code: result.code,
 760        interrupted: wasInterrupted
 761      });
 762  
 763      // Log code indexing tool usage
 764      const codeIndexingTool = detectCodeIndexingFromCommand(input.command);
 765      if (codeIndexingTool) {
 766        logEvent('tengu_code_indexing_tool_used', {
 767          tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
 768          source: 'cli' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
 769          success: result.code === 0
 770        });
 771      }
 772      let strippedStdout = stripEmptyLines(stdout);
 773  
 774      // Claude Code hints protocol: CLIs/SDKs gated on CLAUDECODE=1 emit a
 775      // `<claude-code-hint />` tag to stderr (merged into stdout here). Scan,
 776      // record for useClaudeCodeHintRecommendation to surface, then strip
 777      // so the model never sees the tag — a zero-token side channel.
 778      // Stripping runs unconditionally (subagent output must stay clean too);
 779      // only the dialog recording is main-thread-only.
 780      const extracted = extractClaudeCodeHints(strippedStdout, input.command);
 781      strippedStdout = extracted.stripped;
 782      if (isMainThread && extracted.hints.length > 0) {
 783        for (const hint of extracted.hints) maybeRecordPluginHint(hint);
 784      }
 785      let isImage = isImageOutput(strippedStdout);
 786  
 787      // Cap image dimensions + size if present (CC-304 — see
 788      // resizeShellImageOutput). Scope the decoded buffer so it can be reclaimed
 789      // before we build the output Out object.
 790      let compressedStdout = strippedStdout;
 791      if (isImage) {
 792        const resized = await resizeShellImageOutput(strippedStdout, result.outputFilePath, persistedOutputSize);
 793        if (resized) {
 794          compressedStdout = resized;
 795        } else {
 796          // Parse failed or file too large (e.g. exceeds MAX_IMAGE_FILE_SIZE).
 797          // Keep isImage in sync with what we actually send so the UI label stays
 798          // accurate — mapToolResultToToolResultBlockParam's defensive
 799          // fallthrough will send text, not an image block.
 800          isImage = false;
 801        }
 802      }
 803      const data: Out = {
 804        stdout: compressedStdout,
 805        stderr: stderrForShellReset,
 806        interrupted: wasInterrupted,
 807        isImage,
 808        returnCodeInterpretation: interpretationResult?.message,
 809        noOutputExpected: isSilentBashCommand(input.command),
 810        backgroundTaskId: result.backgroundTaskId,
 811        backgroundedByUser: result.backgroundedByUser,
 812        assistantAutoBackgrounded: result.assistantAutoBackgrounded,
 813        dangerouslyDisableSandbox: 'dangerouslyDisableSandbox' in input ? input.dangerouslyDisableSandbox as boolean | undefined : undefined,
 814        persistedOutputPath,
 815        persistedOutputSize
 816      };
 817      return {
 818        data
 819      };
 820    },
 821    renderToolUseErrorMessage,
 822    isResultTruncated(output: Out): boolean {
 823      return isOutputLineTruncated(output.stdout) || isOutputLineTruncated(output.stderr);
 824    }
 825  } satisfies ToolDef<InputSchema, Out, BashProgress>);
 826  async function* runShellCommand({
 827    input,
 828    abortController,
 829    setAppState,
 830    setToolJSX,
 831    preventCwdChanges,
 832    isMainThread,
 833    toolUseId,
 834    agentId
 835  }: {
 836    input: BashToolInput;
 837    abortController: AbortController;
 838    setAppState: (f: (prev: AppState) => AppState) => void;
 839    setToolJSX?: SetToolJSXFn;
 840    preventCwdChanges?: boolean;
 841    isMainThread?: boolean;
 842    toolUseId?: string;
 843    agentId?: AgentId;
 844  }): AsyncGenerator<{
 845    type: 'progress';
 846    output: string;
 847    fullOutput: string;
 848    elapsedTimeSeconds: number;
 849    totalLines: number;
 850    totalBytes?: number;
 851    taskId?: string;
 852    timeoutMs?: number;
 853  }, ExecResult, void> {
 854    const {
 855      command,
 856      description,
 857      timeout,
 858      run_in_background
 859    } = input;
 860    const timeoutMs = timeout || getDefaultTimeoutMs();
 861    let fullOutput = '';
 862    let lastProgressOutput = '';
 863    let lastTotalLines = 0;
 864    let lastTotalBytes = 0;
 865    let backgroundShellId: string | undefined = undefined;
 866    let assistantAutoBackgrounded = false;
 867  
 868    // Progress signal: resolved by onProgress callback from the shared poller,
 869    // waking the generator to yield a progress update.
 870    let resolveProgress: (() => void) | null = null;
 871    function createProgressSignal(): Promise<null> {
 872      return new Promise<null>(resolve => {
 873        resolveProgress = () => resolve(null);
 874      });
 875    }
 876  
 877    // Determine if auto-backgrounding should be enabled
 878    // Only enable for commands that are allowed to be auto-backgrounded
 879    // and when background tasks are not disabled
 880    const shouldAutoBackground = !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command);
 881    const shellCommand = await exec(command, abortController.signal, 'bash', {
 882      timeout: timeoutMs,
 883      onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {
 884        lastProgressOutput = lastLines;
 885        fullOutput = allLines;
 886        lastTotalLines = totalLines;
 887        lastTotalBytes = isIncomplete ? totalBytes : 0;
 888        // Wake the generator so it yields the new progress data
 889        const resolve = resolveProgress;
 890        if (resolve) {
 891          resolveProgress = null;
 892          resolve();
 893        }
 894      },
 895      preventCwdChanges,
 896      shouldUseSandbox: shouldUseSandbox(input),
 897      shouldAutoBackground
 898    });
 899  
 900    // Start the command execution
 901    const resultPromise = shellCommand.result;
 902  
 903    // Helper to spawn a background task and return its ID
 904    async function spawnBackgroundTask(): Promise<string> {
 905      const handle = await spawnShellTask({
 906        command,
 907        description: description || command,
 908        shellCommand,
 909        toolUseId,
 910        agentId
 911      }, {
 912        abortController,
 913        getAppState: () => {
 914          // We don't have direct access to getAppState here, but spawn doesn't
 915          // actually use it during the spawn process
 916          throw new Error('getAppState not available in runShellCommand context');
 917        },
 918        setAppState
 919      });
 920      return handle.taskId;
 921    }
 922  
 923    // Helper to start backgrounding with optional logging
 924    function startBackgrounding(eventName: string, backgroundFn?: (shellId: string) => void): void {
 925      // If a foreground task is already registered (via registerForeground in the
 926      // progress loop), background it in-place instead of re-spawning. Re-spawning
 927      // would overwrite tasks[taskId], emit a duplicate task_started SDK event,
 928      // and leak the first cleanup callback.
 929      if (foregroundTaskId) {
 930        if (!backgroundExistingForegroundTask(foregroundTaskId, shellCommand, description || command, setAppState, toolUseId)) {
 931          return;
 932        }
 933        backgroundShellId = foregroundTaskId;
 934        logEvent(eventName, {
 935          command_type: getCommandTypeForLogging(command)
 936        });
 937        backgroundFn?.(foregroundTaskId);
 938        return;
 939      }
 940  
 941      // No foreground task registered — spawn a new background task
 942      // Note: spawn is essentially synchronous despite being async
 943      void spawnBackgroundTask().then(shellId => {
 944        backgroundShellId = shellId;
 945  
 946        // Wake the generator's Promise.race so it sees backgroundShellId.
 947        // Without this, if the poller has stopped ticking for this task
 948        // (no output + shared-poller race with sibling stopPolling calls)
 949        // and the process is hung on I/O, the race at line ~1357 never
 950        // resolves and the generator deadlocks despite being backgrounded.
 951        const resolve = resolveProgress;
 952        if (resolve) {
 953          resolveProgress = null;
 954          resolve();
 955        }
 956        logEvent(eventName, {
 957          command_type: getCommandTypeForLogging(command)
 958        });
 959        if (backgroundFn) {
 960          backgroundFn(shellId);
 961        }
 962      });
 963    }
 964  
 965    // Set up auto-backgrounding on timeout if enabled
 966    // Only background commands that are allowed to be auto-backgrounded (not sleep, etc.)
 967    if (shellCommand.onTimeout && shouldAutoBackground) {
 968      shellCommand.onTimeout(backgroundFn => {
 969        startBackgrounding('tengu_bash_command_timeout_backgrounded', backgroundFn);
 970      });
 971    }
 972  
 973    // In assistant mode, the main agent should stay responsive. Auto-background
 974    // blocking commands after ASSISTANT_BLOCKING_BUDGET_MS so the agent can keep
 975    // coordinating instead of waiting. The command keeps running — no state loss.
 976    if (feature('KAIROS') && getKairosActive() && isMainThread && !isBackgroundTasksDisabled && run_in_background !== true) {
 977      setTimeout(() => {
 978        if (shellCommand.status === 'running' && backgroundShellId === undefined) {
 979          assistantAutoBackgrounded = true;
 980          startBackgrounding('tengu_bash_command_assistant_auto_backgrounded');
 981        }
 982      }, ASSISTANT_BLOCKING_BUDGET_MS).unref();
 983    }
 984  
 985    // Handle Claude asking to run it in the background explicitly
 986    // When explicitly requested via run_in_background, always honor the request
 987    // regardless of the command type (isAutobackgroundingAllowed only applies to automatic backgrounding)
 988    // Skip if background tasks are disabled - run in foreground instead
 989    if (run_in_background === true && !isBackgroundTasksDisabled) {
 990      const shellId = await spawnBackgroundTask();
 991      logEvent('tengu_bash_command_explicitly_backgrounded', {
 992        command_type: getCommandTypeForLogging(command)
 993      });
 994      return {
 995        stdout: '',
 996        stderr: '',
 997        code: 0,
 998        interrupted: false,
 999        backgroundTaskId: shellId
1000      };
1001    }
1002  
1003    // Wait for the initial threshold before showing progress
1004    const startTime = Date.now();
1005    let foregroundTaskId: string | undefined = undefined;
1006    {
1007      const initialResult = await Promise.race([resultPromise, new Promise<null>(resolve => {
1008        const t = setTimeout((r: (v: null) => void) => r(null), PROGRESS_THRESHOLD_MS, resolve);
1009        t.unref();
1010      })]);
1011      if (initialResult !== null) {
1012        shellCommand.cleanup();
1013        return initialResult;
1014      }
1015      if (backgroundShellId) {
1016        return {
1017          stdout: '',
1018          stderr: '',
1019          code: 0,
1020          interrupted: false,
1021          backgroundTaskId: backgroundShellId,
1022          assistantAutoBackgrounded
1023        };
1024      }
1025    }
1026  
1027    // Start polling the output file for progress. The poller's #tick calls
1028    // onProgress every second, which resolves progressSignal below.
1029    TaskOutput.startPolling(shellCommand.taskOutput.taskId);
1030  
1031    // Progress loop: wake is driven by the shared poller calling onProgress,
1032    // which resolves the progressSignal.
1033    try {
1034      while (true) {
1035        const progressSignal = createProgressSignal();
1036        const result = await Promise.race([resultPromise, progressSignal]);
1037        if (result !== null) {
1038          // Race: backgrounding fired (15s timer / onTimeout / Ctrl+B) but the
1039          // command completed before the next poll tick. #handleExit sets
1040          // backgroundTaskId but skips outputFilePath (it assumes the background
1041          // message or <task_notification> will carry the path). Strip
1042          // backgroundTaskId so the model sees a clean completed command,
1043          // reconstruct outputFilePath for large outputs, and suppress the
1044          // redundant <task_notification> from the .then() handler.
1045          // Check result.backgroundTaskId (not the closure var) to also cover
1046          // Ctrl+B, which calls shellCommand.background() directly.
1047          if (result.backgroundTaskId !== undefined) {
1048            markTaskNotified(result.backgroundTaskId, setAppState);
1049            const fixedResult: ExecResult = {
1050              ...result,
1051              backgroundTaskId: undefined
1052            };
1053            // Mirror ShellCommand.#handleExit's large-output branch that was
1054            // skipped because #backgroundTaskId was set.
1055            const {
1056              taskOutput
1057            } = shellCommand;
1058            if (taskOutput.stdoutToFile && !taskOutput.outputFileRedundant) {
1059              fixedResult.outputFilePath = taskOutput.path;
1060              fixedResult.outputFileSize = taskOutput.outputFileSize;
1061              fixedResult.outputTaskId = taskOutput.taskId;
1062            }
1063            shellCommand.cleanup();
1064            return fixedResult;
1065          }
1066          // Command has completed - return the actual result
1067          // If we registered as a foreground task, unregister it
1068          if (foregroundTaskId) {
1069            unregisterForeground(foregroundTaskId, setAppState);
1070          }
1071          // Clean up stream resources for foreground commands
1072          // (backgrounded commands are cleaned up by LocalShellTask)
1073          shellCommand.cleanup();
1074          return result;
1075        }
1076  
1077        // Check if command was backgrounded (either via old mechanism or new backgroundAll)
1078        if (backgroundShellId) {
1079          return {
1080            stdout: '',
1081            stderr: '',
1082            code: 0,
1083            interrupted: false,
1084            backgroundTaskId: backgroundShellId,
1085            assistantAutoBackgrounded
1086          };
1087        }
1088  
1089        // Check if this foreground task was backgrounded via backgroundAll()
1090        if (foregroundTaskId) {
1091          // shellCommand.status becomes 'backgrounded' when background() is called
1092          if (shellCommand.status === 'backgrounded') {
1093            return {
1094              stdout: '',
1095              stderr: '',
1096              code: 0,
1097              interrupted: false,
1098              backgroundTaskId: foregroundTaskId,
1099              backgroundedByUser: true
1100            };
1101          }
1102        }
1103  
1104        // Time for a progress update
1105        const elapsed = Date.now() - startTime;
1106        const elapsedSeconds = Math.floor(elapsed / 1000);
1107  
1108        // Show minimal backgrounding UI if available
1109        // Skip if background tasks are disabled
1110        if (!isBackgroundTasksDisabled && backgroundShellId === undefined && elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 && setToolJSX) {
1111          // Register this command as a foreground task so it can be backgrounded via Ctrl+B
1112          if (!foregroundTaskId) {
1113            foregroundTaskId = registerForeground({
1114              command,
1115              description: description || command,
1116              shellCommand,
1117              agentId
1118            }, setAppState, toolUseId);
1119          }
1120          setToolJSX({
1121            jsx: <BackgroundHint />,
1122            shouldHidePromptInput: false,
1123            shouldContinueAnimation: true,
1124            showSpinner: true
1125          });
1126        }
1127        yield {
1128          type: 'progress',
1129          fullOutput,
1130          output: lastProgressOutput,
1131          elapsedTimeSeconds: elapsedSeconds,
1132          totalLines: lastTotalLines,
1133          totalBytes: lastTotalBytes,
1134          taskId: shellCommand.taskOutput.taskId,
1135          ...(timeout ? {
1136            timeoutMs
1137          } : undefined)
1138        };
1139      }
1140    } finally {
1141      TaskOutput.stopPolling(shellCommand.taskOutput.taskId);
1142    }
1143  }
1144  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","ToolResultBlockParam","copyFile","stat","fsStat","truncate","fsTruncate","link","React","CanUseToolFn","AppState","z","getKairosActive","TOOL_SUMMARY_MAX_LENGTH","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","notifyVscodeFileUpdated","SetToolJSXFn","ToolCallProgress","ToolUseContext","ValidationResult","buildTool","ToolDef","backgroundExistingForegroundTask","markTaskNotified","registerForeground","spawnShellTask","unregisterForeground","AgentId","AssistantMessage","parseForSecurity","splitCommand_DEPRECATED","splitCommandWithOperators","extractClaudeCodeHints","detectCodeIndexingFromCommand","isEnvTruthy","isENOENT","ShellError","detectFileEncoding","detectLineEndings","getFileModificationTime","writeTextContent","fileHistoryEnabled","fileHistoryTrackEdit","getFsImplementation","lazySchema","expandPath","PermissionResult","maybeRecordPluginHint","exec","ExecResult","SandboxManager","semanticBoolean","semanticNumber","EndTruncatingAccumulator","getTaskOutputPath","TaskOutput","isOutputLineTruncated","buildLargeToolResultMessage","ensureToolResultsDir","generatePreview","getToolResultPath","PREVIEW_SIZE_BYTES","userFacingName","fileEditUserFacingName","trackGitOperations","bashToolHasPermission","commandHasAnyCd","matchWildcardPattern","permissionRuleExtractPrefix","interpretCommandResult","getDefaultTimeoutMs","getMaxTimeoutMs","getSimplePrompt","checkReadOnlyConstraints","parseSedEditCommand","shouldUseSandbox","BASH_TOOL_NAME","BackgroundHint","renderToolResultMessage","renderToolUseErrorMessage","renderToolUseMessage","renderToolUseProgressMessage","renderToolUseQueuedMessage","buildImageToolResult","isImageOutput","resetCwdIfOutsideProject","resizeShellImageOutput","stdErrAppendShellResetMessage","stripEmptyLines","EOL","PROGRESS_THRESHOLD_MS","ASSISTANT_BLOCKING_BUDGET_MS","BASH_SEARCH_COMMANDS","Set","BASH_READ_COMMANDS","BASH_LIST_COMMANDS","BASH_SEMANTIC_NEUTRAL_COMMANDS","BASH_SILENT_COMMANDS","isSearchOrReadBashCommand","command","isSearch","isRead","isList","partsWithOperators","length","hasSearch","hasRead","hasList","hasNonNeutralCommand","skipNextAsRedirectTarget","part","baseCommand","trim","split","has","isPartSearch","isPartRead","isPartList","isSilentBashCommand","hasNonFallbackCommand","lastOperator","DISALLOWED_AUTO_BACKGROUND_COMMANDS","isBackgroundTasksDisabled","process","env","CLAUDE_CODE_DISABLE_BACKGROUND_TASKS","fullInputSchema","strictObject","string","describe","timeout","number","optional","description","run_in_background","boolean","dangerouslyDisableSandbox","_simulatedSedEdit","object","filePath","newContent","inputSchema","omit","InputSchema","ReturnType","BashToolInput","infer","COMMON_BACKGROUND_COMMANDS","const","getCommandTypeForLogging","parts","includes","outputSchema","stdout","stderr","rawOutputPath","interrupted","isImage","backgroundTaskId","backgroundedByUser","assistantAutoBackgrounded","returnCodeInterpretation","noOutputExpected","structuredContent","array","any","persistedOutputPath","persistedOutputSize","OutputSchema","Out","BashProgress","isAutobackgroundingAllowed","detectBlockedSleepPattern","first","m","secs","parseInt","rest","slice","join","SimulatedSedEditResult","data","SimulatedSedEditContext","Pick","applySedEdit","simulatedEdit","toolUseContext","parentMessage","Promise","absoluteFilePath","fs","encoding","originalContent","readFile","e","updateFileHistoryState","uuid","endings","readFileState","set","content","timestamp","offset","undefined","limit","BashTool","name","searchHint","maxResultSizeChars","strict","prompt","isConcurrencySafe","input","isReadOnly","compoundCommandHasCd","result","behavior","toAutoClassifierInput","preparePermissionMatcher","parsed","kind","subcommands","commands","map","c","argv","pattern","prefix","some","cmd","startsWith","isSearchOrReadCommand","safeParse","success","sedInfo","file_path","old_string","CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR","getToolUseSummary","getActivityDescription","desc","validateInput","sleepPattern","message","errorCode","checkPermissions","context","extractSearchText","mapToolResultToToolResultBlockParam","toolUseID","tool_use_id","type","block","processedStdout","replace","trimEnd","preview","filepath","originalSize","isJson","hasMore","errorMessage","backgroundInfo","outputPath","filter","Boolean","is_error","call","_canUseTool","onProgress","abortController","getAppState","setAppState","setToolJSX","stdoutAccumulator","stderrForShellReset","interpretationResult","progressCounter","wasInterrupted","isMainThread","agentId","preventCwdChanges","commandGenerator","runShellCommand","setAppStateForTasks","toolUseId","generatorResult","next","done","progress","value","output","fullOutput","elapsedTimeSeconds","totalLines","totalBytes","taskId","timeoutMs","code","isInterrupt","signal","reason","append","isError","appState","toolPermissionContext","outputWithSbFailures","annotateStderrWithSandboxFailures","preSpawnError","Error","toString","MAX_PERSISTED_SIZE","outputFilePath","outputTaskId","fileStat","size","dest","commandType","command_type","stdout_length","stderr_length","exit_code","codeIndexingTool","tool","source","strippedStdout","extracted","stripped","hints","hint","compressedStdout","resized","isResultTruncated","AbortController","f","prev","AsyncGenerator","lastProgressOutput","lastTotalLines","lastTotalBytes","backgroundShellId","resolveProgress","createProgressSignal","resolve","shouldAutoBackground","shellCommand","lastLines","allLines","isIncomplete","resultPromise","spawnBackgroundTask","handle","startBackgrounding","eventName","backgroundFn","shellId","foregroundTaskId","then","onTimeout","setTimeout","status","unref","startTime","Date","now","initialResult","race","t","r","v","cleanup","startPolling","taskOutput","progressSignal","fixedResult","stdoutToFile","outputFileRedundant","path","outputFileSize","elapsed","elapsedSeconds","Math","floor","jsx","shouldHidePromptInput","shouldContinueAnimation","showSpinner","stopPolling"],"sources":["BashTool.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'\nimport {\n  copyFile,\n  stat as fsStat,\n  truncate as fsTruncate,\n  link,\n} from 'fs/promises'\nimport * as React from 'react'\nimport type { CanUseToolFn } from 'src/hooks/useCanUseTool.js'\nimport type { AppState } from 'src/state/AppState.js'\nimport { z } from 'zod/v4'\nimport { getKairosActive } from '../../bootstrap/state.js'\nimport { TOOL_SUMMARY_MAX_LENGTH } from '../../constants/toolLimits.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from '../../services/analytics/index.js'\nimport { notifyVscodeFileUpdated } from '../../services/mcp/vscodeSdkMcp.js'\nimport type {\n  SetToolJSXFn,\n  ToolCallProgress,\n  ToolUseContext,\n  ValidationResult,\n} from '../../Tool.js'\nimport { buildTool, type ToolDef } from '../../Tool.js'\nimport {\n  backgroundExistingForegroundTask,\n  markTaskNotified,\n  registerForeground,\n  spawnShellTask,\n  unregisterForeground,\n} from '../../tasks/LocalShellTask/LocalShellTask.js'\nimport type { AgentId } from '../../types/ids.js'\nimport type { AssistantMessage } from '../../types/message.js'\nimport { parseForSecurity } from '../../utils/bash/ast.js'\nimport {\n  splitCommand_DEPRECATED,\n  splitCommandWithOperators,\n} from '../../utils/bash/commands.js'\nimport { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js'\nimport { detectCodeIndexingFromCommand } from '../../utils/codeIndexing.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isENOENT, ShellError } from '../../utils/errors.js'\nimport {\n  detectFileEncoding,\n  detectLineEndings,\n  getFileModificationTime,\n  writeTextContent,\n} from '../../utils/file.js'\nimport {\n  fileHistoryEnabled,\n  fileHistoryTrackEdit,\n} from '../../utils/fileHistory.js'\nimport { truncate } from '../../utils/format.js'\nimport { getFsImplementation } from '../../utils/fsOperations.js'\nimport { lazySchema } from '../../utils/lazySchema.js'\nimport { expandPath } from '../../utils/path.js'\nimport type { PermissionResult } from '../../utils/permissions/PermissionResult.js'\nimport { maybeRecordPluginHint } from '../../utils/plugins/hintRecommendation.js'\nimport { exec } from '../../utils/Shell.js'\nimport type { ExecResult } from '../../utils/ShellCommand.js'\nimport { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'\nimport { semanticBoolean } from '../../utils/semanticBoolean.js'\nimport { semanticNumber } from '../../utils/semanticNumber.js'\nimport { EndTruncatingAccumulator } from '../../utils/stringUtils.js'\nimport { getTaskOutputPath } from '../../utils/task/diskOutput.js'\nimport { TaskOutput } from '../../utils/task/TaskOutput.js'\nimport { isOutputLineTruncated } from '../../utils/terminal.js'\nimport {\n  buildLargeToolResultMessage,\n  ensureToolResultsDir,\n  generatePreview,\n  getToolResultPath,\n  PREVIEW_SIZE_BYTES,\n} from '../../utils/toolResultStorage.js'\nimport { userFacingName as fileEditUserFacingName } from '../FileEditTool/UI.js'\nimport { trackGitOperations } from '../shared/gitOperationTracking.js'\nimport {\n  bashToolHasPermission,\n  commandHasAnyCd,\n  matchWildcardPattern,\n  permissionRuleExtractPrefix,\n} from './bashPermissions.js'\nimport { interpretCommandResult } from './commandSemantics.js'\nimport {\n  getDefaultTimeoutMs,\n  getMaxTimeoutMs,\n  getSimplePrompt,\n} from './prompt.js'\nimport { checkReadOnlyConstraints } from './readOnlyValidation.js'\nimport { parseSedEditCommand } from './sedEditParser.js'\nimport { shouldUseSandbox } from './shouldUseSandbox.js'\nimport { BASH_TOOL_NAME } from './toolName.js'\nimport {\n  BackgroundHint,\n  renderToolResultMessage,\n  renderToolUseErrorMessage,\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n} from './UI.js'\nimport {\n  buildImageToolResult,\n  isImageOutput,\n  resetCwdIfOutsideProject,\n  resizeShellImageOutput,\n  stdErrAppendShellResetMessage,\n  stripEmptyLines,\n} from './utils.js'\n\nconst EOL = '\\n'\n\n// Progress display constants\nconst PROGRESS_THRESHOLD_MS = 2000 // Show progress after 2 seconds\n// In assistant mode, blocking bash auto-backgrounds after this many ms in the main agent\nconst ASSISTANT_BLOCKING_BUDGET_MS = 15_000\n\n// Search commands for collapsible display (grep, find, etc.)\nconst BASH_SEARCH_COMMANDS = new Set([\n  'find',\n  'grep',\n  'rg',\n  'ag',\n  'ack',\n  'locate',\n  'which',\n  'whereis',\n])\n\n// Read/view commands for collapsible display (cat, head, etc.)\nconst BASH_READ_COMMANDS = new Set([\n  'cat',\n  'head',\n  'tail',\n  'less',\n  'more',\n  // Analysis commands\n  'wc',\n  'stat',\n  'file',\n  'strings',\n  // Data processing — commonly used to parse/transform file content in pipes\n  'jq',\n  'awk',\n  'cut',\n  'sort',\n  'uniq',\n  'tr',\n])\n\n// Directory-listing commands for collapsible display (ls, tree, du).\n// Split from BASH_READ_COMMANDS so the summary says \"Listed N directories\"\n// instead of the misleading \"Read N files\".\nconst BASH_LIST_COMMANDS = new Set(['ls', 'tree', 'du'])\n\n// Commands that are semantic-neutral in any position — pure output/status commands\n// that don't change the read/search nature of the overall pipeline.\n// e.g. `ls dir && echo \"---\" && ls dir2` is still a read-only compound command.\nconst BASH_SEMANTIC_NEUTRAL_COMMANDS = new Set([\n  'echo',\n  'printf',\n  'true',\n  'false',\n  ':', // bash no-op\n])\n\n// Commands that typically produce no stdout on success\nconst BASH_SILENT_COMMANDS = new Set([\n  'mv',\n  'cp',\n  'rm',\n  'mkdir',\n  'rmdir',\n  'chmod',\n  'chown',\n  'chgrp',\n  'touch',\n  'ln',\n  'cd',\n  'export',\n  'unset',\n  'wait',\n])\n\n/**\n * Checks if a bash command is a search or read operation.\n * Used to determine if the command should be collapsed in the UI.\n * Returns an object indicating whether it's a search or read operation.\n *\n * For pipelines (e.g., `cat file | bq`), ALL parts must be search/read commands\n * for the whole command to be considered collapsible.\n *\n * Semantic-neutral commands (echo, printf, true, false, :) are skipped in any\n * position, as they're pure output/status commands that don't affect the read/search\n * nature of the pipeline (e.g. `ls dir && echo \"---\" && ls dir2` is still a read).\n */\nexport function isSearchOrReadBashCommand(command: string): {\n  isSearch: boolean\n  isRead: boolean\n  isList: boolean\n} {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    // If we can't parse the command due to malformed syntax,\n    // it's not a search/read command\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  if (partsWithOperators.length === 0) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  let hasSearch = false\n  let hasRead = false\n  let hasList = false\n  let hasNonNeutralCommand = false\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {\n      continue\n    }\n\n    hasNonNeutralCommand = true\n\n    const isPartSearch = BASH_SEARCH_COMMANDS.has(baseCommand)\n    const isPartRead = BASH_READ_COMMANDS.has(baseCommand)\n    const isPartList = BASH_LIST_COMMANDS.has(baseCommand)\n\n    if (!isPartSearch && !isPartRead && !isPartList) {\n      return { isSearch: false, isRead: false, isList: false }\n    }\n\n    if (isPartSearch) hasSearch = true\n    if (isPartRead) hasRead = true\n    if (isPartList) hasList = true\n  }\n\n  // Only neutral commands (e.g., just \"echo foo\") -- not collapsible\n  if (!hasNonNeutralCommand) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  return { isSearch: hasSearch, isRead: hasRead, isList: hasList }\n}\n\n/**\n * Checks if a bash command is expected to produce no stdout on success.\n * Used to show \"Done\" instead of \"(No output)\" in the UI.\n */\nfunction isSilentBashCommand(command: string): boolean {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    return false\n  }\n\n  if (partsWithOperators.length === 0) {\n    return false\n  }\n\n  let hasNonFallbackCommand = false\n  let lastOperator: string | null = null\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      lastOperator = part\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (\n      lastOperator === '||' &&\n      BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)\n    ) {\n      continue\n    }\n\n    hasNonFallbackCommand = true\n\n    if (!BASH_SILENT_COMMANDS.has(baseCommand)) {\n      return false\n    }\n  }\n\n  return hasNonFallbackCommand\n}\n\n// Commands that should not be auto-backgrounded\nconst DISALLOWED_AUTO_BACKGROUND_COMMANDS = [\n  'sleep', // Sleep should run in foreground unless explicitly backgrounded by user\n]\n\n// Check if background tasks are disabled at module load time\nconst isBackgroundTasksDisabled =\n  // eslint-disable-next-line custom-rules/no-process-env-top-level -- Intentional: schema must be defined at module load\n  isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_BACKGROUND_TASKS)\n\nconst fullInputSchema = lazySchema(() =>\n  z.strictObject({\n    command: z.string().describe('The command to execute'),\n    timeout: semanticNumber(z.number().optional()).describe(\n      `Optional timeout in milliseconds (max ${getMaxTimeoutMs()})`,\n    ),\n    description: z\n      .string()\n      .optional()\n      .describe(`Clear, concise description of what this command does in active voice. Never use words like \"complex\" or \"risk\" in the description - just describe what it does.\n\nFor simple commands (git, npm, standard CLI tools), keep it brief (5-10 words):\n- ls → \"List files in current directory\"\n- git status → \"Show working tree status\"\n- npm install → \"Install package dependencies\"\n\nFor commands that are harder to parse at a glance (piped commands, obscure flags, etc.), add enough context to clarify what it does:\n- find . -name \"*.tmp\" -exec rm {} \\\\; → \"Find and delete all .tmp files recursively\"\n- git reset --hard origin/main → \"Discard all local changes and match remote main\"\n- curl -s url | jq '.data[]' → \"Fetch JSON from URL and extract data array elements\"`),\n    run_in_background: semanticBoolean(z.boolean().optional()).describe(\n      `Set to true to run this command in the background. Use Read to read the output later.`,\n    ),\n    dangerouslyDisableSandbox: semanticBoolean(z.boolean().optional()).describe(\n      'Set this to true to dangerously override sandbox mode and run commands without sandboxing.',\n    ),\n    _simulatedSedEdit: z\n      .object({\n        filePath: z.string(),\n        newContent: z.string(),\n      })\n      .optional()\n      .describe('Internal: pre-computed sed edit result from preview'),\n  }),\n)\n\n// Always omit _simulatedSedEdit from the model-facing schema. It is an internal-only\n// field set by SedEditPermissionRequest after the user approves a sed edit preview.\n// Exposing it in the schema would let the model bypass permission checks and the\n// sandbox by pairing an innocuous command with an arbitrary file write.\n// Also conditionally remove run_in_background when background tasks are disabled.\nconst inputSchema = lazySchema(() =>\n  isBackgroundTasksDisabled\n    ? fullInputSchema().omit({\n        run_in_background: true,\n        _simulatedSedEdit: true,\n      })\n    : fullInputSchema().omit({ _simulatedSedEdit: true }),\n)\ntype InputSchema = ReturnType<typeof inputSchema>\n\n// Use fullInputSchema for the type to always include run_in_background\n// (even when it's omitted from the schema, the code needs to handle it)\nexport type BashToolInput = z.infer<ReturnType<typeof fullInputSchema>>\n\nconst COMMON_BACKGROUND_COMMANDS = [\n  'npm',\n  'yarn',\n  'pnpm',\n  'node',\n  'python',\n  'python3',\n  'go',\n  'cargo',\n  'make',\n  'docker',\n  'terraform',\n  'webpack',\n  'vite',\n  'jest',\n  'pytest',\n  'curl',\n  'wget',\n  'build',\n  'test',\n  'serve',\n  'watch',\n  'dev',\n] as const\n\nfunction getCommandTypeForLogging(\n  command: string,\n): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0)\n    return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n\n  // Check each part of the command to see if any match common background commands\n  for (const part of parts) {\n    const baseCommand = part.split(' ')[0] || ''\n    if (\n      COMMON_BACKGROUND_COMMANDS.includes(\n        baseCommand as (typeof COMMON_BACKGROUND_COMMANDS)[number],\n      )\n    ) {\n      return baseCommand as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    }\n  }\n\n  return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n}\n\nconst outputSchema = lazySchema(() =>\n  z.object({\n    stdout: z.string().describe('The standard output of the command'),\n    stderr: z.string().describe('The standard error output of the command'),\n    rawOutputPath: z\n      .string()\n      .optional()\n      .describe('Path to raw output file for large MCP tool outputs'),\n    interrupted: z.boolean().describe('Whether the command was interrupted'),\n    isImage: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if stdout contains image data'),\n    backgroundTaskId: z\n      .string()\n      .optional()\n      .describe(\n        'ID of the background task if command is running in background',\n      ),\n    backgroundedByUser: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if the user manually backgrounded the command with Ctrl+B',\n      ),\n    assistantAutoBackgrounded: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if assistant-mode auto-backgrounded a long-running blocking command',\n      ),\n    dangerouslyDisableSandbox: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if sandbox mode was overridden'),\n    returnCodeInterpretation: z\n      .string()\n      .optional()\n      .describe(\n        'Semantic interpretation for non-error exit codes with special meaning',\n      ),\n    noOutputExpected: z\n      .boolean()\n      .optional()\n      .describe(\n        'Whether the command is expected to produce no output on success',\n      ),\n    structuredContent: z\n      .array(z.any())\n      .optional()\n      .describe('Structured content blocks'),\n    persistedOutputPath: z\n      .string()\n      .optional()\n      .describe(\n        'Path to the persisted full output in tool-results dir (set when output is too large for inline)',\n      ),\n    persistedOutputSize: z\n      .number()\n      .optional()\n      .describe(\n        'Total size of the output in bytes (set when output is too large for inline)',\n      ),\n  }),\n)\n\ntype OutputSchema = ReturnType<typeof outputSchema>\nexport type Out = z.infer<OutputSchema>\n\n// Re-export BashProgress from centralized types to break import cycles\nexport type { BashProgress } from '../../types/tools.js'\n\nimport type { BashProgress } from '../../types/tools.js'\n\n/**\n * Checks if a command is allowed to be automatically backgrounded\n * @param command The command to check\n * @returns false for commands that should not be auto-backgrounded (like sleep)\n */\nfunction isAutobackgroundingAllowed(command: string): boolean {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return true\n\n  // Get the first part which should be the base command\n  const baseCommand = parts[0]?.trim()\n  if (!baseCommand) return true\n\n  return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(baseCommand)\n}\n\n/**\n * Detect standalone or leading `sleep N` patterns that should use Monitor\n * instead. Catches `sleep 5`, `sleep 5 && check`, `sleep 5; check` — but\n * not sleep inside pipelines, subshells, or scripts (those are fine).\n */\nexport function detectBlockedSleepPattern(command: string): string | null {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return null\n\n  const first = parts[0]?.trim() ?? ''\n  // Bare `sleep N` or `sleep N.N` as the first subcommand.\n  // Float durations (sleep 0.5) are allowed — those are legit pacing, not polls.\n  const m = /^sleep\\s+(\\d+)\\s*$/.exec(first)\n  if (!m) return null\n  const secs = parseInt(m[1]!, 10)\n  if (secs < 2) return null // sub-2s sleeps are fine (rate limiting, pacing)\n\n  // `sleep N` alone → \"what are you waiting for?\"\n  // `sleep N && check` → \"use Monitor { command: check }\"\n  const rest = parts.slice(1).join(' ').trim()\n  return rest\n    ? `sleep ${secs} followed by: ${rest}`\n    : `standalone sleep ${secs}`\n}\n\n/**\n * Checks if a command contains tools that shouldn't run in sandbox\n * This includes:\n * - Dynamic config-based disabled commands and substrings (tengu_sandbox_disabled_commands)\n * - User-configured commands from settings.json (sandbox.excludedCommands)\n *\n * User-configured commands support the same pattern syntax as permission rules:\n * - Exact matches: \"npm run lint\"\n * - Prefix patterns: \"npm run test:*\"\n */\n\ntype SimulatedSedEditResult = {\n  data: Out\n}\n\ntype SimulatedSedEditContext = Pick<\n  ToolUseContext,\n  'readFileState' | 'updateFileHistoryState'\n>\n\n/**\n * Applies a simulated sed edit directly instead of running sed.\n * This is used by the permission dialog to ensure what the user previews\n * is exactly what gets written to the file.\n */\nasync function applySedEdit(\n  simulatedEdit: { filePath: string; newContent: string },\n  toolUseContext: SimulatedSedEditContext,\n  parentMessage?: AssistantMessage,\n): Promise<SimulatedSedEditResult> {\n  const { filePath, newContent } = simulatedEdit\n  const absoluteFilePath = expandPath(filePath)\n  const fs = getFsImplementation()\n\n  // Read original content for VS Code notification\n  const encoding = detectFileEncoding(absoluteFilePath)\n  let originalContent: string\n  try {\n    originalContent = await fs.readFile(absoluteFilePath, { encoding })\n  } catch (e) {\n    if (isENOENT(e)) {\n      return {\n        data: {\n          stdout: '',\n          stderr: `sed: ${filePath}: No such file or directory\\nExit code 1`,\n          interrupted: false,\n        },\n      }\n    }\n    throw e\n  }\n\n  // Track file history before making changes (for undo support)\n  if (fileHistoryEnabled() && parentMessage) {\n    await fileHistoryTrackEdit(\n      toolUseContext.updateFileHistoryState,\n      absoluteFilePath,\n      parentMessage.uuid,\n    )\n  }\n\n  // Detect line endings and write new content\n  const endings = detectLineEndings(absoluteFilePath)\n  writeTextContent(absoluteFilePath, newContent, encoding, endings)\n\n  // Notify VS Code about the file change\n  notifyVscodeFileUpdated(absoluteFilePath, originalContent, newContent)\n\n  // Update read timestamp to invalidate stale writes\n  toolUseContext.readFileState.set(absoluteFilePath, {\n    content: newContent,\n    timestamp: getFileModificationTime(absoluteFilePath),\n    offset: undefined,\n    limit: undefined,\n  })\n\n  // Return success result matching sed output format (sed produces no output on success)\n  return {\n    data: {\n      stdout: '',\n      stderr: '',\n      interrupted: false,\n    },\n  }\n}\n\nexport const BashTool = buildTool({\n  name: BASH_TOOL_NAME,\n  searchHint: 'execute shell commands',\n  // 30K chars - tool result persistence threshold\n  maxResultSizeChars: 30_000,\n  strict: true,\n  async description({ description }) {\n    return description || 'Run shell command'\n  },\n  async prompt() {\n    return getSimplePrompt()\n  },\n  isConcurrencySafe(input) {\n    return this.isReadOnly?.(input) ?? false\n  },\n  isReadOnly(input) {\n    const compoundCommandHasCd = commandHasAnyCd(input.command)\n    const result = checkReadOnlyConstraints(input, compoundCommandHasCd)\n    return result.behavior === 'allow'\n  },\n  toAutoClassifierInput(input) {\n    return input.command\n  },\n  async preparePermissionMatcher({ command }) {\n    // Hook `if` filtering is \"no match → skip hook\" (deny-like semantics), so\n    // compound commands must fire the hook if ANY subcommand matches. Without\n    // splitting, `ls && git push` would bypass a `Bash(git *)` security hook.\n    const parsed = await parseForSecurity(command)\n    if (parsed.kind !== 'simple') {\n      // parse-unavailable / too-complex: fail safe by running the hook.\n      return () => true\n    }\n    // Match on argv (strips leading VAR=val) so `FOO=bar git push` still\n    // matches `Bash(git *)`.\n    const subcommands = parsed.commands.map(c => c.argv.join(' '))\n    return pattern => {\n      const prefix = permissionRuleExtractPrefix(pattern)\n      return subcommands.some(cmd => {\n        if (prefix !== null) {\n          return cmd === prefix || cmd.startsWith(`${prefix} `)\n        }\n        return matchWildcardPattern(pattern, cmd)\n      })\n    }\n  },\n  isSearchOrReadCommand(input) {\n    const parsed = inputSchema().safeParse(input)\n    if (!parsed.success)\n      return { isSearch: false, isRead: false, isList: false }\n    return isSearchOrReadBashCommand(parsed.data.command)\n  },\n  get inputSchema(): InputSchema {\n    return inputSchema()\n  },\n  get outputSchema(): OutputSchema {\n    return outputSchema()\n  },\n  userFacingName(input) {\n    if (!input) {\n      return 'Bash'\n    }\n    // Render sed in-place edits as file edits\n    if (input.command) {\n      const sedInfo = parseSedEditCommand(input.command)\n      if (sedInfo) {\n        return fileEditUserFacingName({\n          file_path: sedInfo.filePath,\n          old_string: 'x',\n        })\n      }\n    }\n    // Env var FIRST: shouldUseSandbox → splitCommand_DEPRECATED → shell-quote's\n    // `new RegExp` per call. userFacingName runs per-render for every bash\n    // message in history; with ~50 msgs + one slow-to-tokenize command, this\n    // exceeds the shimmer tick → transition abort → infinite retry (#21605).\n    return isEnvTruthy(process.env.CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR) &&\n      shouldUseSandbox(input)\n      ? 'SandboxedBash'\n      : 'Bash'\n  },\n  getToolUseSummary(input) {\n    if (!input?.command) {\n      return null\n    }\n    const { command, description } = input\n    if (description) {\n      return description\n    }\n    return truncate(command, TOOL_SUMMARY_MAX_LENGTH)\n  },\n  getActivityDescription(input) {\n    if (!input?.command) {\n      return 'Running command'\n    }\n    const desc =\n      input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH)\n    return `Running ${desc}`\n  },\n  async validateInput(input: BashToolInput): Promise<ValidationResult> {\n    if (\n      feature('MONITOR_TOOL') &&\n      !isBackgroundTasksDisabled &&\n      !input.run_in_background\n    ) {\n      const sleepPattern = detectBlockedSleepPattern(input.command)\n      if (sleepPattern !== null) {\n        return {\n          result: false,\n          message: `Blocked: ${sleepPattern}. Run blocking commands in the background with run_in_background: true — you'll get a completion notification when done. For streaming events (watching logs, polling APIs), use the Monitor tool. If you genuinely need a delay (rate limiting, deliberate pacing), keep it under 2 seconds.`,\n          errorCode: 10,\n        }\n      }\n    }\n    return { result: true }\n  },\n  async checkPermissions(input, context): Promise<PermissionResult> {\n    return bashToolHasPermission(input, context)\n  },\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n  renderToolResultMessage,\n  // BashToolResultMessage shows <OutputLine content={stdout}> + stderr.\n  // UI never shows persistedOutputPath wrapper, backgroundInfo — those are\n  // model-facing (mapToolResult... below).\n  extractSearchText({ stdout, stderr }) {\n    return stderr ? `${stdout}\\n${stderr}` : stdout\n  },\n  mapToolResultToToolResultBlockParam(\n    {\n      interrupted,\n      stdout,\n      stderr,\n      isImage,\n      backgroundTaskId,\n      backgroundedByUser,\n      assistantAutoBackgrounded,\n      structuredContent,\n      persistedOutputPath,\n      persistedOutputSize,\n    },\n    toolUseID,\n  ): ToolResultBlockParam {\n    // Handle structured content\n    if (structuredContent && structuredContent.length > 0) {\n      return {\n        tool_use_id: toolUseID,\n        type: 'tool_result',\n        content: structuredContent,\n      }\n    }\n\n    // For image data, format as image content block for Claude\n    if (isImage) {\n      const block = buildImageToolResult(stdout, toolUseID)\n      if (block) return block\n    }\n\n    let processedStdout = stdout\n    if (stdout) {\n      // Replace any leading newlines or lines with only whitespace\n      processedStdout = stdout.replace(/^(\\s*\\n)+/, '')\n      // Still trim the end as before\n      processedStdout = processedStdout.trimEnd()\n    }\n\n    // For large output that was persisted to disk, build <persisted-output>\n    // message for the model. The UI never sees this — it uses data.stdout.\n    if (persistedOutputPath) {\n      const preview = generatePreview(processedStdout, PREVIEW_SIZE_BYTES)\n      processedStdout = buildLargeToolResultMessage({\n        filepath: persistedOutputPath,\n        originalSize: persistedOutputSize ?? 0,\n        isJson: false,\n        preview: preview.preview,\n        hasMore: preview.hasMore,\n      })\n    }\n\n    let errorMessage = stderr.trim()\n    if (interrupted) {\n      if (stderr) errorMessage += EOL\n      errorMessage += '<error>Command was aborted before completion</error>'\n    }\n\n    let backgroundInfo = ''\n    if (backgroundTaskId) {\n      const outputPath = getTaskOutputPath(backgroundTaskId)\n      if (assistantAutoBackgrounded) {\n        backgroundInfo = `Command exceeded the assistant-mode blocking budget (${ASSISTANT_BLOCKING_BUDGET_MS / 1000}s) and was moved to the background with ID: ${backgroundTaskId}. It is still running — you will be notified when it completes. Output is being written to: ${outputPath}. In assistant mode, delegate long-running work to a subagent or use run_in_background to keep this conversation responsive.`\n      } else if (backgroundedByUser) {\n        backgroundInfo = `Command was manually backgrounded by user with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      } else {\n        backgroundInfo = `Command running in background with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      }\n    }\n\n    return {\n      tool_use_id: toolUseID,\n      type: 'tool_result',\n      content: [processedStdout, errorMessage, backgroundInfo]\n        .filter(Boolean)\n        .join('\\n'),\n      is_error: interrupted,\n    }\n  },\n  async call(\n    input: BashToolInput,\n    toolUseContext,\n    _canUseTool?: CanUseToolFn,\n    parentMessage?: AssistantMessage,\n    onProgress?: ToolCallProgress<BashProgress>,\n  ) {\n    // Handle simulated sed edit - apply directly instead of running sed\n    // This ensures what the user previewed is exactly what gets written\n    if (input._simulatedSedEdit) {\n      return applySedEdit(\n        input._simulatedSedEdit,\n        toolUseContext,\n        parentMessage,\n      )\n    }\n\n    const { abortController, getAppState, setAppState, setToolJSX } =\n      toolUseContext\n\n    const stdoutAccumulator = new EndTruncatingAccumulator()\n    let stderrForShellReset = ''\n    let interpretationResult:\n      | ReturnType<typeof interpretCommandResult>\n      | undefined\n\n    let progressCounter = 0\n    let wasInterrupted = false\n    let result: ExecResult\n\n    const isMainThread = !toolUseContext.agentId\n    const preventCwdChanges = !isMainThread\n\n    try {\n      // Use the new async generator version of runShellCommand\n      const commandGenerator = runShellCommand({\n        input,\n        abortController,\n        // Use the always-shared task channel so async agents' background\n        // bash tasks are actually registered (and killable on agent exit).\n        setAppState: toolUseContext.setAppStateForTasks ?? setAppState,\n        setToolJSX,\n        preventCwdChanges,\n        isMainThread,\n        toolUseId: toolUseContext.toolUseId,\n        agentId: toolUseContext.agentId,\n      })\n\n      // Consume the generator and capture the return value\n      let generatorResult\n      do {\n        generatorResult = await commandGenerator.next()\n        if (!generatorResult.done && onProgress) {\n          const progress = generatorResult.value\n          onProgress({\n            toolUseID: `bash-progress-${progressCounter++}`,\n            data: {\n              type: 'bash_progress',\n              output: progress.output,\n              fullOutput: progress.fullOutput,\n              elapsedTimeSeconds: progress.elapsedTimeSeconds,\n              totalLines: progress.totalLines,\n              totalBytes: progress.totalBytes,\n              taskId: progress.taskId,\n              timeoutMs: progress.timeoutMs,\n            },\n          })\n        }\n      } while (!generatorResult.done)\n\n      // Get the final result from the generator's return value\n      result = generatorResult.value\n\n      trackGitOperations(input.command, result.code, result.stdout)\n\n      const isInterrupt =\n        result.interrupted && abortController.signal.reason === 'interrupt'\n\n      // stderr is interleaved in stdout (merged fd) — result.stdout has both\n      stdoutAccumulator.append((result.stdout || '').trimEnd() + EOL)\n\n      // Interpret the command result using semantic rules\n      interpretationResult = interpretCommandResult(\n        input.command,\n        result.code,\n        result.stdout || '',\n        '',\n      )\n\n      // Check for git index.lock error (stderr is in stdout now)\n      if (\n        result.stdout &&\n        result.stdout.includes(\".git/index.lock': File exists\")\n      ) {\n        logEvent('tengu_git_index_lock_error', {})\n      }\n\n      if (interpretationResult.isError && !isInterrupt) {\n        // Only add exit code if it's actually an error\n        if (result.code !== 0) {\n          stdoutAccumulator.append(`Exit code ${result.code}`)\n        }\n      }\n\n      if (!preventCwdChanges) {\n        const appState = getAppState()\n        if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {\n          stderrForShellReset = stdErrAppendShellResetMessage('')\n        }\n      }\n\n      // Annotate output with sandbox violations if any (stderr is in stdout)\n      const outputWithSbFailures =\n        SandboxManager.annotateStderrWithSandboxFailures(\n          input.command,\n          result.stdout || '',\n        )\n\n      if (result.preSpawnError) {\n        throw new Error(result.preSpawnError)\n      }\n      if (interpretationResult.isError && !isInterrupt) {\n        // stderr is merged into stdout (merged fd); outputWithSbFailures\n        // already has the full output. Pass '' for stdout to avoid\n        // duplication in getErrorParts() and processBashCommand.\n        throw new ShellError(\n          '',\n          outputWithSbFailures,\n          result.code,\n          result.interrupted,\n        )\n      }\n      wasInterrupted = result.interrupted\n    } finally {\n      if (setToolJSX) setToolJSX(null)\n    }\n\n    // Get final string from accumulator\n    const stdout = stdoutAccumulator.toString()\n\n    // Large output: the file on disk has more than getMaxOutputLength() bytes.\n    // stdout already contains the first chunk (from getStdout()). Copy the\n    // output file to the tool-results dir so the model can read it via\n    // FileRead. If > 64 MB, truncate after copying.\n    const MAX_PERSISTED_SIZE = 64 * 1024 * 1024\n    let persistedOutputPath: string | undefined\n    let persistedOutputSize: number | undefined\n    if (result.outputFilePath && result.outputTaskId) {\n      try {\n        const fileStat = await fsStat(result.outputFilePath)\n        persistedOutputSize = fileStat.size\n\n        await ensureToolResultsDir()\n        const dest = getToolResultPath(result.outputTaskId, false)\n        if (fileStat.size > MAX_PERSISTED_SIZE) {\n          await fsTruncate(result.outputFilePath, MAX_PERSISTED_SIZE)\n        }\n        try {\n          await link(result.outputFilePath, dest)\n        } catch {\n          await copyFile(result.outputFilePath, dest)\n        }\n        persistedOutputPath = dest\n      } catch {\n        // File may already be gone — stdout preview is sufficient\n      }\n    }\n\n    const commandType = input.command.split(' ')[0]\n\n    logEvent('tengu_bash_tool_command_executed', {\n      command_type:\n        commandType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      stdout_length: stdout.length,\n      stderr_length: 0,\n      exit_code: result.code,\n      interrupted: wasInterrupted,\n    })\n\n    // Log code indexing tool usage\n    const codeIndexingTool = detectCodeIndexingFromCommand(input.command)\n    if (codeIndexingTool) {\n      logEvent('tengu_code_indexing_tool_used', {\n        tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        source:\n          'cli' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        success: result.code === 0,\n      })\n    }\n\n    let strippedStdout = stripEmptyLines(stdout)\n\n    // Claude Code hints protocol: CLIs/SDKs gated on CLAUDECODE=1 emit a\n    // `<claude-code-hint />` tag to stderr (merged into stdout here). Scan,\n    // record for useClaudeCodeHintRecommendation to surface, then strip\n    // so the model never sees the tag — a zero-token side channel.\n    // Stripping runs unconditionally (subagent output must stay clean too);\n    // only the dialog recording is main-thread-only.\n    const extracted = extractClaudeCodeHints(strippedStdout, input.command)\n    strippedStdout = extracted.stripped\n    if (isMainThread && extracted.hints.length > 0) {\n      for (const hint of extracted.hints) maybeRecordPluginHint(hint)\n    }\n\n    let isImage = isImageOutput(strippedStdout)\n\n    // Cap image dimensions + size if present (CC-304 — see\n    // resizeShellImageOutput). Scope the decoded buffer so it can be reclaimed\n    // before we build the output Out object.\n    let compressedStdout = strippedStdout\n    if (isImage) {\n      const resized = await resizeShellImageOutput(\n        strippedStdout,\n        result.outputFilePath,\n        persistedOutputSize,\n      )\n      if (resized) {\n        compressedStdout = resized\n      } else {\n        // Parse failed or file too large (e.g. exceeds MAX_IMAGE_FILE_SIZE).\n        // Keep isImage in sync with what we actually send so the UI label stays\n        // accurate — mapToolResultToToolResultBlockParam's defensive\n        // fallthrough will send text, not an image block.\n        isImage = false\n      }\n    }\n\n    const data: Out = {\n      stdout: compressedStdout,\n      stderr: stderrForShellReset,\n      interrupted: wasInterrupted,\n      isImage,\n      returnCodeInterpretation: interpretationResult?.message,\n      noOutputExpected: isSilentBashCommand(input.command),\n      backgroundTaskId: result.backgroundTaskId,\n      backgroundedByUser: result.backgroundedByUser,\n      assistantAutoBackgrounded: result.assistantAutoBackgrounded,\n      dangerouslyDisableSandbox:\n        'dangerouslyDisableSandbox' in input\n          ? (input.dangerouslyDisableSandbox as boolean | undefined)\n          : undefined,\n      persistedOutputPath,\n      persistedOutputSize,\n    }\n\n    return {\n      data,\n    }\n  },\n  renderToolUseErrorMessage,\n  isResultTruncated(output: Out): boolean {\n    return (\n      isOutputLineTruncated(output.stdout) ||\n      isOutputLineTruncated(output.stderr)\n    )\n  },\n} satisfies ToolDef<InputSchema, Out, BashProgress>)\n\nasync function* runShellCommand({\n  input,\n  abortController,\n  setAppState,\n  setToolJSX,\n  preventCwdChanges,\n  isMainThread,\n  toolUseId,\n  agentId,\n}: {\n  input: BashToolInput\n  abortController: AbortController\n  setAppState: (f: (prev: AppState) => AppState) => void\n  setToolJSX?: SetToolJSXFn\n  preventCwdChanges?: boolean\n  isMainThread?: boolean\n  toolUseId?: string\n  agentId?: AgentId\n}): AsyncGenerator<\n  {\n    type: 'progress'\n    output: string\n    fullOutput: string\n    elapsedTimeSeconds: number\n    totalLines: number\n    totalBytes?: number\n    taskId?: string\n    timeoutMs?: number\n  },\n  ExecResult,\n  void\n> {\n  const { command, description, timeout, run_in_background } = input\n  const timeoutMs = timeout || getDefaultTimeoutMs()\n\n  let fullOutput = ''\n  let lastProgressOutput = ''\n  let lastTotalLines = 0\n  let lastTotalBytes = 0\n  let backgroundShellId: string | undefined = undefined\n  let assistantAutoBackgrounded = false\n\n  // Progress signal: resolved by onProgress callback from the shared poller,\n  // waking the generator to yield a progress update.\n  let resolveProgress: (() => void) | null = null\n  function createProgressSignal(): Promise<null> {\n    return new Promise<null>(resolve => {\n      resolveProgress = () => resolve(null)\n    })\n  }\n\n  // Determine if auto-backgrounding should be enabled\n  // Only enable for commands that are allowed to be auto-backgrounded\n  // and when background tasks are not disabled\n  const shouldAutoBackground =\n    !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command)\n\n  const shellCommand = await exec(command, abortController.signal, 'bash', {\n    timeout: timeoutMs,\n    onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {\n      lastProgressOutput = lastLines\n      fullOutput = allLines\n      lastTotalLines = totalLines\n      lastTotalBytes = isIncomplete ? totalBytes : 0\n      // Wake the generator so it yields the new progress data\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n    },\n    preventCwdChanges,\n    shouldUseSandbox: shouldUseSandbox(input),\n    shouldAutoBackground,\n  })\n\n  // Start the command execution\n  const resultPromise = shellCommand.result\n\n  // Helper to spawn a background task and return its ID\n  async function spawnBackgroundTask(): Promise<string> {\n    const handle = await spawnShellTask(\n      {\n        command,\n        description: description || command,\n        shellCommand,\n        toolUseId,\n        agentId,\n      },\n      {\n        abortController,\n        getAppState: () => {\n          // We don't have direct access to getAppState here, but spawn doesn't\n          // actually use it during the spawn process\n          throw new Error(\n            'getAppState not available in runShellCommand context',\n          )\n        },\n        setAppState,\n      },\n    )\n    return handle.taskId\n  }\n\n  // Helper to start backgrounding with optional logging\n  function startBackgrounding(\n    eventName: string,\n    backgroundFn?: (shellId: string) => void,\n  ): void {\n    // If a foreground task is already registered (via registerForeground in the\n    // progress loop), background it in-place instead of re-spawning. Re-spawning\n    // would overwrite tasks[taskId], emit a duplicate task_started SDK event,\n    // and leak the first cleanup callback.\n    if (foregroundTaskId) {\n      if (\n        !backgroundExistingForegroundTask(\n          foregroundTaskId,\n          shellCommand,\n          description || command,\n          setAppState,\n          toolUseId,\n        )\n      ) {\n        return\n      }\n      backgroundShellId = foregroundTaskId\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n      backgroundFn?.(foregroundTaskId)\n      return\n    }\n\n    // No foreground task registered — spawn a new background task\n    // Note: spawn is essentially synchronous despite being async\n    void spawnBackgroundTask().then(shellId => {\n      backgroundShellId = shellId\n\n      // Wake the generator's Promise.race so it sees backgroundShellId.\n      // Without this, if the poller has stopped ticking for this task\n      // (no output + shared-poller race with sibling stopPolling calls)\n      // and the process is hung on I/O, the race at line ~1357 never\n      // resolves and the generator deadlocks despite being backgrounded.\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n\n      if (backgroundFn) {\n        backgroundFn(shellId)\n      }\n    })\n  }\n\n  // Set up auto-backgrounding on timeout if enabled\n  // Only background commands that are allowed to be auto-backgrounded (not sleep, etc.)\n  if (shellCommand.onTimeout && shouldAutoBackground) {\n    shellCommand.onTimeout(backgroundFn => {\n      startBackgrounding(\n        'tengu_bash_command_timeout_backgrounded',\n        backgroundFn,\n      )\n    })\n  }\n\n  // In assistant mode, the main agent should stay responsive. Auto-background\n  // blocking commands after ASSISTANT_BLOCKING_BUDGET_MS so the agent can keep\n  // coordinating instead of waiting. The command keeps running — no state loss.\n  if (\n    feature('KAIROS') &&\n    getKairosActive() &&\n    isMainThread &&\n    !isBackgroundTasksDisabled &&\n    run_in_background !== true\n  ) {\n    setTimeout(() => {\n      if (\n        shellCommand.status === 'running' &&\n        backgroundShellId === undefined\n      ) {\n        assistantAutoBackgrounded = true\n        startBackgrounding('tengu_bash_command_assistant_auto_backgrounded')\n      }\n    }, ASSISTANT_BLOCKING_BUDGET_MS).unref()\n  }\n\n  // Handle Claude asking to run it in the background explicitly\n  // When explicitly requested via run_in_background, always honor the request\n  // regardless of the command type (isAutobackgroundingAllowed only applies to automatic backgrounding)\n  // Skip if background tasks are disabled - run in foreground instead\n  if (run_in_background === true && !isBackgroundTasksDisabled) {\n    const shellId = await spawnBackgroundTask()\n\n    logEvent('tengu_bash_command_explicitly_backgrounded', {\n      command_type: getCommandTypeForLogging(command),\n    })\n\n    return {\n      stdout: '',\n      stderr: '',\n      code: 0,\n      interrupted: false,\n      backgroundTaskId: shellId,\n    }\n  }\n\n  // Wait for the initial threshold before showing progress\n  const startTime = Date.now()\n  let foregroundTaskId: string | undefined = undefined\n\n  {\n    const initialResult = await Promise.race([\n      resultPromise,\n      new Promise<null>(resolve => {\n        const t = setTimeout(\n          (r: (v: null) => void) => r(null),\n          PROGRESS_THRESHOLD_MS,\n          resolve,\n        )\n        t.unref()\n      }),\n    ])\n\n    if (initialResult !== null) {\n      shellCommand.cleanup()\n      return initialResult\n    }\n\n    if (backgroundShellId) {\n      return {\n        stdout: '',\n        stderr: '',\n        code: 0,\n        interrupted: false,\n        backgroundTaskId: backgroundShellId,\n        assistantAutoBackgrounded,\n      }\n    }\n  }\n\n  // Start polling the output file for progress. The poller's #tick calls\n  // onProgress every second, which resolves progressSignal below.\n  TaskOutput.startPolling(shellCommand.taskOutput.taskId)\n\n  // Progress loop: wake is driven by the shared poller calling onProgress,\n  // which resolves the progressSignal.\n  try {\n    while (true) {\n      const progressSignal = createProgressSignal()\n      const result = await Promise.race([resultPromise, progressSignal])\n\n      if (result !== null) {\n        // Race: backgrounding fired (15s timer / onTimeout / Ctrl+B) but the\n        // command completed before the next poll tick. #handleExit sets\n        // backgroundTaskId but skips outputFilePath (it assumes the background\n        // message or <task_notification> will carry the path). Strip\n        // backgroundTaskId so the model sees a clean completed command,\n        // reconstruct outputFilePath for large outputs, and suppress the\n        // redundant <task_notification> from the .then() handler.\n        // Check result.backgroundTaskId (not the closure var) to also cover\n        // Ctrl+B, which calls shellCommand.background() directly.\n        if (result.backgroundTaskId !== undefined) {\n          markTaskNotified(result.backgroundTaskId, setAppState)\n          const fixedResult: ExecResult = {\n            ...result,\n            backgroundTaskId: undefined,\n          }\n          // Mirror ShellCommand.#handleExit's large-output branch that was\n          // skipped because #backgroundTaskId was set.\n          const { taskOutput } = shellCommand\n          if (taskOutput.stdoutToFile && !taskOutput.outputFileRedundant) {\n            fixedResult.outputFilePath = taskOutput.path\n            fixedResult.outputFileSize = taskOutput.outputFileSize\n            fixedResult.outputTaskId = taskOutput.taskId\n          }\n          shellCommand.cleanup()\n          return fixedResult\n        }\n        // Command has completed - return the actual result\n        // If we registered as a foreground task, unregister it\n        if (foregroundTaskId) {\n          unregisterForeground(foregroundTaskId, setAppState)\n        }\n        // Clean up stream resources for foreground commands\n        // (backgrounded commands are cleaned up by LocalShellTask)\n        shellCommand.cleanup()\n        return result\n      }\n\n      // Check if command was backgrounded (either via old mechanism or new backgroundAll)\n      if (backgroundShellId) {\n        return {\n          stdout: '',\n          stderr: '',\n          code: 0,\n          interrupted: false,\n          backgroundTaskId: backgroundShellId,\n          assistantAutoBackgrounded,\n        }\n      }\n\n      // Check if this foreground task was backgrounded via backgroundAll()\n      if (foregroundTaskId) {\n        // shellCommand.status becomes 'backgrounded' when background() is called\n        if (shellCommand.status === 'backgrounded') {\n          return {\n            stdout: '',\n            stderr: '',\n            code: 0,\n            interrupted: false,\n            backgroundTaskId: foregroundTaskId,\n            backgroundedByUser: true,\n          }\n        }\n      }\n\n      // Time for a progress update\n      const elapsed = Date.now() - startTime\n      const elapsedSeconds = Math.floor(elapsed / 1000)\n\n      // Show minimal backgrounding UI if available\n      // Skip if background tasks are disabled\n      if (\n        !isBackgroundTasksDisabled &&\n        backgroundShellId === undefined &&\n        elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 &&\n        setToolJSX\n      ) {\n        // Register this command as a foreground task so it can be backgrounded via Ctrl+B\n        if (!foregroundTaskId) {\n          foregroundTaskId = registerForeground(\n            {\n              command,\n              description: description || command,\n              shellCommand,\n              agentId,\n            },\n            setAppState,\n            toolUseId,\n          )\n        }\n\n        setToolJSX({\n          jsx: <BackgroundHint />,\n          shouldHidePromptInput: false,\n          shouldContinueAnimation: true,\n          showSpinner: true,\n        })\n      }\n      yield {\n        type: 'progress',\n        fullOutput,\n        output: lastProgressOutput,\n        elapsedTimeSeconds: elapsedSeconds,\n        totalLines: lastTotalLines,\n        totalBytes: lastTotalBytes,\n        taskId: shellCommand.taskOutput.taskId,\n        ...(timeout ? { timeoutMs } : undefined),\n      }\n    }\n  } finally {\n    TaskOutput.stopPolling(shellCommand.taskOutput.taskId)\n  }\n}\n"],"mappings":"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","ignoreList":[]}