Filehigh importancesource

PowerShellTool.tsx

tools/PowerShellTool/PowerShellTool.tsx

1001
Lines
144624
Bytes
4
Exports
47
Imports
10
Keywords

What this is

This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.

Beginner explanation

This file is part of the tool layer, which means it describes actions the system can perform for the user or model.

How it is used

Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.

Expert explanation

Architecturally, this file intersects with tool-system, shell-safety. It contains 1001 lines, 47 detected imports, and 4 detected exports.

Important relationships

Detected exports

  • detectBlockedSleepPattern
  • PowerShellToolInput
  • Out
  • PowerShellTool

Keywords

commandresultstdoutinputbackgroundtaskidutilscodestderrshellcommandbashtool

Detected imports

  • bun:bundle
  • @anthropic-ai/sdk/resources/index.mjs
  • fs/promises
  • react
  • src/hooks/useCanUseTool.js
  • src/state/AppState.js
  • zod/v4
  • ../../bootstrap/state.js
  • ../../constants/toolLimits.js
  • ../../services/analytics/index.js
  • ../../Tool.js
  • ../../Tool.js
  • ../../tasks/LocalShellTask/LocalShellTask.js
  • ../../types/ids.js
  • ../../types/message.js
  • ../../utils/claudeCodeHints.js
  • ../../utils/envUtils.js
  • ../../utils/errors.js
  • ../../utils/format.js
  • ../../utils/lazySchema.js
  • ../../utils/log.js
  • ../../utils/permissions/PermissionResult.js
  • ../../utils/platform.js
  • ../../utils/plugins/hintRecommendation.js
  • ../../utils/Shell.js
  • ../../utils/ShellCommand.js
  • ../../utils/sandbox/sandbox-adapter.js
  • ../../utils/semanticBoolean.js
  • ../../utils/semanticNumber.js
  • ../../utils/shell/powershellDetection.js
  • ../../utils/stringUtils.js
  • ../../utils/task/diskOutput.js
  • ../../utils/task/TaskOutput.js
  • ../../utils/terminal.js
  • ../../utils/toolResultStorage.js
  • ../BashTool/shouldUseSandbox.js
  • ../BashTool/UI.js
  • ../BashTool/utils.js
  • ../shared/gitOperationTracking.js
  • ./commandSemantics.js
  • ./powershellPermissions.js
  • ./prompt.js
  • ./readOnlyValidation.js
  • ./toolName.js
  • ./UI.js
  • ../../types/tools.js
  • ../../types/tools.js

Source notes

This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.

Open parent directory

Full source

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":[]}