π File detail
tools/PowerShellTool/PowerShellTool.tsx
π― Use case
This module implements the βPowerShellToolβ tool (Power Shell) β something the model can call at runtime alongside other agent tools. On the API surface it exposes detectBlockedSleepPattern, PowerShellToolInput, Out, and PowerShellTool β 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)
detectBlockedSleepPatternPowerShellToolInputOutPowerShellTool
π 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 type { SetToolJSXFn, Tool, ToolCallProgress, 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 { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js';
import { isEnvTruthy } from '../../utils/envUtils.js';
import { errorMessage as getErrorMessage, ShellError } from '../../utils/errors.js';
import { truncate } from '../../utils/format.js';
import { lazySchema } from '../../utils/lazySchema.js';
import { logError } from '../../utils/log.js';
import type { PermissionResult } from '../../utils/permissions/PermissionResult.js';
import { getPlatform } from '../../utils/platform.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 { getCachedPowerShellPath } from '../../utils/shell/powershellDetection.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 { shouldUseSandbox } from '../BashTool/shouldUseSandbox.js';
import { BackgroundHint } from '../BashTool/UI.js';
import { buildImageToolResult, isImageOutput, resetCwdIfOutsideProject, resizeShellImageOutput, stdErrAppendShellResetMessage, stripEmptyLines } from '../BashTool/utils.js';
import { trackGitOperations } from '../shared/gitOperationTracking.js';
import { interpretCommandResult } from './commandSemantics.js';
import { powershellToolHasPermission } from './powershellPermissions.js';
import { getDefaultTimeoutMs, getMaxTimeoutMs, getPrompt } from './prompt.js';
import { hasSyncSecurityConcerns, isReadOnlyCommand, resolveToCanonical } from './readOnlyValidation.js';
import { POWERSHELL_TOOL_NAME } from './toolName.js';
import { renderToolResultMessage, renderToolUseErrorMessage, renderToolUseMessage, renderToolUseProgressMessage, renderToolUseQueuedMessage } from './UI.js';
// Never use os.EOL for terminal output β \r\n on Windows breaks Ink rendering
const EOL = '\n';
/**
* PowerShell search commands (grep equivalents) for collapsible display.
* Stored as canonical (lowercase) cmdlet names.
*/
const PS_SEARCH_COMMANDS = new Set(['select-string',
// grep equivalent
'get-childitem',
// find equivalent (with -Recurse)
'findstr',
// native Windows search
'where.exe' // native Windows which
]);
/**
* PowerShell read/view commands for collapsible display.
* Stored as canonical (lowercase) cmdlet names.
*/
const PS_READ_COMMANDS = new Set(['get-content',
// cat equivalent
'get-item',
// file info
'test-path',
// test -e equivalent
'resolve-path',
// realpath equivalent
'get-process',
// ps equivalent
'get-service',
// system info
'get-childitem',
// ls/dir equivalent (also search when recursive)
'get-location',
// pwd equivalent
'get-filehash',
// checksum
'get-acl',
// permissions info
'format-hex' // hexdump equivalent
]);
/**
* PowerShell semantic-neutral commands that don't change the search/read nature.
*/
const PS_SEMANTIC_NEUTRAL_COMMANDS = new Set(['write-output',
// echo equivalent
'write-host']);
/**
* Checks if a PowerShell command is a search or read operation.
* Used to determine if the command should be collapsed in the UI.
*/
function isSearchOrReadPowerShellCommand(command: string): {
isSearch: boolean;
isRead: boolean;
} {
const trimmed = command.trim();
if (!trimmed) {
return {
isSearch: false,
isRead: false
};
}
// Simple split on statement separators and pipe operators
// This is a sync function so we use a lightweight approach
const parts = trimmed.split(/\s*[;|]\s*/).filter(Boolean);
if (parts.length === 0) {
return {
isSearch: false,
isRead: false
};
}
let hasSearch = false;
let hasRead = false;
let hasNonNeutralCommand = false;
for (const part of parts) {
const baseCommand = part.trim().split(/\s+/)[0];
if (!baseCommand) {
continue;
}
const canonical = resolveToCanonical(baseCommand);
if (PS_SEMANTIC_NEUTRAL_COMMANDS.has(canonical)) {
continue;
}
hasNonNeutralCommand = true;
const isPartSearch = PS_SEARCH_COMMANDS.has(canonical);
const isPartRead = PS_READ_COMMANDS.has(canonical);
if (!isPartSearch && !isPartRead) {
return {
isSearch: false,
isRead: false
};
}
if (isPartSearch) hasSearch = true;
if (isPartRead) hasRead = true;
}
if (!hasNonNeutralCommand) {
return {
isSearch: false,
isRead: false
};
}
return {
isSearch: hasSearch,
isRead: hasRead
};
}
// Progress display constants
const PROGRESS_THRESHOLD_MS = 2000;
const PROGRESS_INTERVAL_MS = 1000;
// In assistant mode, blocking commands auto-background after this many ms in the main agent
const ASSISTANT_BLOCKING_BUDGET_MS = 15_000;
// Commands that should not be auto-backgrounded (canonical lowercase).
// 'sleep' is a PS built-in alias for Start-Sleep but not in COMMON_ALIASES,
// so list both forms.
const DISALLOWED_AUTO_BACKGROUND_COMMANDS = ['start-sleep',
// Start-Sleep should run in foreground unless explicitly backgrounded
'sleep'];
/**
* 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 Start-Sleep)
*/
function isAutobackgroundingAllowed(command: string): boolean {
const firstWord = command.trim().split(/\s+/)[0];
if (!firstWord) return true;
const canonical = resolveToCanonical(firstWord);
return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(canonical);
}
/**
* PS-flavored port of BashTool's detectBlockedSleepPattern.
* Catches `Start-Sleep N`, `Start-Sleep -Seconds N`, `sleep N` (built-in alias)
* as the first statement. Does NOT block `Start-Sleep -Milliseconds` (sub-second
* pacing is fine) or float seconds (legit rate limiting).
*/
export function detectBlockedSleepPattern(command: string): string | null {
// First statement only β split on PS statement separators: `;`, `|`,
// `&`/`&&`/`||` (pwsh 7+), and newline (PS's primary separator). This is
// intentionally shallow β sleep inside script blocks, subshells, or later
// pipeline stages is fine. Matches BashTool's splitCommandWithOperators
// intent (src/utils/bash/commands.ts) without a full PS parser.
const first = command.trim().split(/[;|&\r\n]/)[0]?.trim() ?? '';
// Match: Start-Sleep N, Start-Sleep -Seconds N, Start-Sleep -s N, sleep N
// (case-insensitive; -Seconds can be abbreviated to -s per PS convention)
const m = /^(?:start-sleep|sleep)(?:\s+-s(?:econds)?)?\s+(\d+)\s*$/i.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)
const rest = command.trim().slice(first.length).replace(/^[\s;|&]+/, '');
return rest ? `Start-Sleep ${secs} followed by: ${rest}` : `standalone Start-Sleep ${secs}`;
}
/**
* On Windows native, sandbox is unavailable (bwrap/sandbox-exec are
* POSIX-only). If enterprise policy has sandbox.enabled AND forbids
* unsandboxed commands, PowerShell cannot comply β refuse execution
* rather than silently bypass the policy. On Linux/macOS/WSL2, pwsh
* runs as a native binary under the sandbox same as bash, so this
* gate does not apply.
*
* Checked in BOTH validateInput (clean tool-runner error) and call()
* (covers direct callers like promptShellExecution.ts that skip
* validateInput). The call() guard is the load-bearing one.
*/
const WINDOWS_SANDBOX_POLICY_REFUSAL = 'Enterprise policy requires sandboxing, but sandboxing is not available on native Windows. Shell command execution is blocked on this platform by policy.';
function isWindowsSandboxPolicyViolation(): boolean {
return getPlatform() === 'windows' && SandboxManager.isSandboxEnabledInSettings() && !SandboxManager.areUnsandboxedCommandsAllowed();
}
// 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 PowerShell 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.'),
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.')
}));
// Conditionally remove run_in_background from schema when background tasks are disabled
const inputSchema = lazySchema(() => isBackgroundTasksDisabled ? fullInputSchema().omit({
run_in_background: true
}) : fullInputSchema());
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 PowerShellToolInput = z.infer<ReturnType<typeof fullInputSchema>>;
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'),
interrupted: z.boolean().describe('Whether the command was interrupted'),
returnCodeInterpretation: z.string().optional().describe('Semantic interpretation for non-error exit codes with special meaning'),
isImage: z.boolean().optional().describe('Flag to indicate if stdout contains image data'),
persistedOutputPath: z.string().optional().describe('Path to persisted full output when too large for inline'),
persistedOutputSize: z.number().optional().describe('Total output size in bytes when persisted'),
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 the command was auto-backgrounded by the assistant-mode blocking budget')
}));
type OutputSchema = ReturnType<typeof outputSchema>;
export type Out = z.infer<OutputSchema>;
import type { PowerShellProgress } from '../../types/tools.js';
export type { PowerShellProgress } from '../../types/tools.js';
const COMMON_BACKGROUND_COMMANDS = ['npm', 'yarn', 'pnpm', 'node', 'python', 'python3', 'go', 'cargo', 'make', 'docker', 'terraform', 'webpack', 'vite', 'jest', 'pytest', 'curl', 'Invoke-WebRequest', 'build', 'test', 'serve', 'watch', 'dev'] as const;
function getCommandTypeForLogging(command: string): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {
const trimmed = command.trim();
const firstWord = trimmed.split(/\s+/)[0] || '';
for (const cmd of COMMON_BACKGROUND_COMMANDS) {
if (firstWord.toLowerCase() === cmd.toLowerCase()) {
return cmd as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
}
}
return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS;
}
export const PowerShellTool = buildTool({
name: POWERSHELL_TOOL_NAME,
searchHint: 'execute Windows PowerShell commands',
maxResultSizeChars: 30_000,
strict: true,
async description({
description
}: Partial<PowerShellToolInput>): Promise<string> {
return description || 'Run PowerShell command';
},
async prompt(): Promise<string> {
return getPrompt();
},
isConcurrencySafe(input: PowerShellToolInput): boolean {
return this.isReadOnly?.(input) ?? false;
},
isSearchOrReadCommand(input: Partial<PowerShellToolInput>): {
isSearch: boolean;
isRead: boolean;
} {
if (!input.command) {
return {
isSearch: false,
isRead: false
};
}
return isSearchOrReadPowerShellCommand(input.command);
},
isReadOnly(input: PowerShellToolInput): boolean {
// Check sync security heuristics before declaring read-only.
// The full AST parse is async and unavailable here, so we use
// regex-based detection of subexpressions, splatting, member
// invocations, and assignments β matching BashTool's pattern of
// checking security concerns before cmdlet allowlist evaluation.
if (hasSyncSecurityConcerns(input.command)) {
return false;
}
// NOTE: This calls isReadOnlyCommand without the parsed AST. Without the
// AST, isReadOnlyCommand cannot split pipelines/statements and will return
// false for anything but the simplest single-token commands. This is a
// known limitation of the sync Tool.isReadOnly() interface β the real
// read-only auto-allow happens async in powershellToolHasPermission (step
// 4.5) where the parsed AST is available.
return isReadOnlyCommand(input.command);
},
toAutoClassifierInput(input) {
return input.command;
},
get inputSchema(): InputSchema {
return inputSchema();
},
get outputSchema(): OutputSchema {
return outputSchema();
},
userFacingName(): string {
return 'PowerShell';
},
getToolUseSummary(input: Partial<PowerShellToolInput> | undefined): string | null {
if (!input?.command) {
return null;
}
const {
command,
description
} = input;
if (description) {
return description;
}
return truncate(command, TOOL_SUMMARY_MAX_LENGTH);
},
getActivityDescription(input: Partial<PowerShellToolInput> | undefined): string {
if (!input?.command) {
return 'Running command';
}
const desc = input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH);
return `Running ${desc}`;
},
isEnabled(): boolean {
return true;
},
async validateInput(input: PowerShellToolInput): Promise<ValidationResult> {
// Defense-in-depth: also guarded in call() for direct callers.
if (isWindowsSandboxPolicyViolation()) {
return {
result: false,
message: WINDOWS_SANDBOX_POLICY_REFUSAL,
errorCode: 11
};
}
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: PowerShellToolInput, context: Parameters<Tool['checkPermissions']>[1]): Promise<PermissionResult> {
return await powershellToolHasPermission(input, context);
},
renderToolUseMessage,
renderToolUseProgressMessage,
renderToolUseQueuedMessage,
renderToolResultMessage,
renderToolUseErrorMessage,
mapToolResultToToolResultBlockParam({
interrupted,
stdout,
stderr,
isImage,
persistedOutputPath,
persistedOutputSize,
backgroundTaskId,
backgroundedByUser,
assistantAutoBackgrounded
}: Out, toolUseID: string): ToolResultBlockParam {
// 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 (persistedOutputPath) {
const trimmed = stdout ? stdout.replace(/^(\s*\n)+/, '').trimEnd() : '';
const preview = generatePreview(trimmed, PREVIEW_SIZE_BYTES);
processedStdout = buildLargeToolResultMessage({
filepath: persistedOutputPath,
originalSize: persistedOutputSize ?? 0,
isJson: false,
preview: preview.preview,
hasMore: preview.hasMore
});
} else if (stdout) {
processedStdout = stdout.replace(/^(\s*\n)+/, '');
processedStdout = processedStdout.trimEnd();
}
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' as const,
content: [processedStdout, errorMessage, backgroundInfo].filter(Boolean).join('\n'),
is_error: interrupted
};
},
async call(input: PowerShellToolInput, toolUseContext: Parameters<Tool['call']>[1], _canUseTool?: CanUseToolFn, _parentMessage?: AssistantMessage, onProgress?: ToolCallProgress<PowerShellProgress>): Promise<{
data: Out;
}> {
// Load-bearing guard: promptShellExecution.ts and processBashCommand.tsx
// call PowerShellTool.call() directly, bypassing validateInput. This is
// the check that covers ALL callers. See isWindowsSandboxPolicyViolation
// comment for the policy rationale.
if (isWindowsSandboxPolicyViolation()) {
throw new Error(WINDOWS_SANDBOX_POLICY_REFUSAL);
}
const {
abortController,
setAppState,
setToolJSX
} = toolUseContext;
const isMainThread = !toolUseContext.agentId;
let progressCounter = 0;
try {
const commandGenerator = runPowerShellCommand({
input,
abortController,
// Use the always-shared task channel so async agents' background
// shell tasks are actually registered (and killable on agent exit).
setAppState: toolUseContext.setAppStateForTasks ?? setAppState,
setToolJSX,
preventCwdChanges: !isMainThread,
isMainThread,
toolUseId: toolUseContext.toolUseId,
agentId: toolUseContext.agentId
});
let generatorResult;
do {
generatorResult = await commandGenerator.next();
if (!generatorResult.done && onProgress) {
const progress = generatorResult.value;
onProgress({
toolUseID: `ps-progress-${progressCounter++}`,
data: {
type: 'powershell_progress',
output: progress.output,
fullOutput: progress.fullOutput,
elapsedTimeSeconds: progress.elapsedTimeSeconds,
totalLines: progress.totalLines,
totalBytes: progress.totalBytes,
timeoutMs: progress.timeoutMs,
taskId: progress.taskId
}
});
}
} while (!generatorResult.done);
const result = generatorResult.value;
// Feed git/PR usage metrics (same counters as BashTool). PS invokes
// git/gh/glab/curl as external binaries with identical syntax, so the
// shell-agnostic regex detection in trackGitOperations works as-is.
// Called before the backgroundTaskId early-return so backgrounded
// commands are counted too (matches BashTool.tsx:912).
//
// Pre-flight sentinel guard: the two PS pre-flight paths (pwsh-not-found,
// exec-spawn-catch) return code: 0 + empty stdout + stderr so call() can
// surface stderr gracefully instead of throwing ShellError. But
// gitOperationTracking.ts:48 treats code 0 as success and would
// regex-match the command, mis-counting a command that never ran.
// BashTool is safe β its pre-flight goes through createFailedCommand
// (code: 1) so tracking early-returns. Skip tracking on this sentinel.
const isPreFlightSentinel = result.code === 0 && !result.stdout && result.stderr && !result.backgroundTaskId;
if (!isPreFlightSentinel) {
trackGitOperations(input.command, result.code, result.stdout);
}
// Distinguish user-driven interrupt (new message submitted) from other
// interrupted states. Only user-interrupt should suppress ShellError β
// timeout-kill or process-kill with isError should still throw.
// Matches BashTool's isInterrupt.
const isInterrupt = result.interrupted && abortController.signal.reason === 'interrupt';
// Only the main thread tracks/resets cwd; agents have their own cwd
// isolation. Matches BashTool's !preventCwdChanges guard.
// Runs before the backgroundTaskId early-return: a command may change
// CWD before being backgrounded (e.g. `Set-Location C:\temp;
// Start-Sleep 60`), and BashTool has no such early return β its
// backgrounded results flow through resetCwdIfOutsideProject at :945.
let stderrForShellReset = '';
if (isMainThread) {
const appState = toolUseContext.getAppState();
if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {
stderrForShellReset = stdErrAppendShellResetMessage('');
}
}
// If backgrounded, return immediately with task ID. Strip hints first
// so interrupt-backgrounded fullOutput doesn't leak the tag to the
// model (BashTool has no early return, so all paths flow through its
// single extraction site).
if (result.backgroundTaskId) {
const bgExtracted = extractClaudeCodeHints(result.stdout || '', input.command);
if (isMainThread && bgExtracted.hints.length > 0) {
for (const hint of bgExtracted.hints) maybeRecordPluginHint(hint);
}
return {
data: {
stdout: bgExtracted.stripped,
stderr: [result.stderr || '', stderrForShellReset].filter(Boolean).join('\n'),
interrupted: false,
backgroundTaskId: result.backgroundTaskId,
backgroundedByUser: result.backgroundedByUser,
assistantAutoBackgrounded: result.assistantAutoBackgrounded
}
};
}
const stdoutAccumulator = new EndTruncatingAccumulator();
const processedStdout = (result.stdout || '').trimEnd();
stdoutAccumulator.append(processedStdout + EOL);
// Interpret exit code using semantic rules. PS-native cmdlets (Select-String,
// Compare-Object, Test-Path) exit 0 on no-match so they always hit the default
// here. This primarily handles external .exe's (grep, rg, findstr, fc, robocopy)
// where non-zero can mean "no match" / "files copied" rather than failure.
const interpretation = interpretCommandResult(input.command, result.code, processedStdout, result.stderr || '');
// getErrorParts() in toolErrors.ts already prepends 'Exit code N'
// from error.code when building the ShellError message. Do not
// duplicate it into stdout here (BashTool's append at :939 is dead
// code β it throws before stdoutAccumulator.toString() is read).
let stdout = stripEmptyLines(stdoutAccumulator.toString());
// 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(stdout, input.command);
stdout = extracted.stripped;
if (isMainThread && extracted.hints.length > 0) {
for (const hint of extracted.hints) maybeRecordPluginHint(hint);
}
// preSpawnError means exec() succeeded but the inner shell failed before
// the command ran (e.g. CWD deleted). createFailedCommand sets code=1,
// which interpretCommandResult can mistake for grep-no-match / findstr
// string-not-found. Throw it directly. Matches BashTool.tsx:957.
if (result.preSpawnError) {
throw new Error(result.preSpawnError);
}
if (interpretation.isError && !isInterrupt) {
throw new ShellError(stdout, result.stderr || '', result.code, result.interrupted);
}
// Large output: file on disk has more than getMaxOutputLength() bytes.
// stdout already contains the first chunk. Copy the output file to the
// tool-results dir so the model can read it via FileRead. If > 64 MB,
// truncate after copying. Matches BashTool.tsx:983-1005.
//
// Placed AFTER the preSpawnError/ShellError throws (matches BashTool's
// ordering, where persistence is post-try/finally): a failing command
// that also produced >maxOutputLength bytes would otherwise do 3-4 disk
// syscalls, store to tool-results/, then throw β orphaning the file.
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
}
}
// Cap image dimensions + size if present (CC-304 β see
// resizeShellImageOutput). Scope the decoded buffer so it can be
// reclaimed before we build the output object.
let isImage = isImageOutput(stdout);
let compressedStdout = stdout;
if (isImage) {
const resized = await resizeShellImageOutput(stdout, result.outputFilePath, persistedOutputSize);
if (resized) {
compressedStdout = resized;
} else {
// Parse failed (e.g. multi-line stdout after the data URL). 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 finalStderr = [result.stderr || '', stderrForShellReset].filter(Boolean).join('\n');
logEvent('tengu_powershell_tool_command_executed', {
command_type: getCommandTypeForLogging(input.command),
stdout_length: compressedStdout.length,
stderr_length: finalStderr.length,
exit_code: result.code,
interrupted: result.interrupted
});
return {
data: {
stdout: compressedStdout,
stderr: finalStderr,
interrupted: result.interrupted,
returnCodeInterpretation: interpretation.message,
isImage,
persistedOutputPath,
persistedOutputSize
}
};
} finally {
if (setToolJSX) setToolJSX(null);
}
},
isResultTruncated(output: Out): boolean {
return isOutputLineTruncated(output.stdout) || isOutputLineTruncated(output.stderr);
}
} satisfies ToolDef<InputSchema, Out>);
async function* runPowerShellCommand({
input,
abortController,
setAppState,
setToolJSX,
preventCwdChanges,
isMainThread,
toolUseId,
agentId
}: {
input: PowerShellToolInput;
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,
dangerouslyDisableSandbox
} = input;
const timeoutMs = Math.min(timeout || getDefaultTimeoutMs(), getMaxTimeoutMs());
let fullOutput = '';
let lastProgressOutput = '';
let lastTotalLines = 0;
let lastTotalBytes = 0;
let backgroundShellId: string | undefined = undefined;
let interruptBackgroundingStarted = false;
let assistantAutoBackgrounded = false;
// Progress signal: resolved when backgroundShellId is set in the async
// .then() path, waking the generator's Promise.race immediately instead of
// waiting for the next setTimeout tick (matches BashTool pattern).
let resolveProgress: (() => void) | null = null;
function createProgressSignal(): Promise<null> {
return new Promise<null>(resolve => {
resolveProgress = () => resolve(null);
});
}
const shouldAutoBackground = !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command);
const powershellPath = await getCachedPowerShellPath();
if (!powershellPath) {
// Pre-flight failure: pwsh not installed. Return code 0 so call() surfaces
// this as a graceful stderr message rather than throwing ShellError β the
// command never ran, so there is no meaningful non-zero exit to report.
return {
stdout: '',
stderr: 'PowerShell is not available on this system.',
code: 0,
interrupted: false
};
}
let shellCommand: Awaited<ReturnType<typeof exec>>;
try {
shellCommand = await exec(command, abortController.signal, 'powershell', {
timeout: timeoutMs,
onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {
lastProgressOutput = lastLines;
fullOutput = allLines;
lastTotalLines = totalLines;
lastTotalBytes = isIncomplete ? totalBytes : 0;
},
preventCwdChanges,
// Sandbox works on Linux/macOS/WSL2 β pwsh there is a native binary and
// SandboxManager.wrapWithSandbox wraps it same as bash (Shell.ts uses
// /bin/sh for the outer spawn to parse the POSIX-quoted bwrap/sandbox-exec
// string). On Windows native, sandbox is unsupported; shouldUseSandbox()
// returns false via isSandboxingEnabled() β isSupportedPlatform() β false.
// The explicit platform check is redundant-but-obvious.
shouldUseSandbox: getPlatform() === 'windows' ? false : shouldUseSandbox({
command,
dangerouslyDisableSandbox
}),
shouldAutoBackground
});
} catch (e) {
logError(e);
// Pre-flight failure: spawn/exec rejected before the command ran. Use
// code 0 so call() returns stderr gracefully instead of throwing ShellError.
return {
stdout: '',
stderr: `Failed to execute PowerShell command: ${getErrorMessage(e)}`,
code: 0,
interrupted: false
};
}
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: () => {
throw new Error('getAppState not available in runPowerShellCommand context');
},
setAppState
});
return handle.taskId;
}
// Helper to start backgrounding with 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, the generator waits for the current setTimeout to fire
// (up to ~1s) before noticing the backgrounding. Matches BashTool.
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
if (shellCommand.onTimeout && shouldAutoBackground) {
shellCommand.onTimeout(backgroundFn => {
startBackgrounding('tengu_powershell_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_powershell_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)
if (run_in_background === true && !isBackgroundTasksDisabled) {
const shellId = await spawnBackgroundTask();
logEvent('tengu_powershell_command_explicitly_backgrounded', {
command_type: getCommandTypeForLogging(command)
});
return {
stdout: '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: shellId
};
}
// Start polling the output file for progress
TaskOutput.startPolling(shellCommand.taskOutput.taskId);
// Set up progress yielding with periodic checks
const startTime = Date.now();
let nextProgressTime = startTime + PROGRESS_THRESHOLD_MS;
let foregroundTaskId: string | undefined = undefined;
// Progress loop: wrap in try/finally so stopPolling is called on every exit
// path β normal completion, timeout/interrupt backgrounding, and Ctrl+B
// (matches BashTool pattern; see PR #18887 review thread at :560)
try {
while (true) {
const now = Date.now();
const timeUntilNextProgress = Math.max(0, nextProgressTime - now);
const progressSignal = createProgressSignal();
const result = await Promise.race([resultPromise, new Promise<null>(resolve => setTimeout(r => r(null), timeUntilNextProgress, resolve).unref()), 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;
}
// Command completed β cleanup stream listeners here. The finally
// block's guard (!backgroundShellId && status !== 'backgrounded')
// correctly skips cleanup for *running* backgrounded tasks, but
// in this race the process is done. Matches BashTool.tsx:1399.
shellCommand.cleanup();
return fixedResult;
}
// Command has completed
return result;
}
// Check if command was backgrounded (by timeout or interrupt)
if (backgroundShellId) {
return {
stdout: interruptBackgroundingStarted ? fullOutput : '',
stderr: '',
code: 0,
interrupted: false,
backgroundTaskId: backgroundShellId,
assistantAutoBackgrounded
};
}
// User submitted a new message - background instead of killing
if (abortController.signal.aborted && abortController.signal.reason === 'interrupt' && !interruptBackgroundingStarted) {
interruptBackgroundingStarted = true;
if (!isBackgroundTasksDisabled) {
startBackgrounding('tengu_powershell_command_interrupt_backgrounded');
// Reloop so the backgroundShellId check (above) catches the sync
// foregroundTaskIdβbackground path. Without this, we fall through
// to the Ctrl+B check below, which matches status==='backgrounded'
// and incorrectly returns backgroundedByUser:true. (bugs 020/021)
continue;
}
shellCommand.kill();
}
// Check if this foreground task was backgrounded via backgroundAll() (ctrl+b)
if (foregroundTaskId) {
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 backgrounding UI hint after threshold
if (!isBackgroundTasksDisabled && backgroundShellId === undefined && elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 && setToolJSX) {
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)
};
nextProgressTime = Date.now() + PROGRESS_INTERVAL_MS;
}
} finally {
TaskOutput.stopPolling(shellCommand.taskOutput.taskId);
// Ensure cleanup runs on every exit path (success, rejection, abort).
// Skip when backgrounded β LocalShellTask owns cleanup for those.
// Matches main #21105.
if (!backgroundShellId && shellCommand.status !== 'backgrounded') {
if (foregroundTaskId) {
unregisterForeground(foregroundTaskId, setAppState);
}
shellCommand.cleanup();
}
}
}
//# 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","SetToolJSXFn","Tool","ToolCallProgress","ValidationResult","buildTool","ToolDef","backgroundExistingForegroundTask","markTaskNotified","registerForeground","spawnShellTask","unregisterForeground","AgentId","AssistantMessage","extractClaudeCodeHints","isEnvTruthy","errorMessage","getErrorMessage","ShellError","lazySchema","logError","PermissionResult","getPlatform","maybeRecordPluginHint","exec","ExecResult","SandboxManager","semanticBoolean","semanticNumber","getCachedPowerShellPath","EndTruncatingAccumulator","getTaskOutputPath","TaskOutput","isOutputLineTruncated","buildLargeToolResultMessage","ensureToolResultsDir","generatePreview","getToolResultPath","PREVIEW_SIZE_BYTES","shouldUseSandbox","BackgroundHint","buildImageToolResult","isImageOutput","resetCwdIfOutsideProject","resizeShellImageOutput","stdErrAppendShellResetMessage","stripEmptyLines","trackGitOperations","interpretCommandResult","powershellToolHasPermission","getDefaultTimeoutMs","getMaxTimeoutMs","getPrompt","hasSyncSecurityConcerns","isReadOnlyCommand","resolveToCanonical","POWERSHELL_TOOL_NAME","renderToolResultMessage","renderToolUseErrorMessage","renderToolUseMessage","renderToolUseProgressMessage","renderToolUseQueuedMessage","EOL","PS_SEARCH_COMMANDS","Set","PS_READ_COMMANDS","PS_SEMANTIC_NEUTRAL_COMMANDS","isSearchOrReadPowerShellCommand","command","isSearch","isRead","trimmed","trim","parts","split","filter","Boolean","length","hasSearch","hasRead","hasNonNeutralCommand","part","baseCommand","canonical","has","isPartSearch","isPartRead","PROGRESS_THRESHOLD_MS","PROGRESS_INTERVAL_MS","ASSISTANT_BLOCKING_BUDGET_MS","DISALLOWED_AUTO_BACKGROUND_COMMANDS","isAutobackgroundingAllowed","firstWord","includes","detectBlockedSleepPattern","first","m","secs","parseInt","rest","slice","replace","WINDOWS_SANDBOX_POLICY_REFUSAL","isWindowsSandboxPolicyViolation","isSandboxEnabledInSettings","areUnsandboxedCommandsAllowed","isBackgroundTasksDisabled","process","env","CLAUDE_CODE_DISABLE_BACKGROUND_TASKS","fullInputSchema","strictObject","string","describe","timeout","number","optional","description","run_in_background","boolean","dangerouslyDisableSandbox","inputSchema","omit","InputSchema","ReturnType","PowerShellToolInput","infer","outputSchema","object","stdout","stderr","interrupted","returnCodeInterpretation","isImage","persistedOutputPath","persistedOutputSize","backgroundTaskId","backgroundedByUser","assistantAutoBackgrounded","OutputSchema","Out","PowerShellProgress","COMMON_BACKGROUND_COMMANDS","const","getCommandTypeForLogging","cmd","toLowerCase","PowerShellTool","name","searchHint","maxResultSizeChars","strict","Partial","Promise","prompt","isConcurrencySafe","input","isReadOnly","isSearchOrReadCommand","toAutoClassifierInput","userFacingName","getToolUseSummary","getActivityDescription","desc","isEnabled","validateInput","result","message","errorCode","sleepPattern","checkPermissions","context","Parameters","mapToolResultToToolResultBlockParam","toolUseID","block","processedStdout","trimEnd","preview","filepath","originalSize","isJson","hasMore","backgroundInfo","outputPath","tool_use_id","type","content","join","is_error","call","toolUseContext","_canUseTool","_parentMessage","onProgress","data","Error","abortController","setAppState","setToolJSX","isMainThread","agentId","progressCounter","commandGenerator","runPowerShellCommand","setAppStateForTasks","preventCwdChanges","toolUseId","generatorResult","next","done","progress","value","output","fullOutput","elapsedTimeSeconds","totalLines","totalBytes","timeoutMs","taskId","isPreFlightSentinel","code","isInterrupt","signal","reason","stderrForShellReset","appState","getAppState","toolPermissionContext","bgExtracted","hints","hint","stripped","stdoutAccumulator","append","interpretation","toString","extracted","preSpawnError","isError","MAX_PERSISTED_SIZE","outputFilePath","outputTaskId","fileStat","size","dest","compressedStdout","resized","finalStderr","command_type","stdout_length","stderr_length","exit_code","isResultTruncated","AbortController","f","prev","AsyncGenerator","Math","min","lastProgressOutput","lastTotalLines","lastTotalBytes","backgroundShellId","undefined","interruptBackgroundingStarted","resolveProgress","createProgressSignal","resolve","shouldAutoBackground","powershellPath","shellCommand","Awaited","lastLines","allLines","isIncomplete","e","resultPromise","spawnBackgroundTask","handle","startBackgrounding","eventName","backgroundFn","shellId","foregroundTaskId","then","onTimeout","setTimeout","status","unref","startPolling","taskOutput","startTime","Date","now","nextProgressTime","timeUntilNextProgress","max","progressSignal","race","r","fixedResult","stdoutToFile","outputFileRedundant","path","outputFileSize","cleanup","aborted","kill","elapsed","elapsedSeconds","floor","jsx","shouldHidePromptInput","shouldContinueAnimation","showSpinner","stopPolling"],"sources":["PowerShellTool.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 type {\n  SetToolJSXFn,\n  Tool,\n  ToolCallProgress,\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 { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport {\n  errorMessage as getErrorMessage,\n  ShellError,\n} from '../../utils/errors.js'\nimport { truncate } from '../../utils/format.js'\nimport { lazySchema } from '../../utils/lazySchema.js'\nimport { logError } from '../../utils/log.js'\nimport type { PermissionResult } from '../../utils/permissions/PermissionResult.js'\nimport { getPlatform } from '../../utils/platform.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 { getCachedPowerShellPath } from '../../utils/shell/powershellDetection.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 { shouldUseSandbox } from '../BashTool/shouldUseSandbox.js'\nimport { BackgroundHint } from '../BashTool/UI.js'\nimport {\n  buildImageToolResult,\n  isImageOutput,\n  resetCwdIfOutsideProject,\n  resizeShellImageOutput,\n  stdErrAppendShellResetMessage,\n  stripEmptyLines,\n} from '../BashTool/utils.js'\nimport { trackGitOperations } from '../shared/gitOperationTracking.js'\nimport { interpretCommandResult } from './commandSemantics.js'\nimport { powershellToolHasPermission } from './powershellPermissions.js'\nimport { getDefaultTimeoutMs, getMaxTimeoutMs, getPrompt } from './prompt.js'\nimport {\n  hasSyncSecurityConcerns,\n  isReadOnlyCommand,\n  resolveToCanonical,\n} from './readOnlyValidation.js'\nimport { POWERSHELL_TOOL_NAME } from './toolName.js'\nimport {\n  renderToolResultMessage,\n  renderToolUseErrorMessage,\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n} from './UI.js'\n\n// Never use os.EOL for terminal output — \\r\\n on Windows breaks Ink rendering\nconst EOL = '\\n'\n\n/**\n * PowerShell search commands (grep equivalents) for collapsible display.\n * Stored as canonical (lowercase) cmdlet names.\n */\nconst PS_SEARCH_COMMANDS = new Set([\n  'select-string', // grep equivalent\n  'get-childitem', // find equivalent (with -Recurse)\n  'findstr', // native Windows search\n  'where.exe', // native Windows which\n])\n\n/**\n * PowerShell read/view commands for collapsible display.\n * Stored as canonical (lowercase) cmdlet names.\n */\nconst PS_READ_COMMANDS = new Set([\n  'get-content', // cat equivalent\n  'get-item', // file info\n  'test-path', // test -e equivalent\n  'resolve-path', // realpath equivalent\n  'get-process', // ps equivalent\n  'get-service', // system info\n  'get-childitem', // ls/dir equivalent (also search when recursive)\n  'get-location', // pwd equivalent\n  'get-filehash', // checksum\n  'get-acl', // permissions info\n  'format-hex', // hexdump equivalent\n])\n\n/**\n * PowerShell semantic-neutral commands that don't change the search/read nature.\n */\nconst PS_SEMANTIC_NEUTRAL_COMMANDS = new Set([\n  'write-output', // echo equivalent\n  'write-host',\n])\n\n/**\n * Checks if a PowerShell command is a search or read operation.\n * Used to determine if the command should be collapsed in the UI.\n */\nfunction isSearchOrReadPowerShellCommand(command: string): {\n  isSearch: boolean\n  isRead: boolean\n} {\n  const trimmed = command.trim()\n  if (!trimmed) {\n    return { isSearch: false, isRead: false }\n  }\n\n  // Simple split on statement separators and pipe operators\n  // This is a sync function so we use a lightweight approach\n  const parts = trimmed.split(/\\s*[;|]\\s*/).filter(Boolean)\n\n  if (parts.length === 0) {\n    return { isSearch: false, isRead: false }\n  }\n\n  let hasSearch = false\n  let hasRead = false\n  let hasNonNeutralCommand = false\n\n  for (const part of parts) {\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    const canonical = resolveToCanonical(baseCommand)\n\n    if (PS_SEMANTIC_NEUTRAL_COMMANDS.has(canonical)) {\n      continue\n    }\n\n    hasNonNeutralCommand = true\n\n    const isPartSearch = PS_SEARCH_COMMANDS.has(canonical)\n    const isPartRead = PS_READ_COMMANDS.has(canonical)\n\n    if (!isPartSearch && !isPartRead) {\n      return { isSearch: false, isRead: false }\n    }\n\n    if (isPartSearch) hasSearch = true\n    if (isPartRead) hasRead = true\n  }\n\n  if (!hasNonNeutralCommand) {\n    return { isSearch: false, isRead: false }\n  }\n\n  return { isSearch: hasSearch, isRead: hasRead }\n}\n\n// Progress display constants\nconst PROGRESS_THRESHOLD_MS = 2000\nconst PROGRESS_INTERVAL_MS = 1000\n// In assistant mode, blocking commands auto-background after this many ms in the main agent\nconst ASSISTANT_BLOCKING_BUDGET_MS = 15_000\n\n// Commands that should not be auto-backgrounded (canonical lowercase).\n// 'sleep' is a PS built-in alias for Start-Sleep but not in COMMON_ALIASES,\n// so list both forms.\nconst DISALLOWED_AUTO_BACKGROUND_COMMANDS = [\n  'start-sleep', // Start-Sleep should run in foreground unless explicitly backgrounded\n  'sleep',\n]\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 Start-Sleep)\n */\nfunction isAutobackgroundingAllowed(command: string): boolean {\n  const firstWord = command.trim().split(/\\s+/)[0]\n  if (!firstWord) return true\n  const canonical = resolveToCanonical(firstWord)\n  return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(canonical)\n}\n\n/**\n * PS-flavored port of BashTool's detectBlockedSleepPattern.\n * Catches `Start-Sleep N`, `Start-Sleep -Seconds N`, `sleep N` (built-in alias)\n * as the first statement. Does NOT block `Start-Sleep -Milliseconds` (sub-second\n * pacing is fine) or float seconds (legit rate limiting).\n */\nexport function detectBlockedSleepPattern(command: string): string | null {\n  // First statement only — split on PS statement separators: `;`, `|`,\n  // `&`/`&&`/`||` (pwsh 7+), and newline (PS's primary separator). This is\n  // intentionally shallow — sleep inside script blocks, subshells, or later\n  // pipeline stages is fine. Matches BashTool's splitCommandWithOperators\n  // intent (src/utils/bash/commands.ts) without a full PS parser.\n  const first =\n    command\n      .trim()\n      .split(/[;|&\\r\\n]/)[0]\n      ?.trim() ?? ''\n  // Match: Start-Sleep N, Start-Sleep -Seconds N, Start-Sleep -s N, sleep N\n  // (case-insensitive; -Seconds can be abbreviated to -s per PS convention)\n  const m = /^(?:start-sleep|sleep)(?:\\s+-s(?:econds)?)?\\s+(\\d+)\\s*$/i.exec(\n    first,\n  )\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  const rest = command\n    .trim()\n    .slice(first.length)\n    .replace(/^[\\s;|&]+/, '')\n  return rest\n    ? `Start-Sleep ${secs} followed by: ${rest}`\n    : `standalone Start-Sleep ${secs}`\n}\n\n/**\n * On Windows native, sandbox is unavailable (bwrap/sandbox-exec are\n * POSIX-only). If enterprise policy has sandbox.enabled AND forbids\n * unsandboxed commands, PowerShell cannot comply — refuse execution\n * rather than silently bypass the policy. On Linux/macOS/WSL2, pwsh\n * runs as a native binary under the sandbox same as bash, so this\n * gate does not apply.\n *\n * Checked in BOTH validateInput (clean tool-runner error) and call()\n * (covers direct callers like promptShellExecution.ts that skip\n * validateInput). The call() guard is the load-bearing one.\n */\nconst WINDOWS_SANDBOX_POLICY_REFUSAL =\n  'Enterprise policy requires sandboxing, but sandboxing is not available on native Windows. Shell command execution is blocked on this platform by policy.'\nfunction isWindowsSandboxPolicyViolation(): boolean {\n  return (\n    getPlatform() === 'windows' &&\n    SandboxManager.isSandboxEnabledInSettings() &&\n    !SandboxManager.areUnsandboxedCommandsAllowed()\n  )\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 PowerShell 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(\n        'Clear, concise description of what this command does in active voice.',\n      ),\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  }),\n)\n\n// Conditionally remove run_in_background from schema when background tasks are disabled\nconst inputSchema = lazySchema(() =>\n  isBackgroundTasksDisabled\n    ? fullInputSchema().omit({ run_in_background: true })\n    : fullInputSchema(),\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 PowerShellToolInput = z.infer<ReturnType<typeof fullInputSchema>>\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    interrupted: z.boolean().describe('Whether the command was interrupted'),\n    returnCodeInterpretation: z\n      .string()\n      .optional()\n      .describe(\n        'Semantic interpretation for non-error exit codes with special meaning',\n      ),\n    isImage: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if stdout contains image data'),\n    persistedOutputPath: z\n      .string()\n      .optional()\n      .describe('Path to persisted full output when too large for inline'),\n    persistedOutputSize: z\n      .number()\n      .optional()\n      .describe('Total output size in bytes when persisted'),\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 the command was auto-backgrounded by the assistant-mode blocking budget',\n      ),\n  }),\n)\ntype OutputSchema = ReturnType<typeof outputSchema>\nexport type Out = z.infer<OutputSchema>\n\nimport type { PowerShellProgress } from '../../types/tools.js'\n\nexport type { PowerShellProgress } from '../../types/tools.js'\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  'Invoke-WebRequest',\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 trimmed = command.trim()\n  const firstWord = trimmed.split(/\\s+/)[0] || ''\n\n  for (const cmd of COMMON_BACKGROUND_COMMANDS) {\n    if (firstWord.toLowerCase() === cmd.toLowerCase()) {\n      return cmd 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\nexport const PowerShellTool = buildTool({\n  name: POWERSHELL_TOOL_NAME,\n  searchHint: 'execute Windows PowerShell commands',\n  maxResultSizeChars: 30_000,\n  strict: true,\n\n  async description({\n    description,\n  }: Partial<PowerShellToolInput>): Promise<string> {\n    return description || 'Run PowerShell command'\n  },\n\n  async prompt(): Promise<string> {\n    return getPrompt()\n  },\n\n  isConcurrencySafe(input: PowerShellToolInput): boolean {\n    return this.isReadOnly?.(input) ?? false\n  },\n\n  isSearchOrReadCommand(input: Partial<PowerShellToolInput>): {\n    isSearch: boolean\n    isRead: boolean\n  } {\n    if (!input.command) {\n      return { isSearch: false, isRead: false }\n    }\n    return isSearchOrReadPowerShellCommand(input.command)\n  },\n\n  isReadOnly(input: PowerShellToolInput): boolean {\n    // Check sync security heuristics before declaring read-only.\n    // The full AST parse is async and unavailable here, so we use\n    // regex-based detection of subexpressions, splatting, member\n    // invocations, and assignments — matching BashTool's pattern of\n    // checking security concerns before cmdlet allowlist evaluation.\n    if (hasSyncSecurityConcerns(input.command)) {\n      return false\n    }\n    // NOTE: This calls isReadOnlyCommand without the parsed AST. Without the\n    // AST, isReadOnlyCommand cannot split pipelines/statements and will return\n    // false for anything but the simplest single-token commands. This is a\n    // known limitation of the sync Tool.isReadOnly() interface — the real\n    // read-only auto-allow happens async in powershellToolHasPermission (step\n    // 4.5) where the parsed AST is available.\n    return isReadOnlyCommand(input.command)\n  },\n  toAutoClassifierInput(input) {\n    return input.command\n  },\n\n  get inputSchema(): InputSchema {\n    return inputSchema()\n  },\n\n  get outputSchema(): OutputSchema {\n    return outputSchema()\n  },\n\n  userFacingName(): string {\n    return 'PowerShell'\n  },\n\n  getToolUseSummary(\n    input: Partial<PowerShellToolInput> | undefined,\n  ): string | null {\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\n  getActivityDescription(\n    input: Partial<PowerShellToolInput> | undefined,\n  ): string {\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\n  isEnabled(): boolean {\n    return true\n  },\n\n  async validateInput(input: PowerShellToolInput): Promise<ValidationResult> {\n    // Defense-in-depth: also guarded in call() for direct callers.\n    if (isWindowsSandboxPolicyViolation()) {\n      return {\n        result: false,\n        message: WINDOWS_SANDBOX_POLICY_REFUSAL,\n        errorCode: 11,\n      }\n    }\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\n  async checkPermissions(\n    input: PowerShellToolInput,\n    context: Parameters<Tool['checkPermissions']>[1],\n  ): Promise<PermissionResult> {\n    return await powershellToolHasPermission(input, context)\n  },\n\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n  renderToolResultMessage,\n  renderToolUseErrorMessage,\n\n  mapToolResultToToolResultBlockParam(\n    {\n      interrupted,\n      stdout,\n      stderr,\n      isImage,\n      persistedOutputPath,\n      persistedOutputSize,\n      backgroundTaskId,\n      backgroundedByUser,\n      assistantAutoBackgrounded,\n    }: Out,\n    toolUseID: string,\n  ): ToolResultBlockParam {\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\n    if (persistedOutputPath) {\n      const trimmed = stdout ? stdout.replace(/^(\\s*\\n)+/, '').trimEnd() : ''\n      const preview = generatePreview(trimmed, 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    } else if (stdout) {\n      processedStdout = stdout.replace(/^(\\s*\\n)+/, '')\n      processedStdout = processedStdout.trimEnd()\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' as const,\n      content: [processedStdout, errorMessage, backgroundInfo]\n        .filter(Boolean)\n        .join('\\n'),\n      is_error: interrupted,\n    }\n  },\n\n  async call(\n    input: PowerShellToolInput,\n    toolUseContext: Parameters<Tool['call']>[1],\n    _canUseTool?: CanUseToolFn,\n    _parentMessage?: AssistantMessage,\n    onProgress?: ToolCallProgress<PowerShellProgress>,\n  ): Promise<{ data: Out }> {\n    // Load-bearing guard: promptShellExecution.ts and processBashCommand.tsx\n    // call PowerShellTool.call() directly, bypassing validateInput. This is\n    // the check that covers ALL callers. See isWindowsSandboxPolicyViolation\n    // comment for the policy rationale.\n    if (isWindowsSandboxPolicyViolation()) {\n      throw new Error(WINDOWS_SANDBOX_POLICY_REFUSAL)\n    }\n\n    const { abortController, setAppState, setToolJSX } = toolUseContext\n\n    const isMainThread = !toolUseContext.agentId\n\n    let progressCounter = 0\n\n    try {\n      const commandGenerator = runPowerShellCommand({\n        input,\n        abortController,\n        // Use the always-shared task channel so async agents' background\n        // shell tasks are actually registered (and killable on agent exit).\n        setAppState: toolUseContext.setAppStateForTasks ?? setAppState,\n        setToolJSX,\n        preventCwdChanges: !isMainThread,\n        isMainThread,\n        toolUseId: toolUseContext.toolUseId,\n        agentId: toolUseContext.agentId,\n      })\n\n      let generatorResult\n      do {\n        generatorResult = await commandGenerator.next()\n        if (!generatorResult.done && onProgress) {\n          const progress = generatorResult.value\n          onProgress({\n            toolUseID: `ps-progress-${progressCounter++}`,\n            data: {\n              type: 'powershell_progress',\n              output: progress.output,\n              fullOutput: progress.fullOutput,\n              elapsedTimeSeconds: progress.elapsedTimeSeconds,\n              totalLines: progress.totalLines,\n              totalBytes: progress.totalBytes,\n              timeoutMs: progress.timeoutMs,\n              taskId: progress.taskId,\n            },\n          })\n        }\n      } while (!generatorResult.done)\n\n      const result = generatorResult.value\n\n      // Feed git/PR usage metrics (same counters as BashTool). PS invokes\n      // git/gh/glab/curl as external binaries with identical syntax, so the\n      // shell-agnostic regex detection in trackGitOperations works as-is.\n      // Called before the backgroundTaskId early-return so backgrounded\n      // commands are counted too (matches BashTool.tsx:912).\n      //\n      // Pre-flight sentinel guard: the two PS pre-flight paths (pwsh-not-found,\n      // exec-spawn-catch) return code: 0 + empty stdout + stderr so call() can\n      // surface stderr gracefully instead of throwing ShellError. But\n      // gitOperationTracking.ts:48 treats code 0 as success and would\n      // regex-match the command, mis-counting a command that never ran.\n      // BashTool is safe — its pre-flight goes through createFailedCommand\n      // (code: 1) so tracking early-returns. Skip tracking on this sentinel.\n      const isPreFlightSentinel =\n        result.code === 0 &&\n        !result.stdout &&\n        result.stderr &&\n        !result.backgroundTaskId\n      if (!isPreFlightSentinel) {\n        trackGitOperations(input.command, result.code, result.stdout)\n      }\n\n      // Distinguish user-driven interrupt (new message submitted) from other\n      // interrupted states. Only user-interrupt should suppress ShellError —\n      // timeout-kill or process-kill with isError should still throw.\n      // Matches BashTool's isInterrupt.\n      const isInterrupt =\n        result.interrupted && abortController.signal.reason === 'interrupt'\n\n      // Only the main thread tracks/resets cwd; agents have their own cwd\n      // isolation. Matches BashTool's !preventCwdChanges guard.\n      // Runs before the backgroundTaskId early-return: a command may change\n      // CWD before being backgrounded (e.g. `Set-Location C:\\temp;\n      // Start-Sleep 60`), and BashTool has no such early return — its\n      // backgrounded results flow through resetCwdIfOutsideProject at :945.\n      let stderrForShellReset = ''\n      if (isMainThread) {\n        const appState = toolUseContext.getAppState()\n        if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {\n          stderrForShellReset = stdErrAppendShellResetMessage('')\n        }\n      }\n\n      // If backgrounded, return immediately with task ID. Strip hints first\n      // so interrupt-backgrounded fullOutput doesn't leak the tag to the\n      // model (BashTool has no early return, so all paths flow through its\n      // single extraction site).\n      if (result.backgroundTaskId) {\n        const bgExtracted = extractClaudeCodeHints(\n          result.stdout || '',\n          input.command,\n        )\n        if (isMainThread && bgExtracted.hints.length > 0) {\n          for (const hint of bgExtracted.hints) maybeRecordPluginHint(hint)\n        }\n        return {\n          data: {\n            stdout: bgExtracted.stripped,\n            stderr: [result.stderr || '', stderrForShellReset]\n              .filter(Boolean)\n              .join('\\n'),\n            interrupted: false,\n            backgroundTaskId: result.backgroundTaskId,\n            backgroundedByUser: result.backgroundedByUser,\n            assistantAutoBackgrounded: result.assistantAutoBackgrounded,\n          },\n        }\n      }\n\n      const stdoutAccumulator = new EndTruncatingAccumulator()\n      const processedStdout = (result.stdout || '').trimEnd()\n\n      stdoutAccumulator.append(processedStdout + EOL)\n\n      // Interpret exit code using semantic rules. PS-native cmdlets (Select-String,\n      // Compare-Object, Test-Path) exit 0 on no-match so they always hit the default\n      // here. This primarily handles external .exe's (grep, rg, findstr, fc, robocopy)\n      // where non-zero can mean \"no match\" / \"files copied\" rather than failure.\n      const interpretation = interpretCommandResult(\n        input.command,\n        result.code,\n        processedStdout,\n        result.stderr || '',\n      )\n\n      // getErrorParts() in toolErrors.ts already prepends 'Exit code N'\n      // from error.code when building the ShellError message. Do not\n      // duplicate it into stdout here (BashTool's append at :939 is dead\n      // code — it throws before stdoutAccumulator.toString() is read).\n\n      let stdout = stripEmptyLines(stdoutAccumulator.toString())\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(stdout, input.command)\n      stdout = extracted.stripped\n      if (isMainThread && extracted.hints.length > 0) {\n        for (const hint of extracted.hints) maybeRecordPluginHint(hint)\n      }\n\n      // preSpawnError means exec() succeeded but the inner shell failed before\n      // the command ran (e.g. CWD deleted). createFailedCommand sets code=1,\n      // which interpretCommandResult can mistake for grep-no-match / findstr\n      // string-not-found. Throw it directly. Matches BashTool.tsx:957.\n      if (result.preSpawnError) {\n        throw new Error(result.preSpawnError)\n      }\n      if (interpretation.isError && !isInterrupt) {\n        throw new ShellError(\n          stdout,\n          result.stderr || '',\n          result.code,\n          result.interrupted,\n        )\n      }\n\n      // Large output: file on disk has more than getMaxOutputLength() bytes.\n      // stdout already contains the first chunk. Copy the output file to the\n      // tool-results dir so the model can read it via FileRead. If > 64 MB,\n      // truncate after copying. Matches BashTool.tsx:983-1005.\n      //\n      // Placed AFTER the preSpawnError/ShellError throws (matches BashTool's\n      // ordering, where persistence is post-try/finally): a failing command\n      // that also produced >maxOutputLength bytes would otherwise do 3-4 disk\n      // syscalls, store to tool-results/, then throw — orphaning the file.\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      // Cap image dimensions + size if present (CC-304 — see\n      // resizeShellImageOutput). Scope the decoded buffer so it can be\n      // reclaimed before we build the output object.\n      let isImage = isImageOutput(stdout)\n      let compressedStdout = stdout\n      if (isImage) {\n        const resized = await resizeShellImageOutput(\n          stdout,\n          result.outputFilePath,\n          persistedOutputSize,\n        )\n        if (resized) {\n          compressedStdout = resized\n        } else {\n          // Parse failed (e.g. multi-line stdout after the data URL). Keep\n          // 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 finalStderr = [result.stderr || '', stderrForShellReset]\n        .filter(Boolean)\n        .join('\\n')\n\n      logEvent('tengu_powershell_tool_command_executed', {\n        command_type: getCommandTypeForLogging(input.command),\n        stdout_length: compressedStdout.length,\n        stderr_length: finalStderr.length,\n        exit_code: result.code,\n        interrupted: result.interrupted,\n      })\n\n      return {\n        data: {\n          stdout: compressedStdout,\n          stderr: finalStderr,\n          interrupted: result.interrupted,\n          returnCodeInterpretation: interpretation.message,\n          isImage,\n          persistedOutputPath,\n          persistedOutputSize,\n        },\n      }\n    } finally {\n      if (setToolJSX) setToolJSX(null)\n    }\n  },\n  isResultTruncated(output: Out): boolean {\n    return (\n      isOutputLineTruncated(output.stdout) ||\n      isOutputLineTruncated(output.stderr)\n    )\n  },\n} satisfies ToolDef<InputSchema, Out>)\n\nasync function* runPowerShellCommand({\n  input,\n  abortController,\n  setAppState,\n  setToolJSX,\n  preventCwdChanges,\n  isMainThread,\n  toolUseId,\n  agentId,\n}: {\n  input: PowerShellToolInput\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 {\n    command,\n    description,\n    timeout,\n    run_in_background,\n    dangerouslyDisableSandbox,\n  } = input\n  const timeoutMs = Math.min(\n    timeout || getDefaultTimeoutMs(),\n    getMaxTimeoutMs(),\n  )\n\n  let fullOutput = ''\n  let lastProgressOutput = ''\n  let lastTotalLines = 0\n  let lastTotalBytes = 0\n  let backgroundShellId: string | undefined = undefined\n  let interruptBackgroundingStarted = false\n  let assistantAutoBackgrounded = false\n\n  // Progress signal: resolved when backgroundShellId is set in the async\n  // .then() path, waking the generator's Promise.race immediately instead of\n  // waiting for the next setTimeout tick (matches BashTool pattern).\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  const shouldAutoBackground =\n    !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command)\n\n  const powershellPath = await getCachedPowerShellPath()\n  if (!powershellPath) {\n    // Pre-flight failure: pwsh not installed. Return code 0 so call() surfaces\n    // this as a graceful stderr message rather than throwing ShellError — the\n    // command never ran, so there is no meaningful non-zero exit to report.\n    return {\n      stdout: '',\n      stderr: 'PowerShell is not available on this system.',\n      code: 0,\n      interrupted: false,\n    }\n  }\n\n  let shellCommand: Awaited<ReturnType<typeof exec>>\n  try {\n    shellCommand = await exec(command, abortController.signal, 'powershell', {\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      },\n      preventCwdChanges,\n      // Sandbox works on Linux/macOS/WSL2 — pwsh there is a native binary and\n      // SandboxManager.wrapWithSandbox wraps it same as bash (Shell.ts uses\n      // /bin/sh for the outer spawn to parse the POSIX-quoted bwrap/sandbox-exec\n      // string). On Windows native, sandbox is unsupported; shouldUseSandbox()\n      // returns false via isSandboxingEnabled() → isSupportedPlatform() → false.\n      // The explicit platform check is redundant-but-obvious.\n      shouldUseSandbox:\n        getPlatform() === 'windows'\n          ? false\n          : shouldUseSandbox({ command, dangerouslyDisableSandbox }),\n      shouldAutoBackground,\n    })\n  } catch (e) {\n    logError(e)\n    // Pre-flight failure: spawn/exec rejected before the command ran. Use\n    // code 0 so call() returns stderr gracefully instead of throwing ShellError.\n    return {\n      stdout: '',\n      stderr: `Failed to execute PowerShell command: ${getErrorMessage(e)}`,\n      code: 0,\n      interrupted: false,\n    }\n  }\n\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          throw new Error(\n            'getAppState not available in runPowerShellCommand context',\n          )\n        },\n        setAppState,\n      },\n    )\n    return handle.taskId\n  }\n\n  // Helper to start backgrounding with 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, the generator waits for the current setTimeout to fire\n      // (up to ~1s) before noticing the backgrounding. Matches BashTool.\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  if (shellCommand.onTimeout && shouldAutoBackground) {\n    shellCommand.onTimeout(backgroundFn => {\n      startBackgrounding(\n        'tengu_powershell_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(\n          'tengu_powershell_command_assistant_auto_backgrounded',\n        )\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  if (run_in_background === true && !isBackgroundTasksDisabled) {\n    const shellId = await spawnBackgroundTask()\n\n    logEvent('tengu_powershell_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  // Start polling the output file for progress\n  TaskOutput.startPolling(shellCommand.taskOutput.taskId)\n\n  // Set up progress yielding with periodic checks\n  const startTime = Date.now()\n  let nextProgressTime = startTime + PROGRESS_THRESHOLD_MS\n  let foregroundTaskId: string | undefined = undefined\n\n  // Progress loop: wrap in try/finally so stopPolling is called on every exit\n  // path — normal completion, timeout/interrupt backgrounding, and Ctrl+B\n  // (matches BashTool pattern; see PR #18887 review thread at :560)\n  try {\n    while (true) {\n      const now = Date.now()\n      const timeUntilNextProgress = Math.max(0, nextProgressTime - now)\n\n      const progressSignal = createProgressSignal()\n      const result = await Promise.race([\n        resultPromise,\n        new Promise<null>(resolve =>\n          setTimeout(r => r(null), timeUntilNextProgress, resolve).unref(),\n        ),\n        progressSignal,\n      ])\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          // Command completed — cleanup stream listeners here. The finally\n          // block's guard (!backgroundShellId && status !== 'backgrounded')\n          // correctly skips cleanup for *running* backgrounded tasks, but\n          // in this race the process is done. Matches BashTool.tsx:1399.\n          shellCommand.cleanup()\n          return fixedResult\n        }\n        // Command has completed\n        return result\n      }\n\n      // Check if command was backgrounded (by timeout or interrupt)\n      if (backgroundShellId) {\n        return {\n          stdout: interruptBackgroundingStarted ? fullOutput : '',\n          stderr: '',\n          code: 0,\n          interrupted: false,\n          backgroundTaskId: backgroundShellId,\n          assistantAutoBackgrounded,\n        }\n      }\n\n      // User submitted a new message - background instead of killing\n      if (\n        abortController.signal.aborted &&\n        abortController.signal.reason === 'interrupt' &&\n        !interruptBackgroundingStarted\n      ) {\n        interruptBackgroundingStarted = true\n        if (!isBackgroundTasksDisabled) {\n          startBackgrounding('tengu_powershell_command_interrupt_backgrounded')\n          // Reloop so the backgroundShellId check (above) catches the sync\n          // foregroundTaskId→background path. Without this, we fall through\n          // to the Ctrl+B check below, which matches status==='backgrounded'\n          // and incorrectly returns backgroundedByUser:true. (bugs 020/021)\n          continue\n        }\n        shellCommand.kill()\n      }\n\n      // Check if this foreground task was backgrounded via backgroundAll() (ctrl+b)\n      if (foregroundTaskId) {\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 backgrounding UI hint after threshold\n      if (\n        !isBackgroundTasksDisabled &&\n        backgroundShellId === undefined &&\n        elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 &&\n        setToolJSX\n      ) {\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\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      nextProgressTime = Date.now() + PROGRESS_INTERVAL_MS\n    }\n  } finally {\n    TaskOutput.stopPolling(shellCommand.taskOutput.taskId)\n    // Ensure cleanup runs on every exit path (success, rejection, abort).\n    // Skip when backgrounded — LocalShellTask owns cleanup for those.\n    // Matches main #21105.\n    if (!backgroundShellId && shellCommand.status !== 'backgrounded') {\n      if (foregroundTaskId) {\n        unregisterForeground(foregroundTaskId, setAppState)\n      }\n      shellCommand.cleanup()\n    }\n  }\n}\n"],"mappings":"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","ignoreList":[]}