π File detail
tools/BashTool/BashTool.tsx
π― Use case
This module implements the βBashToolβ tool (Bash) β something the model can call at runtime alongside other agent tools. On the API surface it exposes isSearchOrReadBashCommand, BashToolInput, Out, detectBlockedSleepPattern, and BashTool β mainly types, interfaces, or factory objects. Dependencies touch bun:bundle, @anthropic-ai, Node filesystem, and React UI. It composes internal code from bootstrap, constants, services, Tool, and tasks (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { feature } from 'bun:bundle'; import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'; import { copyFile, stat as fsStat, truncate as fsTruncate, link } from 'fs/promises'; import * as React from 'react'; import type { CanUseToolFn } from 'src/hooks/useCanUseTool.js';
π€ Exports (heuristic)
isSearchOrReadBashCommandBashToolInputOutdetectBlockedSleepPatternBashTool
π External import roots
Package roots from from "β¦" (relative paths omitted).
bun:bundle@anthropic-aifsreactsrczod
π₯οΈ Source preview
import { feature } from 'bun:bundle';
import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs';
import { copyFile, stat as fsStat, truncate as fsTruncate, link } from 'fs/promises';
import * as React from 'react';
import type { CanUseToolFn } from 'src/hooks/useCanUseTool.js';
import type { AppState } from 'src/state/AppState.js';
import { z } from 'zod/v4';
import { getKairosActive } from '../../bootstrap/state.js';
import { TOOL_SUMMARY_MAX_LENGTH } from '../../constants/toolLimits.js';
import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from '../../services/analytics/index.js';
import { notifyVscodeFileUpdated } from '../../services/mcp/vscodeSdkMcp.js';
import type { SetToolJSXFn, ToolCallProgress, ToolUseContext, ValidationResult } from '../../Tool.js';
import { buildTool, type ToolDef } from '../../Tool.js';
import { backgroundExistingForegroundTask, markTaskNotified, registerForeground, spawnShellTask, unregisterForeground } from '../../tasks/LocalShellTask/LocalShellTask.js';
import type { AgentId } from '../../types/ids.js';
import type { AssistantMessage } from '../../types/message.js';
import { parseForSecurity } from '../../utils/bash/ast.js';
import { splitCommand_DEPRECATED, splitCommandWithOperators } from '../../utils/bash/commands.js';
import { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js';
import { detectCodeIndexingFromCommand } from '../../utils/codeIndexing.js';
import { isEnvTruthy } from '../../utils/envUtils.js';
import { isENOENT, ShellError } from '../../utils/errors.js';
import { detectFileEncoding, detectLineEndings, getFileModificationTime, writeTextContent } from '../../utils/file.js';
import { fileHistoryEnabled, fileHistoryTrackEdit } from '../../utils/fileHistory.js';
import { truncate } from '../../utils/format.js';
import { getFsImplementation } from '../../utils/fsOperations.js';
import { lazySchema } from '../../utils/lazySchema.js';
import { expandPath } from '../../utils/path.js';
import type { PermissionResult } from '../../utils/permissions/PermissionResult.js';
import { maybeRecordPluginHint } from '../../utils/plugins/hintRecommendation.js';
import { exec } from '../../utils/Shell.js';
import type { ExecResult } from '../../utils/ShellCommand.js';
import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js';
import { semanticBoolean } from '../../utils/semanticBoolean.js';
import { semanticNumber } from '../../utils/semanticNumber.js';
import { EndTruncatingAccumulator } from '../../utils/stringUtils.js';
import { getTaskOutputPath } from '../../utils/task/diskOutput.js';
import { TaskOutput } from '../../utils/task/TaskOutput.js';
import { isOutputLineTruncated } from '../../utils/terminal.js';
import { buildLargeToolResultMessage, ensureToolResultsDir, generatePreview, getToolResultPath, PREVIEW_SIZE_BYTES } from '../../utils/toolResultStorage.js';
import { userFacingName as fileEditUserFacingName } from '../FileEditTool/UI.js';
import { trackGitOperations } from '../shared/gitOperationTracking.js';
import { bashToolHasPermission, commandHasAnyCd, matchWildcardPattern, permissionRuleExtractPrefix } from './bashPermissions.js';
import { interpretCommandResult } from './commandSemantics.js';
import { getDefaultTimeoutMs, getMaxTimeoutMs, getSimplePrompt } from './prompt.js';
import { checkReadOnlyConstraints } from './readOnlyValidation.js';
import { parseSedEditCommand } from './sedEditParser.js';
import { shouldUseSandbox } from './shouldUseSandbox.js';
import { BASH_TOOL_NAME } from './toolName.js';
import { BackgroundHint, renderToolResultMessage, renderToolUseErrorMessage, renderToolUseMessage, renderToolUseProgressMessage, renderToolUseQueuedMessage } from './UI.js';
import { buildImageToolResult, isImageOutput, resetCwdIfOutsideProject, resizeShellImageOutput, stdErrAppendShellResetMessage, stripEmptyLines } from './utils.js';
const EOL = '\n';
// Progress display constants
const PROGRESS_THRESHOLD_MS = 2000; // Show progress after 2 seconds
// In assistant mode, blocking bash auto-backgrounds after this many ms in the main agent
const ASSISTANT_BLOCKING_BUDGET_MS = 15_000;
// Search commands for collapsible display (grep, find, etc.)
const BASH_SEARCH_COMMANDS = new Set(['find', 'grep', 'rg', 'ag', 'ack', 'locate', 'which', 'whereis']);
// Read/view commands for collapsible display (cat, head, etc.)
const BASH_READ_COMMANDS = new Set(['cat', 'head', 'tail', 'less', 'more',
// Analysis commands
'wc', 'stat', 'file', 'strings',
// Data processing β commonly used to parse/transform file content in pipes
'jq', 'awk', 'cut', 'sort', 'uniq', 'tr']);
// Directory-listing commands for collapsible display (ls, tree, du).
// Split from BASH_READ_COMMANDS so the summary says "Listed N directories"
// instead of the misleading "Read N files".
const BASH_LIST_COMMANDS = new Set(['ls', 'tree', 'du']);
// Commands that are semantic-neutral in any position β pure output/status commands
// that don't change the read/search nature of the overall pipeline.
// e.g. `ls dir && echo "---" && ls dir2` is still a read-only compound command.
const BASH_SEMANTIC_NEUTRAL_COMMANDS = new Set(['echo', 'printf', 'true', 'false', ':' // bash no-op
]);
// Commands that typically produce no stdout on success
const BASH_SILENT_COMMANDS = new Set(['mv', 'cp', 'rm', 'mkdir', 'rmdir', 'chmod', 'chown', 'chgrp', 'touch', 'ln', 'cd', 'export', 'unset', 'wait']);
/**
* Checks if a bash command is a search or read operation.
* Used to determine if the command should be collapsed in the UI.
* Returns an object indicating whether it's a search or read operation.
*
* For pipelines (e.g., `cat file | bq`), ALL parts must be search/read commands
* for the whole command to be considered collapsible.
*
* Semantic-neutral commands (echo, printf, true, false, :) are skipped in any
* position, as they're pure output/status commands that don't affect the read/search
* nature of the pipeline (e.g. `ls dir && echo "---" && ls dir2` is still a read).
*/
export function isSearchOrReadBashCommand(command: string): {
isSearch: boolean;
isRead: boolean;
isList: boolean;
} {
let partsWithOperators: string[];
try {
partsWithOperators = splitCommandWithOperators(command);
} catch {
// If we can't parse the command due to malformed syntax,
// it's not a search/read command
return {
isSearch: false,
isRead: false,
isList: false
};
}
if (partsWithOperators.length === 0) {
return {
isSearch: false,
isRead: false,
isList: false
};
}
let hasSearch = false;
let hasRead = false;
let hasList = false;
let hasNonNeutralCommand = false;
let skipNextAsRedirectTarget = false;
for (const part of partsWithOperators) {
if (skipNextAsRedirectTarget) {
skipNextAsRedirectTarget = false;
continue;
}
if (part === '>' || part === '>>' || part === '>&') {
skipNextAsRedirectTarget = true;
continue;
}
if (part === '||' || part === '&&' || part === '|' || part === ';') {
continue;
}
const baseCommand = part.trim().split(/\s+/)[0];
if (!baseCommand) {
continue;
}
if (BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {
continue;
}
hasNonNeutralCommand = true;
const isPartSearch = BASH_SEARCH_COMMANDS.has(baseCommand);
const isPartRead = BASH_READ_COMMANDS.has(baseCommand);
const isPartList = BASH_LIST_COMMANDS.has(baseCommand);
if (!isPartSearch && !isPartRead && !isPartList) {
return {
isSearch: false,
isRead: false,
isList: false
};
}
if (isPartSearch) hasSearch = true;
if (isPartRead) hasRead = true;
if (isPartList) hasList = true;
}
// Only neutral commands (e.g., just "echo foo") -- not collapsible
if (!hasNonNeutralCommand) {
return {
isSearch: false,
isRead: false,
isList: false
};
}
return {
isSearch: hasSearch,
isRead: hasRead,
isList: hasList
};
}
/**
* Checks if a bash command is expected to produce no stdout on success.
* Used to show "Done" instead of "(No output)" in the UI.
*/
function isSilentBashCommand(command: string): boolean {
let partsWithOperators: string[];
try {
partsWithOperators = splitCommandWithOperators(command);
} catch {
return false;
}
if (partsWithOperators.length === 0) {
return false;
}
let hasNonFallbackCommand = false;
let lastOperator: string | null = null;
let skipNextAsRedirectTarget = false;
for (const part of partsWithOperators) {
if (skipNextAsRedirectTarget) {
skipNextAsRedirectTarget = false;
continue;
}
if (part === '>' || part === '>>' || part === '>&') {
skipNextAsRedirectTarget = true;
continue;
}
if (part === '||' || part === '&&' || part === '|' || part === ';') {
lastOperator = part;
continue;
}
const baseCommand = part.trim().split(/\s+/)[0];
if (!baseCommand) {
continue;
}
if (lastOperator === '||' && BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {
continue;
}
hasNonFallbackCommand = true;
if (!BASH_SILENT_COMMANDS.has(baseCommand)) {
return false;
}
}
return hasNonFallbackCommand;
}
// Commands that should not be auto-backgrounded
const DISALLOWED_AUTO_BACKGROUND_COMMANDS = ['sleep' // Sleep should run in foreground unless explicitly backgrounded by user
];
// Check if background tasks are disabled at module load time
const isBackgroundTasksDisabled =
// eslint-disable-next-line custom-rules/no-process-env-top-level -- Intentional: schema must be defined at module load
isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_BACKGROUND_TASKS);
const fullInputSchema = lazySchema(() => z.strictObject({
command: z.string().describe('The command to execute'),
timeout: semanticNumber(z.number().optional()).describe(`Optional timeout in milliseconds (max ${getMaxTimeoutMs()})`),
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.
For simple commands (git, npm, standard CLI tools), keep it brief (5-10 words):
- ls β "List files in current directory"
- git status β "Show working tree status"
- npm install β "Install package dependencies"
For commands that are harder to parse at a glance (piped commands, obscure flags, etc.), add enough context to clarify what it does:
- find . -name "*.tmp" -exec rm {} \\; β "Find and delete all .tmp files recursively"
- git reset --hard origin/main β "Discard all local changes and match remote main"
- curl -s url | jq '.data[]' β "Fetch JSON from URL and extract data array elements"`),
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.`),
dangerouslyDisableSandbox: semanticBoolean(z.boolean().optional()).describe('Set this to true to dangerously override sandbox mode and run commands without sandboxing.'),
_simulatedSedEdit: z.object({
filePath: z.string(),
newContent: z.string()
}).optional().describe('Internal: pre-computed sed edit result from preview')
}));
// Always omit _simulatedSedEdit from the model-facing schema. It is an internal-only
// field set by SedEditPermissionRequest after the user approves a sed edit preview.
// Exposing it in the schema would let the model bypass permission checks and the
// sandbox by pairing an innocuous command with an arbitrary file write.
// Also conditionally remove run_in_background when background tasks are disabled.
const inputSchema = lazySchema(() => isBackgroundTasksDisabled ? fullInputSchema().omit({
run_in_background: true,
_simulatedSedEdit: true
}) : fullInputSchema().omit({
_simulatedSedEdit: true
}));
type InputSchema = ReturnType<typeof inputSchema>;
// Use fullInputSchema for the type to always include run_in_background
// (even when it's omitted from the schema, the code needs to handle it)
export type BashToolInput = z.infer<ReturnType<typeof fullInputSchema>>;
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;
function getCommandTypeForLogging(command: string): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {
const parts = splitCommand_DEPRECATED(command);
if (parts.length === 0) return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
// Check each part of the command to see if any match common background commands
for (const part of parts) {
const baseCommand = part.split(' ')[0] || '';
if (COMMON_BACKGROUND_COMMANDS.includes(baseCommand as (typeof COMMON_BACKGROUND_COMMANDS)[number])) {
return baseCommand as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
}
}
return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
}
const outputSchema = lazySchema(() => z.object({
stdout: z.string().describe('The standard output of the command'),
stderr: z.string().describe('The standard error output of the command'),
rawOutputPath: z.string().optional().describe('Path to raw output file for large MCP tool outputs'),
interrupted: z.boolean().describe('Whether the command was interrupted'),
isImage: z.boolean().optional().describe('Flag to indicate if stdout contains image data'),
backgroundTaskId: z.string().optional().describe('ID of the background task if command is running in background'),
backgroundedByUser: z.boolean().optional().describe('True if the user manually backgrounded the command with Ctrl+B'),
assistantAutoBackgrounded: z.boolean().optional().describe('True if assistant-mode auto-backgrounded a long-running blocking command'),
dangerouslyDisableSandbox: z.boolean().optional().describe('Flag to indicate if sandbox mode was overridden'),
returnCodeInterpretation: z.string().optional().describe('Semantic interpretation for non-error exit codes with special meaning'),
noOutputExpected: z.boolean().optional().describe('Whether the command is expected to produce no output on success'),
structuredContent: z.array(z.any()).optional().describe('Structured content blocks'),
persistedOutputPath: z.string().optional().describe('Path to the persisted full output in tool-results dir (set when output is too large for inline)'),
persistedOutputSize: z.number().optional().describe('Total size of the output in bytes (set when output is too large for inline)')
}));
type OutputSchema = ReturnType<typeof outputSchema>;
export type Out = z.infer<OutputSchema>;
// Re-export BashProgress from centralized types to break import cycles
export type { BashProgress } from '../../types/tools.js';
import type { BashProgress } from '../../types/tools.js';
/**
* Checks if a command is allowed to be automatically backgrounded
* @param command The command to check
* @returns false for commands that should not be auto-backgrounded (like sleep)
*/
function isAutobackgroundingAllowed(command: string): boolean {
const parts = splitCommand_DEPRECATED(command);
if (parts.length === 0) return true;
// Get the first part which should be the base command
const baseCommand = parts[0]?.trim();
if (!baseCommand) return true;
return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(baseCommand);
}
/**
* Detect standalone or leading `sleep N` patterns that should use Monitor
* instead. Catches `sleep 5`, `sleep 5 && check`, `sleep 5; check` β but
* not sleep inside pipelines, subshells, or scripts (those are fine).
*/
export function detectBlockedSleepPattern(command: string): string | null {
const parts = splitCommand_DEPRECATED(command);
if (parts.length === 0) return null;
const first = parts[0]?.trim() ?? '';
// Bare `sleep N` or `sleep N.N` as the first subcommand.
// Float durations (sleep 0.5) are allowed β those are legit pacing, not polls.
const m = /^sleep\s+(\d+)\s*$/.exec(first);
if (!m) return null;
const secs = parseInt(m[1]!, 10);
if (secs < 2) return null; // sub-2s sleeps are fine (rate limiting, pacing)
// `sleep N` alone β "what are you waiting for?"
// `sleep N && check` β "use Monitor { command: check }"
const rest = parts.slice(1).join(' ').trim();
return rest ? `sleep ${secs} followed by: ${rest}` : `standalone sleep ${secs}`;
}
/**
* Checks if a command contains tools that shouldn't run in sandbox
* This includes:
* - Dynamic config-based disabled commands and substrings (tengu_sandbox_disabled_commands)
* - User-configured commands from settings.json (sandbox.excludedCommands)
*
* User-configured commands support the same pattern syntax as permission rules:
* - Exact matches: "npm run lint"
* - Prefix patterns: "npm run test:*"
*/
type SimulatedSedEditResult = {
data: Out;
};
type SimulatedSedEditContext = Pick<ToolUseContext, 'readFileState' | 'updateFileHistoryState'>;
/**
* Applies a simulated sed edit directly instead of running sed.
* This is used by the permission dialog to ensure what the user previews
* is exactly what gets written to the file.
*/
async function applySedEdit(simulatedEdit: {
filePath: string;
newContent: string;
}, toolUseContext: SimulatedSedEditContext, parentMessage?: AssistantMessage): Promise<SimulatedSedEditResult> {
const {
filePath,
newContent
} = simulatedEdit;
const absoluteFilePath = expandPath(filePath);
const fs = getFsImplementation();
// Read original content for VS Code notification
const encoding = detectFileEncoding(absoluteFilePath);
let originalContent: string;
try {
originalContent = await fs.readFile(absoluteFilePath, {
encoding
});
} catch (e) {
if (isENOENT(e)) {
return {
data: {
stdout: '',
stderr: `sed: ${filePath}: No such file or directory\nExit code 1`,
interrupted: false
}
};
}
throw e;
}
// Track file history before making changes (for undo support)
if (fileHistoryEnabled() && parentMessage) {
await fileHistoryTrackEdit(toolUseContext.updateFileHistoryState, absoluteFilePath, parentMessage.uuid);
}
// Detect line endings and write new content
const endings = detectLineEndings(absoluteFilePath);
writeTextContent(absoluteFilePath, newContent, encoding, endings);
// Notify VS Code about the file change
notifyVscodeFileUpdated(absoluteFilePath, originalContent, newContent);
// Update read timestamp to invalidate stale writes
toolUseContext.readFileState.set(absoluteFilePath, {
content: newContent,
timestamp: getFileModificationTime(absoluteFilePath),
offset: undefined,
limit: undefined
});
// Return success result matching sed output format (sed produces no output on success)
return {
data: {
stdout: '',
stderr: '',
interrupted: false
}
};
}
export const BashTool = buildTool({
name: BASH_TOOL_NAME,
searchHint: 'execute shell commands',
// 30K chars - tool result persistence threshold
maxResultSizeChars: 30_000,
strict: true,
async description({
description
}) {
return description || 'Run shell command';
},
async prompt() {
return getSimplePrompt();
},
isConcurrencySafe(input) {
return this.isReadOnly?.(input) ?? false;
},
isReadOnly(input) {
const compoundCommandHasCd = commandHasAnyCd(input.command);
const result = checkReadOnlyConstraints(input, compoundCommandHasCd);
return result.behavior === 'allow';
},
toAutoClassifierInput(input) {
return input.command;
},
async preparePermissionMatcher({
command
}) {
// Hook `if` filtering is "no match β skip hook" (deny-like semantics), so
// compound commands must fire the hook if ANY subcommand matches. Without
// splitting, `ls && git push` would bypass a `Bash(git *)` security hook.
const parsed = await parseForSecurity(command);
if (parsed.kind !== 'simple') {
// parse-unavailable / too-complex: fail safe by running the hook.
return () => true;
}
// Match on argv (strips leading VAR=val) so `FOO=bar git push` still
// matches `Bash(git *)`.
const subcommands = parsed.commands.map(c => c.argv.join(' '));
return pattern => {
const prefix = permissionRuleExtractPrefix(pattern);
return subcommands.some(cmd => {
if (prefix !== null) {
return cmd === prefix || cmd.startsWith(`${prefix} `);
}
return matchWildcardPattern(pattern, cmd);
});
};
},
isSearchOrReadCommand(input) {
const parsed = inputSchema().safeParse(input);
if (!parsed.success) return {
isSearch: false,
isRead: false,
isList: false
};
return isSearchOrReadBashCommand(parsed.data.command);
},
get inputSchema(): InputSchema {
return inputSchema();
},
get outputSchema(): OutputSchema {
return outputSchema();
},
userFacingName(input) {
if (!input) {
return 'Bash';
}
// Render sed in-place edits as file edits
if (input.command) {
const sedInfo = parseSedEditCommand(input.command);
if (sedInfo) {
return fileEditUserFacingName({
file_path: sedInfo.filePath,
old_string: 'x'
});
}
}
// Env var FIRST: shouldUseSandbox β splitCommand_DEPRECATED β shell-quote's
// `new RegExp` per call. userFacingName runs per-render for every bash
// message in history; with ~50 msgs + one slow-to-tokenize command, this
// exceeds the shimmer tick β transition abort β infinite retry (#21605).
return isEnvTruthy(process.env.CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR) && shouldUseSandbox(input) ? 'SandboxedBash' : 'Bash';
},
getToolUseSummary(input) {
if (!input?.command) {
return null;
}
const {
command,
description
} = input;
if (description) {
return description;
}
return truncate(command, TOOL_SUMMARY_MAX_LENGTH);
},
getActivityDescription(input) {
if (!input?.command) {
return 'Running command';
}
const desc = input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH);
return `Running ${desc}`;
},
async validateInput(input: BashToolInput): Promise<ValidationResult> {
if (feature('MONITOR_TOOL') && !isBackgroundTasksDisabled && !input.run_in_background) {
const sleepPattern = detectBlockedSleepPattern(input.command);
if (sleepPattern !== null) {
return {
result: false,
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.`,
errorCode: 10
};
}
}
return {
result: true
};
},
async checkPermissions(input, context): Promise<PermissionResult> {
return bashToolHasPermission(input, context);
},
renderToolUseMessage,
renderToolUseProgressMessage,
renderToolUseQueuedMessage,
renderToolResultMessage,
// BashToolResultMessage shows <OutputLine content={stdout}> + stderr.
// UI never shows persistedOutputPath wrapper, backgroundInfo β those are
// model-facing (mapToolResult... below).
extractSearchText({
stdout,
stderr
}) {
return stderr ? `${stdout}\n${stderr}` : stdout;
},
mapToolResultToToolResultBlockParam({
interrupted,
stdout,
stderr,
isImage,
backgroundTaskId,
backgroundedByUser,
assistantAutoBackgrounded,
structuredContent,
persistedOutputPath,
persistedOutputSize
}, toolUseID): ToolResultBlockParam {
// Handle structured content
if (structuredContent && structuredContent.length > 0) {
return {
tool_use_id: toolUseID,
type: 'tool_result',
content: structuredContent
};
}
// For image data, format as image content block for Claude
if (isImage) {
const block = buildImageToolResult(stdout, toolUseID);
if (block) return block;
}
let processedStdout = stdout;
if (stdout) {
// Replace any leading newlines or lines with only whitespace
processedStdout = stdout.replace(/^(\s*\n)+/, '');
// Still trim the end as before
processedStdout = processedStdout.trimEnd();
}
// For large output that was persisted to disk, build <persisted-output>
// message for the model. The UI never sees this β it uses data.stdout.
if (persistedOutputPath) {
const preview = generatePreview(processedStdout, PREVIEW_SIZE_BYTES);
processedStdout = buildLargeToolResultMessage({
filepath: persistedOutputPath,
originalSize: persistedOutputSize ?? 0,
isJson: false,
preview: preview.preview,
hasMore: preview.hasMore
});
}
let errorMessage = stderr.trim();
if (interrupted) {
if (stderr) errorMessage += EOL;
errorMessage += '<error>Command was aborted before completion</error>';
}
let backgroundInfo = '';
if (backgroundTaskId) {
const outputPath = getTaskOutputPath(backgroundTaskId);
if (assistantAutoBackgrounded) {
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.`;
} else if (backgroundedByUser) {
backgroundInfo = `Command was manually backgrounded by user with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`;
} else {
backgroundInfo = `Command running in background with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`;
}
}
return {
tool_use_id: toolUseID,
type: 'tool_result',
content: [processedStdout, errorMessage, backgroundInfo].filter(Boolean).join('\n'),
is_error: interrupted
};
},
async call(input: BashToolInput, toolUseContext, _canUseTool?: CanUseToolFn, parentMessage?: AssistantMessage, onProgress?: ToolCallProgress<BashProgress>) {
// Handle simulated sed edit - apply directly instead of running sed
// This ensures what the user previewed is exactly what gets written
if (input._simulatedSedEdit) {
return applySedEdit(input._simulatedSedEdit, toolUseContext, parentMessage);
}
const {
abortController,
getAppState,
setAppState,
setToolJSX
} = toolUseContext;
const stdoutAccumulator = new EndTruncatingAccumulator();
let stderrForShellReset = '';
let interpretationResult: ReturnType<typeof interpretCommandResult> | undefined;
let progressCounter = 0;
let wasInterrupted = false;
let result: ExecResult;
const isMainThread = !toolUseContext.agentId;
const preventCwdChanges = !isMainThread;
try {
// Use the new async generator version of runShellCommand
const commandGenerator = runShellCommand({
input,
abortController,
// Use the always-shared task channel so async agents' background
// bash tasks are actually registered (and killable on agent exit).
setAppState: toolUseContext.setAppStateForTasks ?? setAppState,
setToolJSX,
preventCwdChanges,
isMainThread,
toolUseId: toolUseContext.toolUseId,
agentId: toolUseContext.agentId
});
// Consume the generator and capture the return value
let generatorResult;
do {
generatorResult = await commandGenerator.next();
if (!generatorResult.done && onProgress) {
const progress = generatorResult.value;
onProgress({
toolUseID: `bash-progress-${progressCounter++}`,
data: {
type: 'bash_progress',
output: progress.output,
fullOutput: progress.fullOutput,
elapsedTimeSeconds: progress.elapsedTimeSeconds,
totalLines: progress.totalLines,
totalBytes: progress.totalBytes,
taskId: progress.taskId,
timeoutMs: progress.timeoutMs
}
});
}
} while (!generatorResult.done);
// Get the final result from the generator's return value
result = generatorResult.value;
trackGitOperations(input.command, result.code, result.stdout);
const isInterrupt = result.interrupted && abortController.signal.reason === 'interrupt';
// stderr is interleaved in stdout (merged fd) β result.stdout has both
stdoutAccumulator.append((result.stdout || '').trimEnd() + EOL);
// Interpret the command result using semantic rules
interpretationResult = interpretCommandResult(input.command, result.code, result.stdout || '', '');
// Check for git index.lock error (stderr is in stdout now)
if (result.stdout && result.stdout.includes(".git/index.lock': File exists")) {
logEvent('tengu_git_index_lock_error', {});
}
if (interpretationResult.isError && !isInterrupt) {
// Only add exit code if it's actually an error
if (result.code !== 0) {
stdoutAccumulator.append(`Exit code ${result.code}`);
}
}
if (!preventCwdChanges) {
const appState = getAppState();
if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {
stderrForShellReset = stdErrAppendShellResetMessage('');
}
}
// Annotate output with sandbox violations if any (stderr is in stdout)
const outputWithSbFailures = SandboxManager.annotateStderrWithSandboxFailures(input.command, result.stdout || '');
if (result.preSpawnError) {
throw new Error(result.preSpawnError);
}
if (interpretationResult.isError && !isInterrupt) {
// stderr is merged into stdout (merged fd); outputWithSbFailures
// already has the full output. Pass '' for stdout to avoid
// duplication in getErrorParts() and processBashCommand.
throw new ShellError('', outputWithSbFailures, result.code, result.interrupted);
}
wasInterrupted = result.interrupted;
} finally {
if (setToolJSX) setToolJSX(null);
}
// Get final string from accumulator
const stdout = stdoutAccumulator.toString();
// Large output: the file on disk has more than getMaxOutputLength() bytes.
// stdout already contains the first chunk (from getStdout()). Copy the
// output file to the tool-results dir so the model can read it via
// FileRead. If > 64 MB, truncate after copying.
const MAX_PERSISTED_SIZE = 64 * 1024 * 1024;
let persistedOutputPath: string | undefined;
let persistedOutputSize: number | undefined;
if (result.outputFilePath && result.outputTaskId) {
try {
const fileStat = await fsStat(result.outputFilePath);
persistedOutputSize = fileStat.size;
await ensureToolResultsDir();
const dest = getToolResultPath(result.outputTaskId, false);
if (fileStat.size > MAX_PERSISTED_SIZE) {
await fsTruncate(result.outputFilePath, MAX_PERSISTED_SIZE);
}
try {
await link(result.outputFilePath, dest);
} catch {
await copyFile(result.outputFilePath, dest);
}
persistedOutputPath = dest;
} catch {
// File may already be gone β stdout preview is sufficient
}
}
const commandType = input.command.split(' ')[0];
logEvent('tengu_bash_tool_command_executed', {
command_type: commandType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
stdout_length: stdout.length,
stderr_length: 0,
exit_code: result.code,
interrupted: wasInterrupted
});
// Log code indexing tool usage
const codeIndexingTool = detectCodeIndexingFromCommand(input.command);
if (codeIndexingTool) {
logEvent('tengu_code_indexing_tool_used', {
tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
source: 'cli' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
success: result.code === 0
});
}
let strippedStdout = stripEmptyLines(stdout);
// Claude Code hints protocol: CLIs/SDKs gated on CLAUDECODE=1 emit a
// `<claude-code-hint />` tag to stderr (merged into stdout here). Scan,
// record for useClaudeCodeHintRecommendation to surface, then strip
// so the model never sees the tag β a zero-token side channel.
// Stripping runs unconditionally (subagent output must stay clean too);
// only the dialog recording is main-thread-only.
const extracted = extractClaudeCodeHints(strippedStdout, input.command);
strippedStdout = extracted.stripped;
if (isMainThread && extracted.hints.length > 0) {
for (const hint of extracted.hints) maybeRecordPluginHint(hint);
}
let isImage = isImageOutput(strippedStdout);
// Cap image dimensions + size if present (CC-304 β see
// resizeShellImageOutput). Scope the decoded buffer so it can be reclaimed
// before we build the output Out object.
let compressedStdout = strippedStdout;
if (isImage) {
const resized = await resizeShellImageOutput(strippedStdout, result.outputFilePath, persistedOutputSize);
if (resized) {
compressedStdout = resized;
} else {
// Parse failed or file too large (e.g. exceeds MAX_IMAGE_FILE_SIZE).
// Keep isImage in sync with what we actually send so the UI label stays
// accurate β mapToolResultToToolResultBlockParam's defensive
// fallthrough will send text, not an image block.
isImage = false;
}
}
const data: Out = {
stdout: compressedStdout,
stderr: stderrForShellReset,
interrupted: wasInterrupted,
isImage,
returnCodeInterpretation: interpretationResult?.message,
noOutputExpected: isSilentBashCommand(input.command),
backgroundTaskId: result.backgroundTaskId,
backgroundedByUser: result.backgroundedByUser,
assistantAutoBackgrounded: result.assistantAutoBackgrounded,
dangerouslyDisableSandbox: 'dangerouslyDisableSandbox' in input ? input.dangerouslyDisableSandbox as boolean | undefined : undefined,
persistedOutputPath,
persistedOutputSize
};
return {
data
};
},
renderToolUseErrorMessage,
isResultTruncated(output: Out): boolean {
return isOutputLineTruncated(output.stdout) || isOutputLineTruncated(output.stderr);
}
} satisfies ToolDef<InputSchema, Out, BashProgress>);
async function* runShellCommand({
input,
abortController,
setAppState,
setToolJSX,
preventCwdChanges,
isMainThread,
toolUseId,
agentId
}: {
input: BashToolInput;
abortController: AbortController;
setAppState: (f: (prev: AppState) => AppState) => void;
setToolJSX?: SetToolJSXFn;
preventCwdChanges?: boolean;
isMainThread?: boolean;
toolUseId?: string;
agentId?: AgentId;
}): AsyncGenerator<{
type: 'progress';
output: string;
fullOutput: string;
elapsedTimeSeconds: number;
totalLines: number;
totalBytes?: number;
taskId?: string;
timeoutMs?: number;
}, ExecResult, void> {
const {
command,
description,
timeout,
run_in_background
} = input;
const timeoutMs = timeout || getDefaultTimeoutMs();
let fullOutput = '';
let lastProgressOutput = '';
let lastTotalLines = 0;
let lastTotalBytes = 0;
let backgroundShellId: string | undefined = undefined;
let assistantAutoBackgrounded = false;
// Progress signal: resolved by onProgress callback from the shared poller,
// waking the generator to yield a progress update.
let resolveProgress: (() => void) | null = null;
function createProgressSignal(): Promise<null> {
return new Promise<null>(resolve => {
resolveProgress = () => resolve(null);
});
}
// Determine if auto-backgrounding should be enabled
// Only enable for commands that are allowed to be auto-backgrounded
// and when background tasks are not disabled
const shouldAutoBackground = !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command);
const shellCommand = await exec(command, abortController.signal, 'bash', {
timeout: timeoutMs,
onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {
lastProgressOutput = lastLines;
fullOutput = allLines;
lastTotalLines = totalLines;
lastTotalBytes = isIncomplete ? totalBytes : 0;
// Wake the generator so it yields the new progress data
const resolve = resolveProgress;
if (resolve) {
resolveProgress = null;
resolve();
}
},
preventCwdChanges,
shouldUseSandbox: shouldUseSandbox(input),
shouldAutoBackground
});
// Start the command execution
const resultPromise = shellCommand.result;
// Helper to spawn a background task and return its ID
async function spawnBackgroundTask(): Promise<string> {
const handle = await spawnShellTask({
command,
description: description || command,
shellCommand,
toolUseId,
agentId
}, {
abortController,
getAppState: () => {
// We don't have direct access to getAppState here, but spawn doesn't
// actually use it during the spawn process
throw new Error('getAppState not available in runShellCommand context');
},
setAppState
});
return handle.taskId;
}
// Helper to start backgrounding with optional logging
function startBackgrounding(eventName: string, backgroundFn?: (shellId: string) => void): void {
// If a foreground task is already registered (via registerForeground in the
// progress loop), background it in-place instead of re-spawning. Re-spawning
// would overwrite tasks[taskId], emit a duplicate task_started SDK event,
// and leak the first cleanup callback.
if (foregroundTaskId) {
if (!backgroundExistingForegroundTask(foregroundTaskId, shellCommand, description || command, setAppState, toolUseId)) {
return;
}
backgroundShellId = foregroundTaskId;
logEvent(eventName, {
command_type: getCommandTypeForLogging(command)
});
backgroundFn?.(foregroundTaskId);
return;
}
// No foreground task registered β spawn a new background task
// Note: spawn is essentially synchronous despite being async
void spawnBackgroundTask().then(shellId => {
backgroundShellId = shellId;
// Wake the generator's Promise.race so it sees backgroundShellId.
// Without this, if the poller has stopped ticking for this task
// (no output + shared-poller race with sibling stopPolling calls)
// and the process is hung on I/O, the race at line ~1357 never
// resolves and the generator deadlocks despite being backgrounded.
const resolve = resolveProgress;
if (resolve) {
resolveProgress = null;
resolve();
}
logEvent(eventName, {
command_type: getCommandTypeForLogging(command)
});
if (backgroundFn) {
backgroundFn(shellId);
}
});
}
// Set up auto-backgrounding on timeout if enabled
// Only background commands that are allowed to be auto-backgrounded (not sleep, etc.)
if (shellCommand.onTimeout && shouldAutoBackground) {
shellCommand.onTimeout(backgroundFn => {
startBackgrounding('tengu_bash_command_timeout_backgrounded', backgroundFn);
});
}
// In assistant mode, the main agent should stay responsive. Auto-background
// blocking commands after ASSISTANT_BLOCKING_BUDGET_MS so the agent can keep
// coordinating instead of waiting. The command keeps running β no state loss.
if (feature('KAIROS') && getKairosActive() && isMainThread && !isBackgroundTasksDisabled && run_in_background !== true) {
setTimeout(() => {
if (shellCommand.status === 'running' && backgroundShellId === undefined) {
assistantAutoBackgrounded = true;
startBackgrounding('tengu_bash_command_assistant_auto_backgrounded');
}
}, ASSISTANT_BLOCKING_BUDGET_MS).unref();
}
// Handle Claude asking to run it in the background explicitly
// When explicitly requested via run_in_background, always honor the request
// regardless of the command type (isAutobackgroundingAllowed only applies to automatic backgrounding)
// Skip if background tasks are disabled - run in foreground instead
if (run_in_background === true && !isBackgroundTasksDisabled) {
const shellId = await spawnBackgroundTask();
logEvent('tengu_bash_command_explicitly_backgrounded', {
command_type: getCommandTypeForLogging(command)
});
return {
stdout: '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: shellId
};
}
// Wait for the initial threshold before showing progress
const startTime = Date.now();
let foregroundTaskId: string | undefined = undefined;
{
const initialResult = await Promise.race([resultPromise, new Promise<null>(resolve => {
const t = setTimeout((r: (v: null) => void) => r(null), PROGRESS_THRESHOLD_MS, resolve);
t.unref();
})]);
if (initialResult !== null) {
shellCommand.cleanup();
return initialResult;
}
if (backgroundShellId) {
return {
stdout: '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: backgroundShellId,
assistantAutoBackgrounded
};
}
}
// Start polling the output file for progress. The poller's #tick calls
// onProgress every second, which resolves progressSignal below.
TaskOutput.startPolling(shellCommand.taskOutput.taskId);
// Progress loop: wake is driven by the shared poller calling onProgress,
// which resolves the progressSignal.
try {
while (true) {
const progressSignal = createProgressSignal();
const result = await Promise.race([resultPromise, progressSignal]);
if (result !== null) {
// Race: backgrounding fired (15s timer / onTimeout / Ctrl+B) but the
// command completed before the next poll tick. #handleExit sets
// backgroundTaskId but skips outputFilePath (it assumes the background
// message or <task_notification> will carry the path). Strip
// backgroundTaskId so the model sees a clean completed command,
// reconstruct outputFilePath for large outputs, and suppress the
// redundant <task_notification> from the .then() handler.
// Check result.backgroundTaskId (not the closure var) to also cover
// Ctrl+B, which calls shellCommand.background() directly.
if (result.backgroundTaskId !== undefined) {
markTaskNotified(result.backgroundTaskId, setAppState);
const fixedResult: ExecResult = {
...result,
backgroundTaskId: undefined
};
// Mirror ShellCommand.#handleExit's large-output branch that was
// skipped because #backgroundTaskId was set.
const {
taskOutput
} = shellCommand;
if (taskOutput.stdoutToFile && !taskOutput.outputFileRedundant) {
fixedResult.outputFilePath = taskOutput.path;
fixedResult.outputFileSize = taskOutput.outputFileSize;
fixedResult.outputTaskId = taskOutput.taskId;
}
shellCommand.cleanup();
return fixedResult;
}
// Command has completed - return the actual result
// If we registered as a foreground task, unregister it
if (foregroundTaskId) {
unregisterForeground(foregroundTaskId, setAppState);
}
// Clean up stream resources for foreground commands
// (backgrounded commands are cleaned up by LocalShellTask)
shellCommand.cleanup();
return result;
}
// Check if command was backgrounded (either via old mechanism or new backgroundAll)
if (backgroundShellId) {
return {
stdout: '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: backgroundShellId,
assistantAutoBackgrounded
};
}
// Check if this foreground task was backgrounded via backgroundAll()
if (foregroundTaskId) {
// shellCommand.status becomes 'backgrounded' when background() is called
if (shellCommand.status === 'backgrounded') {
return {
stdout: '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: foregroundTaskId,
backgroundedByUser: true
};
}
}
// Time for a progress update
const elapsed = Date.now() - startTime;
const elapsedSeconds = Math.floor(elapsed / 1000);
// Show minimal backgrounding UI if available
// Skip if background tasks are disabled
if (!isBackgroundTasksDisabled && backgroundShellId === undefined && elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 && setToolJSX) {
// Register this command as a foreground task so it can be backgrounded via Ctrl+B
if (!foregroundTaskId) {
foregroundTaskId = registerForeground({
command,
description: description || command,
shellCommand,
agentId
}, setAppState, toolUseId);
}
setToolJSX({
jsx: <BackgroundHint />,
shouldHidePromptInput: false,
shouldContinueAnimation: true,
showSpinner: true
});
}
yield {
type: 'progress',
fullOutput,
output: lastProgressOutput,
elapsedTimeSeconds: elapsedSeconds,
totalLines: lastTotalLines,
totalBytes: lastTotalBytes,
taskId: shellCommand.taskOutput.taskId,
...(timeout ? {
timeoutMs
} : undefined)
};
}
} finally {
TaskOutput.stopPolling(shellCommand.taskOutput.taskId);
}
}
//# 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":[]}