From f3a984dde153cfb71c8f6045bae0f23f03d1d1f3 Mon Sep 17 00:00:00 2001 From: KRATOS <84986124+gnanam1990@users.noreply.github.com> Date: Fri, 3 Apr 2026 13:44:28 +0530 Subject: [PATCH] fix(security-review): Handle null shell output (#231) Normalize shell command stdout and stderr before the prompt-shell path and shared tool-result mappers use string operations. This prevents /security-review from crashing when a shell tool returns null output fields and adds regression coverage for both direct mapper calls and prompt generation. Fixes #165 Co-authored-by: Claude --- src/tools/BashTool/BashTool.tsx | 14 ++-- src/tools/PowerShellTool/PowerShellTool.tsx | 18 +++-- src/tools/shellToolResultMappers.test.ts | 71 +++++++++++++++++++ src/utils/promptShellExecution.test.ts | 77 +++++++++++++++++++++ src/utils/promptShellExecution.ts | 34 ++++++--- 5 files changed, 191 insertions(+), 23 deletions(-) create mode 100644 src/tools/shellToolResultMappers.test.ts create mode 100644 src/utils/promptShellExecution.test.ts diff --git a/src/tools/BashTool/BashTool.tsx b/src/tools/BashTool/BashTool.tsx index 3fcc8876..f18a7f08 100644 --- a/src/tools/BashTool/BashTool.tsx +++ b/src/tools/BashTool/BashTool.tsx @@ -578,10 +578,12 @@ export const BashTool = buildTool({ const block = buildImageToolResult(stdout, toolUseID); if (block) return block; } - let processedStdout = stdout; - if (stdout) { + const normalizedStdout = typeof stdout === 'string' ? stdout : ''; + const normalizedStderr = typeof stderr === 'string' ? stderr : ''; + let processedStdout = normalizedStdout; + if (normalizedStdout) { // Replace any leading newlines or lines with only whitespace - processedStdout = stdout.replace(/^(\s*\n)+/, ''); + processedStdout = normalizedStdout.replace(/^(\s*\n)+/, ''); // Still trim the end as before processedStdout = processedStdout.trimEnd(); } @@ -598,9 +600,9 @@ export const BashTool = buildTool({ hasMore: preview.hasMore }); } - let errorMessage = stderr.trim(); + let errorMessage = normalizedStderr.trim(); if (interrupted) { - if (stderr) errorMessage += EOL; + if (normalizedStderr) errorMessage += EOL; errorMessage += 'Command was aborted before completion'; } let backgroundInfo = ''; @@ -1141,4 +1143,4 @@ async function* runShellCommand({ TaskOutput.stopPolling(shellCommand.taskOutput.taskId); } } -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","ToolResultBlockParam","copyFile","stat","fsStat","truncate","fsTruncate","link","React","CanUseToolFn","AppState","z","getKairosActive","TOOL_SUMMARY_MAX_LENGTH","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","notifyVscodeFileUpdated","SetToolJSXFn","ToolCallProgress","ToolUseContext","ValidationResult","buildTool","ToolDef","backgroundExistingForegroundTask","markTaskNotified","registerForeground","spawnShellTask","unregisterForeground","AgentId","AssistantMessage","parseForSecurity","splitCommand_DEPRECATED","splitCommandWithOperators","extractClaudeCodeHints","detectCodeIndexingFromCommand","isEnvTruthy","isENOENT","ShellError","detectFileEncoding","detectLineEndings","getFileModificationTime","writeTextContent","fileHistoryEnabled","fileHistoryTrackEdit","getFsImplementation","lazySchema","expandPath","PermissionResult","maybeRecordPluginHint","exec","ExecResult","SandboxManager","semanticBoolean","semanticNumber","EndTruncatingAccumulator","getTaskOutputPath","TaskOutput","isOutputLineTruncated","buildLargeToolResultMessage","ensureToolResultsDir","generatePreview","getToolResultPath","PREVIEW_SIZE_BYTES","userFacingName","fileEditUserFacingName","trackGitOperations","bashToolHasPermission","commandHasAnyCd","matchWildcardPattern","permissionRuleExtractPrefix","interpretCommandResult","getDefaultTimeoutMs","getMaxTimeoutMs","getSimplePrompt","checkReadOnlyConstraints","parseSedEditCommand","shouldUseSandbox","BASH_TOOL_NAME","BackgroundHint","renderToolResultMessage","renderToolUseErrorMessage","renderToolUseMessage","renderToolUseProgressMessage","renderToolUseQueuedMessage","buildImageToolResult","isImageOutput","resetCwdIfOutsideProject","resizeShellImageOutput","stdErrAppendShellResetMessage","stripEmptyLines","EOL","PROGRESS_THRESHOLD_MS","ASSISTANT_BLOCKING_BUDGET_MS","BASH_SEARCH_COMMANDS","Set","BASH_READ_COMMANDS","BASH_LIST_COMMANDS","BASH_SEMANTIC_NEUTRAL_COMMANDS","BASH_SILENT_COMMANDS","isSearchOrReadBashCommand","command","isSearch","isRead","isList","partsWithOperators","length","hasSearch","hasRead","hasList","hasNonNeutralCommand","skipNextAsRedirectTarget","part","baseCommand","trim","split","has","isPartSearch","isPartRead","isPartList","isSilentBashCommand","hasNonFallbackCommand","lastOperator","DISALLOWED_AUTO_BACKGROUND_COMMANDS","isBackgroundTasksDisabled","process","env","CLAUDE_CODE_DISABLE_BACKGROUND_TASKS","fullInputSchema","strictObject","string","describe","timeout","number","optional","description","run_in_background","boolean","dangerouslyDisableSandbox","_simulatedSedEdit","object","filePath","newContent","inputSchema","omit","InputSchema","ReturnType","BashToolInput","infer","COMMON_BACKGROUND_COMMANDS","const","getCommandTypeForLogging","parts","includes","outputSchema","stdout","stderr","rawOutputPath","interrupted","isImage","backgroundTaskId","backgroundedByUser","assistantAutoBackgrounded","returnCodeInterpretation","noOutputExpected","structuredContent","array","any","persistedOutputPath","persistedOutputSize","OutputSchema","Out","BashProgress","isAutobackgroundingAllowed","detectBlockedSleepPattern","first","m","secs","parseInt","rest","slice","join","SimulatedSedEditResult","data","SimulatedSedEditContext","Pick","applySedEdit","simulatedEdit","toolUseContext","parentMessage","Promise","absoluteFilePath","fs","encoding","originalContent","readFile","e","updateFileHistoryState","uuid","endings","readFileState","set","content","timestamp","offset","undefined","limit","BashTool","name","searchHint","maxResultSizeChars","strict","prompt","isConcurrencySafe","input","isReadOnly","compoundCommandHasCd","result","behavior","toAutoClassifierInput","preparePermissionMatcher","parsed","kind","subcommands","commands","map","c","argv","pattern","prefix","some","cmd","startsWith","isSearchOrReadCommand","safeParse","success","sedInfo","file_path","old_string","CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR","getToolUseSummary","getActivityDescription","desc","validateInput","sleepPattern","message","errorCode","checkPermissions","context","extractSearchText","mapToolResultToToolResultBlockParam","toolUseID","tool_use_id","type","block","processedStdout","replace","trimEnd","preview","filepath","originalSize","isJson","hasMore","errorMessage","backgroundInfo","outputPath","filter","Boolean","is_error","call","_canUseTool","onProgress","abortController","getAppState","setAppState","setToolJSX","stdoutAccumulator","stderrForShellReset","interpretationResult","progressCounter","wasInterrupted","isMainThread","agentId","preventCwdChanges","commandGenerator","runShellCommand","setAppStateForTasks","toolUseId","generatorResult","next","done","progress","value","output","fullOutput","elapsedTimeSeconds","totalLines","totalBytes","taskId","timeoutMs","code","isInterrupt","signal","reason","append","isError","appState","toolPermissionContext","outputWithSbFailures","annotateStderrWithSandboxFailures","preSpawnError","Error","toString","MAX_PERSISTED_SIZE","outputFilePath","outputTaskId","fileStat","size","dest","commandType","command_type","stdout_length","stderr_length","exit_code","codeIndexingTool","tool","source","strippedStdout","extracted","stripped","hints","hint","compressedStdout","resized","isResultTruncated","AbortController","f","prev","AsyncGenerator","lastProgressOutput","lastTotalLines","lastTotalBytes","backgroundShellId","resolveProgress","createProgressSignal","resolve","shouldAutoBackground","shellCommand","lastLines","allLines","isIncomplete","resultPromise","spawnBackgroundTask","handle","startBackgrounding","eventName","backgroundFn","shellId","foregroundTaskId","then","onTimeout","setTimeout","status","unref","startTime","Date","now","initialResult","race","t","r","v","cleanup","startPolling","taskOutput","progressSignal","fixedResult","stdoutToFile","outputFileRedundant","path","outputFileSize","elapsed","elapsedSeconds","Math","floor","jsx","shouldHidePromptInput","shouldContinueAnimation","showSpinner","stopPolling"],"sources":["BashTool.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'\nimport {\n  copyFile,\n  stat as fsStat,\n  truncate as fsTruncate,\n  link,\n} from 'fs/promises'\nimport * as React from 'react'\nimport type { CanUseToolFn } from 'src/hooks/useCanUseTool.js'\nimport type { AppState } from 'src/state/AppState.js'\nimport { z } from 'zod/v4'\nimport { getKairosActive } from '../../bootstrap/state.js'\nimport { TOOL_SUMMARY_MAX_LENGTH } from '../../constants/toolLimits.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from '../../services/analytics/index.js'\nimport { notifyVscodeFileUpdated } from '../../services/mcp/vscodeSdkMcp.js'\nimport type {\n  SetToolJSXFn,\n  ToolCallProgress,\n  ToolUseContext,\n  ValidationResult,\n} from '../../Tool.js'\nimport { buildTool, type ToolDef } from '../../Tool.js'\nimport {\n  backgroundExistingForegroundTask,\n  markTaskNotified,\n  registerForeground,\n  spawnShellTask,\n  unregisterForeground,\n} from '../../tasks/LocalShellTask/LocalShellTask.js'\nimport type { AgentId } from '../../types/ids.js'\nimport type { AssistantMessage } from '../../types/message.js'\nimport { parseForSecurity } from '../../utils/bash/ast.js'\nimport {\n  splitCommand_DEPRECATED,\n  splitCommandWithOperators,\n} from '../../utils/bash/commands.js'\nimport { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js'\nimport { detectCodeIndexingFromCommand } from '../../utils/codeIndexing.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isENOENT, ShellError } from '../../utils/errors.js'\nimport {\n  detectFileEncoding,\n  detectLineEndings,\n  getFileModificationTime,\n  writeTextContent,\n} from '../../utils/file.js'\nimport {\n  fileHistoryEnabled,\n  fileHistoryTrackEdit,\n} from '../../utils/fileHistory.js'\nimport { truncate } from '../../utils/format.js'\nimport { getFsImplementation } from '../../utils/fsOperations.js'\nimport { lazySchema } from '../../utils/lazySchema.js'\nimport { expandPath } from '../../utils/path.js'\nimport type { PermissionResult } from '../../utils/permissions/PermissionResult.js'\nimport { maybeRecordPluginHint } from '../../utils/plugins/hintRecommendation.js'\nimport { exec } from '../../utils/Shell.js'\nimport type { ExecResult } from '../../utils/ShellCommand.js'\nimport { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'\nimport { semanticBoolean } from '../../utils/semanticBoolean.js'\nimport { semanticNumber } from '../../utils/semanticNumber.js'\nimport { EndTruncatingAccumulator } from '../../utils/stringUtils.js'\nimport { getTaskOutputPath } from '../../utils/task/diskOutput.js'\nimport { TaskOutput } from '../../utils/task/TaskOutput.js'\nimport { isOutputLineTruncated } from '../../utils/terminal.js'\nimport {\n  buildLargeToolResultMessage,\n  ensureToolResultsDir,\n  generatePreview,\n  getToolResultPath,\n  PREVIEW_SIZE_BYTES,\n} from '../../utils/toolResultStorage.js'\nimport { userFacingName as fileEditUserFacingName } from '../FileEditTool/UI.js'\nimport { trackGitOperations } from '../shared/gitOperationTracking.js'\nimport {\n  bashToolHasPermission,\n  commandHasAnyCd,\n  matchWildcardPattern,\n  permissionRuleExtractPrefix,\n} from './bashPermissions.js'\nimport { interpretCommandResult } from './commandSemantics.js'\nimport {\n  getDefaultTimeoutMs,\n  getMaxTimeoutMs,\n  getSimplePrompt,\n} from './prompt.js'\nimport { checkReadOnlyConstraints } from './readOnlyValidation.js'\nimport { parseSedEditCommand } from './sedEditParser.js'\nimport { shouldUseSandbox } from './shouldUseSandbox.js'\nimport { BASH_TOOL_NAME } from './toolName.js'\nimport {\n  BackgroundHint,\n  renderToolResultMessage,\n  renderToolUseErrorMessage,\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n} from './UI.js'\nimport {\n  buildImageToolResult,\n  isImageOutput,\n  resetCwdIfOutsideProject,\n  resizeShellImageOutput,\n  stdErrAppendShellResetMessage,\n  stripEmptyLines,\n} from './utils.js'\n\nconst EOL = '\\n'\n\n// Progress display constants\nconst PROGRESS_THRESHOLD_MS = 2000 // Show progress after 2 seconds\n// In assistant mode, blocking bash auto-backgrounds after this many ms in the main agent\nconst ASSISTANT_BLOCKING_BUDGET_MS = 15_000\n\n// Search commands for collapsible display (grep, find, etc.)\nconst BASH_SEARCH_COMMANDS = new Set([\n  'find',\n  'grep',\n  'rg',\n  'ag',\n  'ack',\n  'locate',\n  'which',\n  'whereis',\n])\n\n// Read/view commands for collapsible display (cat, head, etc.)\nconst BASH_READ_COMMANDS = new Set([\n  'cat',\n  'head',\n  'tail',\n  'less',\n  'more',\n  // Analysis commands\n  'wc',\n  'stat',\n  'file',\n  'strings',\n  // Data processing — commonly used to parse/transform file content in pipes\n  'jq',\n  'awk',\n  'cut',\n  'sort',\n  'uniq',\n  'tr',\n])\n\n// Directory-listing commands for collapsible display (ls, tree, du).\n// Split from BASH_READ_COMMANDS so the summary says \"Listed N directories\"\n// instead of the misleading \"Read N files\".\nconst BASH_LIST_COMMANDS = new Set(['ls', 'tree', 'du'])\n\n// Commands that are semantic-neutral in any position — pure output/status commands\n// that don't change the read/search nature of the overall pipeline.\n// e.g. `ls dir && echo \"---\" && ls dir2` is still a read-only compound command.\nconst BASH_SEMANTIC_NEUTRAL_COMMANDS = new Set([\n  'echo',\n  'printf',\n  'true',\n  'false',\n  ':', // bash no-op\n])\n\n// Commands that typically produce no stdout on success\nconst BASH_SILENT_COMMANDS = new Set([\n  'mv',\n  'cp',\n  'rm',\n  'mkdir',\n  'rmdir',\n  'chmod',\n  'chown',\n  'chgrp',\n  'touch',\n  'ln',\n  'cd',\n  'export',\n  'unset',\n  'wait',\n])\n\n/**\n * Checks if a bash command is a search or read operation.\n * Used to determine if the command should be collapsed in the UI.\n * Returns an object indicating whether it's a search or read operation.\n *\n * For pipelines (e.g., `cat file | bq`), ALL parts must be search/read commands\n * for the whole command to be considered collapsible.\n *\n * Semantic-neutral commands (echo, printf, true, false, :) are skipped in any\n * position, as they're pure output/status commands that don't affect the read/search\n * nature of the pipeline (e.g. `ls dir && echo \"---\" && ls dir2` is still a read).\n */\nexport function isSearchOrReadBashCommand(command: string): {\n  isSearch: boolean\n  isRead: boolean\n  isList: boolean\n} {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    // If we can't parse the command due to malformed syntax,\n    // it's not a search/read command\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  if (partsWithOperators.length === 0) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  let hasSearch = false\n  let hasRead = false\n  let hasList = false\n  let hasNonNeutralCommand = false\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {\n      continue\n    }\n\n    hasNonNeutralCommand = true\n\n    const isPartSearch = BASH_SEARCH_COMMANDS.has(baseCommand)\n    const isPartRead = BASH_READ_COMMANDS.has(baseCommand)\n    const isPartList = BASH_LIST_COMMANDS.has(baseCommand)\n\n    if (!isPartSearch && !isPartRead && !isPartList) {\n      return { isSearch: false, isRead: false, isList: false }\n    }\n\n    if (isPartSearch) hasSearch = true\n    if (isPartRead) hasRead = true\n    if (isPartList) hasList = true\n  }\n\n  // Only neutral commands (e.g., just \"echo foo\") -- not collapsible\n  if (!hasNonNeutralCommand) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  return { isSearch: hasSearch, isRead: hasRead, isList: hasList }\n}\n\n/**\n * Checks if a bash command is expected to produce no stdout on success.\n * Used to show \"Done\" instead of \"(No output)\" in the UI.\n */\nfunction isSilentBashCommand(command: string): boolean {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    return false\n  }\n\n  if (partsWithOperators.length === 0) {\n    return false\n  }\n\n  let hasNonFallbackCommand = false\n  let lastOperator: string | null = null\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      lastOperator = part\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (\n      lastOperator === '||' &&\n      BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)\n    ) {\n      continue\n    }\n\n    hasNonFallbackCommand = true\n\n    if (!BASH_SILENT_COMMANDS.has(baseCommand)) {\n      return false\n    }\n  }\n\n  return hasNonFallbackCommand\n}\n\n// Commands that should not be auto-backgrounded\nconst DISALLOWED_AUTO_BACKGROUND_COMMANDS = [\n  'sleep', // Sleep should run in foreground unless explicitly backgrounded by user\n]\n\n// Check if background tasks are disabled at module load time\nconst isBackgroundTasksDisabled =\n  // eslint-disable-next-line custom-rules/no-process-env-top-level -- Intentional: schema must be defined at module load\n  isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_BACKGROUND_TASKS)\n\nconst fullInputSchema = lazySchema(() =>\n  z.strictObject({\n    command: z.string().describe('The command to execute'),\n    timeout: semanticNumber(z.number().optional()).describe(\n      `Optional timeout in milliseconds (max ${getMaxTimeoutMs()})`,\n    ),\n    description: z\n      .string()\n      .optional()\n      .describe(`Clear, concise description of what this command does in active voice. Never use words like \"complex\" or \"risk\" in the description - just describe what it does.\n\nFor simple commands (git, npm, standard CLI tools), keep it brief (5-10 words):\n- ls → \"List files in current directory\"\n- git status → \"Show working tree status\"\n- npm install → \"Install package dependencies\"\n\nFor commands that are harder to parse at a glance (piped commands, obscure flags, etc.), add enough context to clarify what it does:\n- find . -name \"*.tmp\" -exec rm {} \\\\; → \"Find and delete all .tmp files recursively\"\n- git reset --hard origin/main → \"Discard all local changes and match remote main\"\n- curl -s url | jq '.data[]' → \"Fetch JSON from URL and extract data array elements\"`),\n    run_in_background: semanticBoolean(z.boolean().optional()).describe(\n      `Set to true to run this command in the background. Use Read to read the output later.`,\n    ),\n    dangerouslyDisableSandbox: semanticBoolean(z.boolean().optional()).describe(\n      'Set this to true to dangerously override sandbox mode and run commands without sandboxing.',\n    ),\n    _simulatedSedEdit: z\n      .object({\n        filePath: z.string(),\n        newContent: z.string(),\n      })\n      .optional()\n      .describe('Internal: pre-computed sed edit result from preview'),\n  }),\n)\n\n// Always omit _simulatedSedEdit from the model-facing schema. It is an internal-only\n// field set by SedEditPermissionRequest after the user approves a sed edit preview.\n// Exposing it in the schema would let the model bypass permission checks and the\n// sandbox by pairing an innocuous command with an arbitrary file write.\n// Also conditionally remove run_in_background when background tasks are disabled.\nconst inputSchema = lazySchema(() =>\n  isBackgroundTasksDisabled\n    ? fullInputSchema().omit({\n        run_in_background: true,\n        _simulatedSedEdit: true,\n      })\n    : fullInputSchema().omit({ _simulatedSedEdit: true }),\n)\ntype InputSchema = ReturnType<typeof inputSchema>\n\n// Use fullInputSchema for the type to always include run_in_background\n// (even when it's omitted from the schema, the code needs to handle it)\nexport type BashToolInput = z.infer<ReturnType<typeof fullInputSchema>>\n\nconst COMMON_BACKGROUND_COMMANDS = [\n  'npm',\n  'yarn',\n  'pnpm',\n  'node',\n  'python',\n  'python3',\n  'go',\n  'cargo',\n  'make',\n  'docker',\n  'terraform',\n  'webpack',\n  'vite',\n  'jest',\n  'pytest',\n  'curl',\n  'wget',\n  'build',\n  'test',\n  'serve',\n  'watch',\n  'dev',\n] as const\n\nfunction getCommandTypeForLogging(\n  command: string,\n): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0)\n    return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n\n  // Check each part of the command to see if any match common background commands\n  for (const part of parts) {\n    const baseCommand = part.split(' ')[0] || ''\n    if (\n      COMMON_BACKGROUND_COMMANDS.includes(\n        baseCommand as (typeof COMMON_BACKGROUND_COMMANDS)[number],\n      )\n    ) {\n      return baseCommand as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    }\n  }\n\n  return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n}\n\nconst outputSchema = lazySchema(() =>\n  z.object({\n    stdout: z.string().describe('The standard output of the command'),\n    stderr: z.string().describe('The standard error output of the command'),\n    rawOutputPath: z\n      .string()\n      .optional()\n      .describe('Path to raw output file for large MCP tool outputs'),\n    interrupted: z.boolean().describe('Whether the command was interrupted'),\n    isImage: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if stdout contains image data'),\n    backgroundTaskId: z\n      .string()\n      .optional()\n      .describe(\n        'ID of the background task if command is running in background',\n      ),\n    backgroundedByUser: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if the user manually backgrounded the command with Ctrl+B',\n      ),\n    assistantAutoBackgrounded: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if assistant-mode auto-backgrounded a long-running blocking command',\n      ),\n    dangerouslyDisableSandbox: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if sandbox mode was overridden'),\n    returnCodeInterpretation: z\n      .string()\n      .optional()\n      .describe(\n        'Semantic interpretation for non-error exit codes with special meaning',\n      ),\n    noOutputExpected: z\n      .boolean()\n      .optional()\n      .describe(\n        'Whether the command is expected to produce no output on success',\n      ),\n    structuredContent: z\n      .array(z.any())\n      .optional()\n      .describe('Structured content blocks'),\n    persistedOutputPath: z\n      .string()\n      .optional()\n      .describe(\n        'Path to the persisted full output in tool-results dir (set when output is too large for inline)',\n      ),\n    persistedOutputSize: z\n      .number()\n      .optional()\n      .describe(\n        'Total size of the output in bytes (set when output is too large for inline)',\n      ),\n  }),\n)\n\ntype OutputSchema = ReturnType<typeof outputSchema>\nexport type Out = z.infer<OutputSchema>\n\n// Re-export BashProgress from centralized types to break import cycles\nexport type { BashProgress } from '../../types/tools.js'\n\nimport type { BashProgress } from '../../types/tools.js'\n\n/**\n * Checks if a command is allowed to be automatically backgrounded\n * @param command The command to check\n * @returns false for commands that should not be auto-backgrounded (like sleep)\n */\nfunction isAutobackgroundingAllowed(command: string): boolean {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return true\n\n  // Get the first part which should be the base command\n  const baseCommand = parts[0]?.trim()\n  if (!baseCommand) return true\n\n  return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(baseCommand)\n}\n\n/**\n * Detect standalone or leading `sleep N` patterns that should use Monitor\n * instead. Catches `sleep 5`, `sleep 5 && check`, `sleep 5; check` — but\n * not sleep inside pipelines, subshells, or scripts (those are fine).\n */\nexport function detectBlockedSleepPattern(command: string): string | null {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return null\n\n  const first = parts[0]?.trim() ?? ''\n  // Bare `sleep N` or `sleep N.N` as the first subcommand.\n  // Float durations (sleep 0.5) are allowed — those are legit pacing, not polls.\n  const m = /^sleep\\s+(\\d+)\\s*$/.exec(first)\n  if (!m) return null\n  const secs = parseInt(m[1]!, 10)\n  if (secs < 2) return null // sub-2s sleeps are fine (rate limiting, pacing)\n\n  // `sleep N` alone → \"what are you waiting for?\"\n  // `sleep N && check` → \"use Monitor { command: check }\"\n  const rest = parts.slice(1).join(' ').trim()\n  return rest\n    ? `sleep ${secs} followed by: ${rest}`\n    : `standalone sleep ${secs}`\n}\n\n/**\n * Checks if a command contains tools that shouldn't run in sandbox\n * This includes:\n * - Dynamic config-based disabled commands and substrings (tengu_sandbox_disabled_commands)\n * - User-configured commands from settings.json (sandbox.excludedCommands)\n *\n * User-configured commands support the same pattern syntax as permission rules:\n * - Exact matches: \"npm run lint\"\n * - Prefix patterns: \"npm run test:*\"\n */\n\ntype SimulatedSedEditResult = {\n  data: Out\n}\n\ntype SimulatedSedEditContext = Pick<\n  ToolUseContext,\n  'readFileState' | 'updateFileHistoryState'\n>\n\n/**\n * Applies a simulated sed edit directly instead of running sed.\n * This is used by the permission dialog to ensure what the user previews\n * is exactly what gets written to the file.\n */\nasync function applySedEdit(\n  simulatedEdit: { filePath: string; newContent: string },\n  toolUseContext: SimulatedSedEditContext,\n  parentMessage?: AssistantMessage,\n): Promise<SimulatedSedEditResult> {\n  const { filePath, newContent } = simulatedEdit\n  const absoluteFilePath = expandPath(filePath)\n  const fs = getFsImplementation()\n\n  // Read original content for VS Code notification\n  const encoding = detectFileEncoding(absoluteFilePath)\n  let originalContent: string\n  try {\n    originalContent = await fs.readFile(absoluteFilePath, { encoding })\n  } catch (e) {\n    if (isENOENT(e)) {\n      return {\n        data: {\n          stdout: '',\n          stderr: `sed: ${filePath}: No such file or directory\\nExit code 1`,\n          interrupted: false,\n        },\n      }\n    }\n    throw e\n  }\n\n  // Track file history before making changes (for undo support)\n  if (fileHistoryEnabled() && parentMessage) {\n    await fileHistoryTrackEdit(\n      toolUseContext.updateFileHistoryState,\n      absoluteFilePath,\n      parentMessage.uuid,\n    )\n  }\n\n  // Detect line endings and write new content\n  const endings = detectLineEndings(absoluteFilePath)\n  writeTextContent(absoluteFilePath, newContent, encoding, endings)\n\n  // Notify VS Code about the file change\n  notifyVscodeFileUpdated(absoluteFilePath, originalContent, newContent)\n\n  // Update read timestamp to invalidate stale writes\n  toolUseContext.readFileState.set(absoluteFilePath, {\n    content: newContent,\n    timestamp: getFileModificationTime(absoluteFilePath),\n    offset: undefined,\n    limit: undefined,\n  })\n\n  // Return success result matching sed output format (sed produces no output on success)\n  return {\n    data: {\n      stdout: '',\n      stderr: '',\n      interrupted: false,\n    },\n  }\n}\n\nexport const BashTool = buildTool({\n  name: BASH_TOOL_NAME,\n  searchHint: 'execute shell commands',\n  // 30K chars - tool result persistence threshold\n  maxResultSizeChars: 30_000,\n  strict: true,\n  async description({ description }) {\n    return description || 'Run shell command'\n  },\n  async prompt() {\n    return getSimplePrompt()\n  },\n  isConcurrencySafe(input) {\n    return this.isReadOnly?.(input) ?? false\n  },\n  isReadOnly(input) {\n    const compoundCommandHasCd = commandHasAnyCd(input.command)\n    const result = checkReadOnlyConstraints(input, compoundCommandHasCd)\n    return result.behavior === 'allow'\n  },\n  toAutoClassifierInput(input) {\n    return input.command\n  },\n  async preparePermissionMatcher({ command }) {\n    // Hook `if` filtering is \"no match → skip hook\" (deny-like semantics), so\n    // compound commands must fire the hook if ANY subcommand matches. Without\n    // splitting, `ls && git push` would bypass a `Bash(git *)` security hook.\n    const parsed = await parseForSecurity(command)\n    if (parsed.kind !== 'simple') {\n      // parse-unavailable / too-complex: fail safe by running the hook.\n      return () => true\n    }\n    // Match on argv (strips leading VAR=val) so `FOO=bar git push` still\n    // matches `Bash(git *)`.\n    const subcommands = parsed.commands.map(c => c.argv.join(' '))\n    return pattern => {\n      const prefix = permissionRuleExtractPrefix(pattern)\n      return subcommands.some(cmd => {\n        if (prefix !== null) {\n          return cmd === prefix || cmd.startsWith(`${prefix} `)\n        }\n        return matchWildcardPattern(pattern, cmd)\n      })\n    }\n  },\n  isSearchOrReadCommand(input) {\n    const parsed = inputSchema().safeParse(input)\n    if (!parsed.success)\n      return { isSearch: false, isRead: false, isList: false }\n    return isSearchOrReadBashCommand(parsed.data.command)\n  },\n  get inputSchema(): InputSchema {\n    return inputSchema()\n  },\n  get outputSchema(): OutputSchema {\n    return outputSchema()\n  },\n  userFacingName(input) {\n    if (!input) {\n      return 'Bash'\n    }\n    // Render sed in-place edits as file edits\n    if (input.command) {\n      const sedInfo = parseSedEditCommand(input.command)\n      if (sedInfo) {\n        return fileEditUserFacingName({\n          file_path: sedInfo.filePath,\n          old_string: 'x',\n        })\n      }\n    }\n    // Env var FIRST: shouldUseSandbox → splitCommand_DEPRECATED → shell-quote's\n    // `new RegExp` per call. userFacingName runs per-render for every bash\n    // message in history; with ~50 msgs + one slow-to-tokenize command, this\n    // exceeds the shimmer tick → transition abort → infinite retry (#21605).\n    return isEnvTruthy(process.env.CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR) &&\n      shouldUseSandbox(input)\n      ? 'SandboxedBash'\n      : 'Bash'\n  },\n  getToolUseSummary(input) {\n    if (!input?.command) {\n      return null\n    }\n    const { command, description } = input\n    if (description) {\n      return description\n    }\n    return truncate(command, TOOL_SUMMARY_MAX_LENGTH)\n  },\n  getActivityDescription(input) {\n    if (!input?.command) {\n      return 'Running command'\n    }\n    const desc =\n      input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH)\n    return `Running ${desc}`\n  },\n  async validateInput(input: BashToolInput): Promise<ValidationResult> {\n    if (\n      feature('MONITOR_TOOL') &&\n      !isBackgroundTasksDisabled &&\n      !input.run_in_background\n    ) {\n      const sleepPattern = detectBlockedSleepPattern(input.command)\n      if (sleepPattern !== null) {\n        return {\n          result: false,\n          message: `Blocked: ${sleepPattern}. Run blocking commands in the background with run_in_background: true — you'll get a completion notification when done. For streaming events (watching logs, polling APIs), use the Monitor tool. If you genuinely need a delay (rate limiting, deliberate pacing), keep it under 2 seconds.`,\n          errorCode: 10,\n        }\n      }\n    }\n    return { result: true }\n  },\n  async checkPermissions(input, context): Promise<PermissionResult> {\n    return bashToolHasPermission(input, context)\n  },\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n  renderToolResultMessage,\n  // BashToolResultMessage shows <OutputLine content={stdout}> + stderr.\n  // UI never shows persistedOutputPath wrapper, backgroundInfo — those are\n  // model-facing (mapToolResult... below).\n  extractSearchText({ stdout, stderr }) {\n    return stderr ? `${stdout}\\n${stderr}` : stdout\n  },\n  mapToolResultToToolResultBlockParam(\n    {\n      interrupted,\n      stdout,\n      stderr,\n      isImage,\n      backgroundTaskId,\n      backgroundedByUser,\n      assistantAutoBackgrounded,\n      structuredContent,\n      persistedOutputPath,\n      persistedOutputSize,\n    },\n    toolUseID,\n  ): ToolResultBlockParam {\n    // Handle structured content\n    if (structuredContent && structuredContent.length > 0) {\n      return {\n        tool_use_id: toolUseID,\n        type: 'tool_result',\n        content: structuredContent,\n      }\n    }\n\n    // For image data, format as image content block for Claude\n    if (isImage) {\n      const block = buildImageToolResult(stdout, toolUseID)\n      if (block) return block\n    }\n\n    let processedStdout = stdout\n    if (stdout) {\n      // Replace any leading newlines or lines with only whitespace\n      processedStdout = stdout.replace(/^(\\s*\\n)+/, '')\n      // Still trim the end as before\n      processedStdout = processedStdout.trimEnd()\n    }\n\n    // For large output that was persisted to disk, build <persisted-output>\n    // message for the model. The UI never sees this — it uses data.stdout.\n    if (persistedOutputPath) {\n      const preview = generatePreview(processedStdout, PREVIEW_SIZE_BYTES)\n      processedStdout = buildLargeToolResultMessage({\n        filepath: persistedOutputPath,\n        originalSize: persistedOutputSize ?? 0,\n        isJson: false,\n        preview: preview.preview,\n        hasMore: preview.hasMore,\n      })\n    }\n\n    let errorMessage = stderr.trim()\n    if (interrupted) {\n      if (stderr) errorMessage += EOL\n      errorMessage += '<error>Command was aborted before completion</error>'\n    }\n\n    let backgroundInfo = ''\n    if (backgroundTaskId) {\n      const outputPath = getTaskOutputPath(backgroundTaskId)\n      if (assistantAutoBackgrounded) {\n        backgroundInfo = `Command exceeded the assistant-mode blocking budget (${ASSISTANT_BLOCKING_BUDGET_MS / 1000}s) and was moved to the background with ID: ${backgroundTaskId}. It is still running — you will be notified when it completes. Output is being written to: ${outputPath}. In assistant mode, delegate long-running work to a subagent or use run_in_background to keep this conversation responsive.`\n      } else if (backgroundedByUser) {\n        backgroundInfo = `Command was manually backgrounded by user with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      } else {\n        backgroundInfo = `Command running in background with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      }\n    }\n\n    return {\n      tool_use_id: toolUseID,\n      type: 'tool_result',\n      content: [processedStdout, errorMessage, backgroundInfo]\n        .filter(Boolean)\n        .join('\\n'),\n      is_error: interrupted,\n    }\n  },\n  async call(\n    input: BashToolInput,\n    toolUseContext,\n    _canUseTool?: CanUseToolFn,\n    parentMessage?: AssistantMessage,\n    onProgress?: ToolCallProgress<BashProgress>,\n  ) {\n    // Handle simulated sed edit - apply directly instead of running sed\n    // This ensures what the user previewed is exactly what gets written\n    if (input._simulatedSedEdit) {\n      return applySedEdit(\n        input._simulatedSedEdit,\n        toolUseContext,\n        parentMessage,\n      )\n    }\n\n    const { abortController, getAppState, setAppState, setToolJSX } =\n      toolUseContext\n\n    const stdoutAccumulator = new EndTruncatingAccumulator()\n    let stderrForShellReset = ''\n    let interpretationResult:\n      | ReturnType<typeof interpretCommandResult>\n      | undefined\n\n    let progressCounter = 0\n    let wasInterrupted = false\n    let result: ExecResult\n\n    const isMainThread = !toolUseContext.agentId\n    const preventCwdChanges = !isMainThread\n\n    try {\n      // Use the new async generator version of runShellCommand\n      const commandGenerator = runShellCommand({\n        input,\n        abortController,\n        // Use the always-shared task channel so async agents' background\n        // bash tasks are actually registered (and killable on agent exit).\n        setAppState: toolUseContext.setAppStateForTasks ?? setAppState,\n        setToolJSX,\n        preventCwdChanges,\n        isMainThread,\n        toolUseId: toolUseContext.toolUseId,\n        agentId: toolUseContext.agentId,\n      })\n\n      // Consume the generator and capture the return value\n      let generatorResult\n      do {\n        generatorResult = await commandGenerator.next()\n        if (!generatorResult.done && onProgress) {\n          const progress = generatorResult.value\n          onProgress({\n            toolUseID: `bash-progress-${progressCounter++}`,\n            data: {\n              type: 'bash_progress',\n              output: progress.output,\n              fullOutput: progress.fullOutput,\n              elapsedTimeSeconds: progress.elapsedTimeSeconds,\n              totalLines: progress.totalLines,\n              totalBytes: progress.totalBytes,\n              taskId: progress.taskId,\n              timeoutMs: progress.timeoutMs,\n            },\n          })\n        }\n      } while (!generatorResult.done)\n\n      // Get the final result from the generator's return value\n      result = generatorResult.value\n\n      trackGitOperations(input.command, result.code, result.stdout)\n\n      const isInterrupt =\n        result.interrupted && abortController.signal.reason === 'interrupt'\n\n      // stderr is interleaved in stdout (merged fd) — result.stdout has both\n      stdoutAccumulator.append((result.stdout || '').trimEnd() + EOL)\n\n      // Interpret the command result using semantic rules\n      interpretationResult = interpretCommandResult(\n        input.command,\n        result.code,\n        result.stdout || '',\n        '',\n      )\n\n      // Check for git index.lock error (stderr is in stdout now)\n      if (\n        result.stdout &&\n        result.stdout.includes(\".git/index.lock': File exists\")\n      ) {\n        logEvent('tengu_git_index_lock_error', {})\n      }\n\n      if (interpretationResult.isError && !isInterrupt) {\n        // Only add exit code if it's actually an error\n        if (result.code !== 0) {\n          stdoutAccumulator.append(`Exit code ${result.code}`)\n        }\n      }\n\n      if (!preventCwdChanges) {\n        const appState = getAppState()\n        if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {\n          stderrForShellReset = stdErrAppendShellResetMessage('')\n        }\n      }\n\n      // Annotate output with sandbox violations if any (stderr is in stdout)\n      const outputWithSbFailures =\n        SandboxManager.annotateStderrWithSandboxFailures(\n          input.command,\n          result.stdout || '',\n        )\n\n      if (result.preSpawnError) {\n        throw new Error(result.preSpawnError)\n      }\n      if (interpretationResult.isError && !isInterrupt) {\n        // stderr is merged into stdout (merged fd); outputWithSbFailures\n        // already has the full output. Pass '' for stdout to avoid\n        // duplication in getErrorParts() and processBashCommand.\n        throw new ShellError(\n          '',\n          outputWithSbFailures,\n          result.code,\n          result.interrupted,\n        )\n      }\n      wasInterrupted = result.interrupted\n    } finally {\n      if (setToolJSX) setToolJSX(null)\n    }\n\n    // Get final string from accumulator\n    const stdout = stdoutAccumulator.toString()\n\n    // Large output: the file on disk has more than getMaxOutputLength() bytes.\n    // stdout already contains the first chunk (from getStdout()). Copy the\n    // output file to the tool-results dir so the model can read it via\n    // FileRead. If > 64 MB, truncate after copying.\n    const MAX_PERSISTED_SIZE = 64 * 1024 * 1024\n    let persistedOutputPath: string | undefined\n    let persistedOutputSize: number | undefined\n    if (result.outputFilePath && result.outputTaskId) {\n      try {\n        const fileStat = await fsStat(result.outputFilePath)\n        persistedOutputSize = fileStat.size\n\n        await ensureToolResultsDir()\n        const dest = getToolResultPath(result.outputTaskId, false)\n        if (fileStat.size > MAX_PERSISTED_SIZE) {\n          await fsTruncate(result.outputFilePath, MAX_PERSISTED_SIZE)\n        }\n        try {\n          await link(result.outputFilePath, dest)\n        } catch {\n          await copyFile(result.outputFilePath, dest)\n        }\n        persistedOutputPath = dest\n      } catch {\n        // File may already be gone — stdout preview is sufficient\n      }\n    }\n\n    const commandType = input.command.split(' ')[0]\n\n    logEvent('tengu_bash_tool_command_executed', {\n      command_type:\n        commandType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      stdout_length: stdout.length,\n      stderr_length: 0,\n      exit_code: result.code,\n      interrupted: wasInterrupted,\n    })\n\n    // Log code indexing tool usage\n    const codeIndexingTool = detectCodeIndexingFromCommand(input.command)\n    if (codeIndexingTool) {\n      logEvent('tengu_code_indexing_tool_used', {\n        tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        source:\n          'cli' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        success: result.code === 0,\n      })\n    }\n\n    let strippedStdout = stripEmptyLines(stdout)\n\n    // Claude Code hints protocol: CLIs/SDKs gated on CLAUDECODE=1 emit a\n    // `<claude-code-hint />` tag to stderr (merged into stdout here). Scan,\n    // record for useClaudeCodeHintRecommendation to surface, then strip\n    // so the model never sees the tag — a zero-token side channel.\n    // Stripping runs unconditionally (subagent output must stay clean too);\n    // only the dialog recording is main-thread-only.\n    const extracted = extractClaudeCodeHints(strippedStdout, input.command)\n    strippedStdout = extracted.stripped\n    if (isMainThread && extracted.hints.length > 0) {\n      for (const hint of extracted.hints) maybeRecordPluginHint(hint)\n    }\n\n    let isImage = isImageOutput(strippedStdout)\n\n    // Cap image dimensions + size if present (CC-304 — see\n    // resizeShellImageOutput). Scope the decoded buffer so it can be reclaimed\n    // before we build the output Out object.\n    let compressedStdout = strippedStdout\n    if (isImage) {\n      const resized = await resizeShellImageOutput(\n        strippedStdout,\n        result.outputFilePath,\n        persistedOutputSize,\n      )\n      if (resized) {\n        compressedStdout = resized\n      } else {\n        // Parse failed or file too large (e.g. exceeds MAX_IMAGE_FILE_SIZE).\n        // Keep isImage in sync with what we actually send so the UI label stays\n        // accurate — mapToolResultToToolResultBlockParam's defensive\n        // fallthrough will send text, not an image block.\n        isImage = false\n      }\n    }\n\n    const data: Out = {\n      stdout: compressedStdout,\n      stderr: stderrForShellReset,\n      interrupted: wasInterrupted,\n      isImage,\n      returnCodeInterpretation: interpretationResult?.message,\n      noOutputExpected: isSilentBashCommand(input.command),\n      backgroundTaskId: result.backgroundTaskId,\n      backgroundedByUser: result.backgroundedByUser,\n      assistantAutoBackgrounded: result.assistantAutoBackgrounded,\n      dangerouslyDisableSandbox:\n        'dangerouslyDisableSandbox' in input\n          ? (input.dangerouslyDisableSandbox as boolean | undefined)\n          : undefined,\n      persistedOutputPath,\n      persistedOutputSize,\n    }\n\n    return {\n      data,\n    }\n  },\n  renderToolUseErrorMessage,\n  isResultTruncated(output: Out): boolean {\n    return (\n      isOutputLineTruncated(output.stdout) ||\n      isOutputLineTruncated(output.stderr)\n    )\n  },\n} satisfies ToolDef<InputSchema, Out, BashProgress>)\n\nasync function* runShellCommand({\n  input,\n  abortController,\n  setAppState,\n  setToolJSX,\n  preventCwdChanges,\n  isMainThread,\n  toolUseId,\n  agentId,\n}: {\n  input: BashToolInput\n  abortController: AbortController\n  setAppState: (f: (prev: AppState) => AppState) => void\n  setToolJSX?: SetToolJSXFn\n  preventCwdChanges?: boolean\n  isMainThread?: boolean\n  toolUseId?: string\n  agentId?: AgentId\n}): AsyncGenerator<\n  {\n    type: 'progress'\n    output: string\n    fullOutput: string\n    elapsedTimeSeconds: number\n    totalLines: number\n    totalBytes?: number\n    taskId?: string\n    timeoutMs?: number\n  },\n  ExecResult,\n  void\n> {\n  const { command, description, timeout, run_in_background } = input\n  const timeoutMs = timeout || getDefaultTimeoutMs()\n\n  let fullOutput = ''\n  let lastProgressOutput = ''\n  let lastTotalLines = 0\n  let lastTotalBytes = 0\n  let backgroundShellId: string | undefined = undefined\n  let assistantAutoBackgrounded = false\n\n  // Progress signal: resolved by onProgress callback from the shared poller,\n  // waking the generator to yield a progress update.\n  let resolveProgress: (() => void) | null = null\n  function createProgressSignal(): Promise<null> {\n    return new Promise<null>(resolve => {\n      resolveProgress = () => resolve(null)\n    })\n  }\n\n  // Determine if auto-backgrounding should be enabled\n  // Only enable for commands that are allowed to be auto-backgrounded\n  // and when background tasks are not disabled\n  const shouldAutoBackground =\n    !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command)\n\n  const shellCommand = await exec(command, abortController.signal, 'bash', {\n    timeout: timeoutMs,\n    onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {\n      lastProgressOutput = lastLines\n      fullOutput = allLines\n      lastTotalLines = totalLines\n      lastTotalBytes = isIncomplete ? totalBytes : 0\n      // Wake the generator so it yields the new progress data\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n    },\n    preventCwdChanges,\n    shouldUseSandbox: shouldUseSandbox(input),\n    shouldAutoBackground,\n  })\n\n  // Start the command execution\n  const resultPromise = shellCommand.result\n\n  // Helper to spawn a background task and return its ID\n  async function spawnBackgroundTask(): Promise<string> {\n    const handle = await spawnShellTask(\n      {\n        command,\n        description: description || command,\n        shellCommand,\n        toolUseId,\n        agentId,\n      },\n      {\n        abortController,\n        getAppState: () => {\n          // We don't have direct access to getAppState here, but spawn doesn't\n          // actually use it during the spawn process\n          throw new Error(\n            'getAppState not available in runShellCommand context',\n          )\n        },\n        setAppState,\n      },\n    )\n    return handle.taskId\n  }\n\n  // Helper to start backgrounding with optional logging\n  function startBackgrounding(\n    eventName: string,\n    backgroundFn?: (shellId: string) => void,\n  ): void {\n    // If a foreground task is already registered (via registerForeground in the\n    // progress loop), background it in-place instead of re-spawning. Re-spawning\n    // would overwrite tasks[taskId], emit a duplicate task_started SDK event,\n    // and leak the first cleanup callback.\n    if (foregroundTaskId) {\n      if (\n        !backgroundExistingForegroundTask(\n          foregroundTaskId,\n          shellCommand,\n          description || command,\n          setAppState,\n          toolUseId,\n        )\n      ) {\n        return\n      }\n      backgroundShellId = foregroundTaskId\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n      backgroundFn?.(foregroundTaskId)\n      return\n    }\n\n    // No foreground task registered — spawn a new background task\n    // Note: spawn is essentially synchronous despite being async\n    void spawnBackgroundTask().then(shellId => {\n      backgroundShellId = shellId\n\n      // Wake the generator's Promise.race so it sees backgroundShellId.\n      // Without this, if the poller has stopped ticking for this task\n      // (no output + shared-poller race with sibling stopPolling calls)\n      // and the process is hung on I/O, the race at line ~1357 never\n      // resolves and the generator deadlocks despite being backgrounded.\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n\n      if (backgroundFn) {\n        backgroundFn(shellId)\n      }\n    })\n  }\n\n  // Set up auto-backgrounding on timeout if enabled\n  // Only background commands that are allowed to be auto-backgrounded (not sleep, etc.)\n  if (shellCommand.onTimeout && shouldAutoBackground) {\n    shellCommand.onTimeout(backgroundFn => {\n      startBackgrounding(\n        'tengu_bash_command_timeout_backgrounded',\n        backgroundFn,\n      )\n    })\n  }\n\n  // In assistant mode, the main agent should stay responsive. Auto-background\n  // blocking commands after ASSISTANT_BLOCKING_BUDGET_MS so the agent can keep\n  // coordinating instead of waiting. The command keeps running — no state loss.\n  if (\n    feature('KAIROS') &&\n    getKairosActive() &&\n    isMainThread &&\n    !isBackgroundTasksDisabled &&\n    run_in_background !== true\n  ) {\n    setTimeout(() => {\n      if (\n        shellCommand.status === 'running' &&\n        backgroundShellId === undefined\n      ) {\n        assistantAutoBackgrounded = true\n        startBackgrounding('tengu_bash_command_assistant_auto_backgrounded')\n      }\n    }, ASSISTANT_BLOCKING_BUDGET_MS).unref()\n  }\n\n  // Handle Claude asking to run it in the background explicitly\n  // When explicitly requested via run_in_background, always honor the request\n  // regardless of the command type (isAutobackgroundingAllowed only applies to automatic backgrounding)\n  // Skip if background tasks are disabled - run in foreground instead\n  if (run_in_background === true && !isBackgroundTasksDisabled) {\n    const shellId = await spawnBackgroundTask()\n\n    logEvent('tengu_bash_command_explicitly_backgrounded', {\n      command_type: getCommandTypeForLogging(command),\n    })\n\n    return {\n      stdout: '',\n      stderr: '',\n      code: 0,\n      interrupted: false,\n      backgroundTaskId: shellId,\n    }\n  }\n\n  // Wait for the initial threshold before showing progress\n  const startTime = Date.now()\n  let foregroundTaskId: string | undefined = undefined\n\n  {\n    const initialResult = await Promise.race([\n      resultPromise,\n      new Promise<null>(resolve => {\n        const t = setTimeout(\n          (r: (v: null) => void) => r(null),\n          PROGRESS_THRESHOLD_MS,\n          resolve,\n        )\n        t.unref()\n      }),\n    ])\n\n    if (initialResult !== null) {\n      shellCommand.cleanup()\n      return initialResult\n    }\n\n    if (backgroundShellId) {\n      return {\n        stdout: '',\n        stderr: '',\n        code: 0,\n        interrupted: false,\n        backgroundTaskId: backgroundShellId,\n        assistantAutoBackgrounded,\n      }\n    }\n  }\n\n  // Start polling the output file for progress. The poller's #tick calls\n  // onProgress every second, which resolves progressSignal below.\n  TaskOutput.startPolling(shellCommand.taskOutput.taskId)\n\n  // Progress loop: wake is driven by the shared poller calling onProgress,\n  // which resolves the progressSignal.\n  try {\n    while (true) {\n      const progressSignal = createProgressSignal()\n      const result = await Promise.race([resultPromise, progressSignal])\n\n      if (result !== null) {\n        // Race: backgrounding fired (15s timer / onTimeout / Ctrl+B) but the\n        // command completed before the next poll tick. #handleExit sets\n        // backgroundTaskId but skips outputFilePath (it assumes the background\n        // message or <task_notification> will carry the path). Strip\n        // backgroundTaskId so the model sees a clean completed command,\n        // reconstruct outputFilePath for large outputs, and suppress the\n        // redundant <task_notification> from the .then() handler.\n        // Check result.backgroundTaskId (not the closure var) to also cover\n        // Ctrl+B, which calls shellCommand.background() directly.\n        if (result.backgroundTaskId !== undefined) {\n          markTaskNotified(result.backgroundTaskId, setAppState)\n          const fixedResult: ExecResult = {\n            ...result,\n            backgroundTaskId: undefined,\n          }\n          // Mirror ShellCommand.#handleExit's large-output branch that was\n          // skipped because #backgroundTaskId was set.\n          const { taskOutput } = shellCommand\n          if (taskOutput.stdoutToFile && !taskOutput.outputFileRedundant) {\n            fixedResult.outputFilePath = taskOutput.path\n            fixedResult.outputFileSize = taskOutput.outputFileSize\n            fixedResult.outputTaskId = taskOutput.taskId\n          }\n          shellCommand.cleanup()\n          return fixedResult\n        }\n        // Command has completed - return the actual result\n        // If we registered as a foreground task, unregister it\n        if (foregroundTaskId) {\n          unregisterForeground(foregroundTaskId, setAppState)\n        }\n        // Clean up stream resources for foreground commands\n        // (backgrounded commands are cleaned up by LocalShellTask)\n        shellCommand.cleanup()\n        return result\n      }\n\n      // Check if command was backgrounded (either via old mechanism or new backgroundAll)\n      if (backgroundShellId) {\n        return {\n          stdout: '',\n          stderr: '',\n          code: 0,\n          interrupted: false,\n          backgroundTaskId: backgroundShellId,\n          assistantAutoBackgrounded,\n        }\n      }\n\n      // Check if this foreground task was backgrounded via backgroundAll()\n      if (foregroundTaskId) {\n        // shellCommand.status becomes 'backgrounded' when background() is called\n        if (shellCommand.status === 'backgrounded') {\n          return {\n            stdout: '',\n            stderr: '',\n            code: 0,\n            interrupted: false,\n            backgroundTaskId: foregroundTaskId,\n            backgroundedByUser: true,\n          }\n        }\n      }\n\n      // Time for a progress update\n      const elapsed = Date.now() - startTime\n      const elapsedSeconds = Math.floor(elapsed / 1000)\n\n      // Show minimal backgrounding UI if available\n      // Skip if background tasks are disabled\n      if (\n        !isBackgroundTasksDisabled &&\n        backgroundShellId === undefined &&\n        elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 &&\n        setToolJSX\n      ) {\n        // Register this command as a foreground task so it can be backgrounded via Ctrl+B\n        if (!foregroundTaskId) {\n          foregroundTaskId = registerForeground(\n            {\n              command,\n              description: description || command,\n              shellCommand,\n              agentId,\n            },\n            setAppState,\n            toolUseId,\n          )\n        }\n\n        setToolJSX({\n          jsx: <BackgroundHint />,\n          shouldHidePromptInput: false,\n          shouldContinueAnimation: true,\n          showSpinner: true,\n        })\n      }\n      yield {\n        type: 'progress',\n        fullOutput,\n        output: lastProgressOutput,\n        elapsedTimeSeconds: elapsedSeconds,\n        totalLines: lastTotalLines,\n        totalBytes: lastTotalBytes,\n        taskId: shellCommand.taskOutput.taskId,\n        ...(timeout ? { timeoutMs } : undefined),\n      }\n    }\n  } finally {\n    TaskOutput.stopPolling(shellCommand.taskOutput.taskId)\n  }\n}\n"],"mappings":"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","ignoreList":[]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","ToolResultBlockParam","copyFile","stat","fsStat","truncate","fsTruncate","link","React","CanUseToolFn","AppState","z","getKairosActive","TOOL_SUMMARY_MAX_LENGTH","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","notifyVscodeFileUpdated","SetToolJSXFn","ToolCallProgress","ToolUseContext","ValidationResult","buildTool","ToolDef","backgroundExistingForegroundTask","markTaskNotified","registerForeground","spawnShellTask","unregisterForeground","AgentId","AssistantMessage","parseForSecurity","splitCommand_DEPRECATED","splitCommandWithOperators","extractClaudeCodeHints","detectCodeIndexingFromCommand","isEnvTruthy","isENOENT","ShellError","detectFileEncoding","detectLineEndings","getFileModificationTime","writeTextContent","fileHistoryEnabled","fileHistoryTrackEdit","getFsImplementation","lazySchema","expandPath","PermissionResult","maybeRecordPluginHint","exec","ExecResult","SandboxManager","semanticBoolean","semanticNumber","EndTruncatingAccumulator","getTaskOutputPath","TaskOutput","isOutputLineTruncated","buildLargeToolResultMessage","ensureToolResultsDir","generatePreview","getToolResultPath","PREVIEW_SIZE_BYTES","userFacingName","fileEditUserFacingName","trackGitOperations","bashToolHasPermission","commandHasAnyCd","matchWildcardPattern","permissionRuleExtractPrefix","interpretCommandResult","getDefaultTimeoutMs","getMaxTimeoutMs","getSimplePrompt","checkReadOnlyConstraints","parseSedEditCommand","shouldUseSandbox","BASH_TOOL_NAME","BackgroundHint","renderToolResultMessage","renderToolUseErrorMessage","renderToolUseMessage","renderToolUseProgressMessage","renderToolUseQueuedMessage","buildImageToolResult","isImageOutput","resetCwdIfOutsideProject","resizeShellImageOutput","stdErrAppendShellResetMessage","stripEmptyLines","EOL","PROGRESS_THRESHOLD_MS","ASSISTANT_BLOCKING_BUDGET_MS","BASH_SEARCH_COMMANDS","Set","BASH_READ_COMMANDS","BASH_LIST_COMMANDS","BASH_SEMANTIC_NEUTRAL_COMMANDS","BASH_SILENT_COMMANDS","isSearchOrReadBashCommand","command","isSearch","isRead","isList","partsWithOperators","length","hasSearch","hasRead","hasList","hasNonNeutralCommand","skipNextAsRedirectTarget","part","baseCommand","trim","split","has","isPartSearch","isPartRead","isPartList","isSilentBashCommand","hasNonFallbackCommand","lastOperator","DISALLOWED_AUTO_BACKGROUND_COMMANDS","isBackgroundTasksDisabled","process","env","CLAUDE_CODE_DISABLE_BACKGROUND_TASKS","fullInputSchema","strictObject","string","describe","timeout","number","optional","description","run_in_background","boolean","dangerouslyDisableSandbox","_simulatedSedEdit","object","filePath","newContent","inputSchema","omit","InputSchema","ReturnType","BashToolInput","infer","COMMON_BACKGROUND_COMMANDS","const","getCommandTypeForLogging","parts","includes","outputSchema","stdout","stderr","rawOutputPath","interrupted","isImage","backgroundTaskId","backgroundedByUser","assistantAutoBackgrounded","returnCodeInterpretation","noOutputExpected","structuredContent","array","any","persistedOutputPath","persistedOutputSize","OutputSchema","Out","BashProgress","isAutobackgroundingAllowed","detectBlockedSleepPattern","first","m","secs","parseInt","rest","slice","join","SimulatedSedEditResult","data","SimulatedSedEditContext","Pick","applySedEdit","simulatedEdit","toolUseContext","parentMessage","Promise","absoluteFilePath","fs","encoding","originalContent","readFile","e","updateFileHistoryState","uuid","endings","readFileState","set","content","timestamp","offset","undefined","limit","BashTool","name","searchHint","maxResultSizeChars","strict","prompt","isConcurrencySafe","input","isReadOnly","compoundCommandHasCd","result","behavior","toAutoClassifierInput","preparePermissionMatcher","parsed","kind","subcommands","commands","map","c","argv","pattern","prefix","some","cmd","startsWith","isSearchOrReadCommand","safeParse","success","sedInfo","file_path","old_string","CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR","getToolUseSummary","getActivityDescription","desc","validateInput","sleepPattern","message","errorCode","checkPermissions","context","extractSearchText","mapToolResultToToolResultBlockParam","toolUseID","tool_use_id","type","block","processedStdout","replace","trimEnd","preview","filepath","originalSize","isJson","hasMore","errorMessage","backgroundInfo","outputPath","filter","Boolean","is_error","call","_canUseTool","onProgress","abortController","getAppState","setAppState","setToolJSX","stdoutAccumulator","stderrForShellReset","interpretationResult","progressCounter","wasInterrupted","isMainThread","agentId","preventCwdChanges","commandGenerator","runShellCommand","setAppStateForTasks","toolUseId","generatorResult","next","done","progress","value","output","fullOutput","elapsedTimeSeconds","totalLines","totalBytes","taskId","timeoutMs","code","isInterrupt","signal","reason","append","isError","appState","toolPermissionContext","outputWithSbFailures","annotateStderrWithSandboxFailures","preSpawnError","Error","toString","MAX_PERSISTED_SIZE","outputFilePath","outputTaskId","fileStat","size","dest","commandType","command_type","stdout_length","stderr_length","exit_code","codeIndexingTool","tool","source","strippedStdout","extracted","stripped","hints","hint","compressedStdout","resized","isResultTruncated","AbortController","f","prev","AsyncGenerator","lastProgressOutput","lastTotalLines","lastTotalBytes","backgroundShellId","resolveProgress","createProgressSignal","resolve","shouldAutoBackground","shellCommand","lastLines","allLines","isIncomplete","resultPromise","spawnBackgroundTask","handle","startBackgrounding","eventName","backgroundFn","shellId","foregroundTaskId","then","onTimeout","setTimeout","status","unref","startTime","Date","now","initialResult","race","t","r","v","cleanup","startPolling","taskOutput","progressSignal","fixedResult","stdoutToFile","outputFileRedundant","path","outputFileSize","elapsed","elapsedSeconds","Math","floor","jsx","shouldHidePromptInput","shouldContinueAnimation","showSpinner","stopPolling"],"sources":["BashTool.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'\nimport {\n  copyFile,\n  stat as fsStat,\n  truncate as fsTruncate,\n  link,\n} from 'fs/promises'\nimport * as React from 'react'\nimport type { CanUseToolFn } from 'src/hooks/useCanUseTool.js'\nimport type { AppState } from 'src/state/AppState.js'\nimport { z } from 'zod/v4'\nimport { getKairosActive } from '../../bootstrap/state.js'\nimport { TOOL_SUMMARY_MAX_LENGTH } from '../../constants/toolLimits.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from '../../services/analytics/index.js'\nimport { notifyVscodeFileUpdated } from '../../services/mcp/vscodeSdkMcp.js'\nimport type {\n  SetToolJSXFn,\n  ToolCallProgress,\n  ToolUseContext,\n  ValidationResult,\n} from '../../Tool.js'\nimport { buildTool, type ToolDef } from '../../Tool.js'\nimport {\n  backgroundExistingForegroundTask,\n  markTaskNotified,\n  registerForeground,\n  spawnShellTask,\n  unregisterForeground,\n} from '../../tasks/LocalShellTask/LocalShellTask.js'\nimport type { AgentId } from '../../types/ids.js'\nimport type { AssistantMessage } from '../../types/message.js'\nimport { parseForSecurity } from '../../utils/bash/ast.js'\nimport {\n  splitCommand_DEPRECATED,\n  splitCommandWithOperators,\n} from '../../utils/bash/commands.js'\nimport { extractClaudeCodeHints } from '../../utils/claudeCodeHints.js'\nimport { detectCodeIndexingFromCommand } from '../../utils/codeIndexing.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isENOENT, ShellError } from '../../utils/errors.js'\nimport {\n  detectFileEncoding,\n  detectLineEndings,\n  getFileModificationTime,\n  writeTextContent,\n} from '../../utils/file.js'\nimport {\n  fileHistoryEnabled,\n  fileHistoryTrackEdit,\n} from '../../utils/fileHistory.js'\nimport { truncate } from '../../utils/format.js'\nimport { getFsImplementation } from '../../utils/fsOperations.js'\nimport { lazySchema } from '../../utils/lazySchema.js'\nimport { expandPath } from '../../utils/path.js'\nimport type { PermissionResult } from '../../utils/permissions/PermissionResult.js'\nimport { maybeRecordPluginHint } from '../../utils/plugins/hintRecommendation.js'\nimport { exec } from '../../utils/Shell.js'\nimport type { ExecResult } from '../../utils/ShellCommand.js'\nimport { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js'\nimport { semanticBoolean } from '../../utils/semanticBoolean.js'\nimport { semanticNumber } from '../../utils/semanticNumber.js'\nimport { EndTruncatingAccumulator } from '../../utils/stringUtils.js'\nimport { getTaskOutputPath } from '../../utils/task/diskOutput.js'\nimport { TaskOutput } from '../../utils/task/TaskOutput.js'\nimport { isOutputLineTruncated } from '../../utils/terminal.js'\nimport {\n  buildLargeToolResultMessage,\n  ensureToolResultsDir,\n  generatePreview,\n  getToolResultPath,\n  PREVIEW_SIZE_BYTES,\n} from '../../utils/toolResultStorage.js'\nimport { userFacingName as fileEditUserFacingName } from '../FileEditTool/UI.js'\nimport { trackGitOperations } from '../shared/gitOperationTracking.js'\nimport {\n  bashToolHasPermission,\n  commandHasAnyCd,\n  matchWildcardPattern,\n  permissionRuleExtractPrefix,\n} from './bashPermissions.js'\nimport { interpretCommandResult } from './commandSemantics.js'\nimport {\n  getDefaultTimeoutMs,\n  getMaxTimeoutMs,\n  getSimplePrompt,\n} from './prompt.js'\nimport { checkReadOnlyConstraints } from './readOnlyValidation.js'\nimport { parseSedEditCommand } from './sedEditParser.js'\nimport { shouldUseSandbox } from './shouldUseSandbox.js'\nimport { BASH_TOOL_NAME } from './toolName.js'\nimport {\n  BackgroundHint,\n  renderToolResultMessage,\n  renderToolUseErrorMessage,\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n} from './UI.js'\nimport {\n  buildImageToolResult,\n  isImageOutput,\n  resetCwdIfOutsideProject,\n  resizeShellImageOutput,\n  stdErrAppendShellResetMessage,\n  stripEmptyLines,\n} from './utils.js'\n\nconst EOL = '\\n'\n\n// Progress display constants\nconst PROGRESS_THRESHOLD_MS = 2000 // Show progress after 2 seconds\n// In assistant mode, blocking bash auto-backgrounds after this many ms in the main agent\nconst ASSISTANT_BLOCKING_BUDGET_MS = 15_000\n\n// Search commands for collapsible display (grep, find, etc.)\nconst BASH_SEARCH_COMMANDS = new Set([\n  'find',\n  'grep',\n  'rg',\n  'ag',\n  'ack',\n  'locate',\n  'which',\n  'whereis',\n])\n\n// Read/view commands for collapsible display (cat, head, etc.)\nconst BASH_READ_COMMANDS = new Set([\n  'cat',\n  'head',\n  'tail',\n  'less',\n  'more',\n  // Analysis commands\n  'wc',\n  'stat',\n  'file',\n  'strings',\n  // Data processing — commonly used to parse/transform file content in pipes\n  'jq',\n  'awk',\n  'cut',\n  'sort',\n  'uniq',\n  'tr',\n])\n\n// Directory-listing commands for collapsible display (ls, tree, du).\n// Split from BASH_READ_COMMANDS so the summary says \"Listed N directories\"\n// instead of the misleading \"Read N files\".\nconst BASH_LIST_COMMANDS = new Set(['ls', 'tree', 'du'])\n\n// Commands that are semantic-neutral in any position — pure output/status commands\n// that don't change the read/search nature of the overall pipeline.\n// e.g. `ls dir && echo \"---\" && ls dir2` is still a read-only compound command.\nconst BASH_SEMANTIC_NEUTRAL_COMMANDS = new Set([\n  'echo',\n  'printf',\n  'true',\n  'false',\n  ':', // bash no-op\n])\n\n// Commands that typically produce no stdout on success\nconst BASH_SILENT_COMMANDS = new Set([\n  'mv',\n  'cp',\n  'rm',\n  'mkdir',\n  'rmdir',\n  'chmod',\n  'chown',\n  'chgrp',\n  'touch',\n  'ln',\n  'cd',\n  'export',\n  'unset',\n  'wait',\n])\n\n/**\n * Checks if a bash command is a search or read operation.\n * Used to determine if the command should be collapsed in the UI.\n * Returns an object indicating whether it's a search or read operation.\n *\n * For pipelines (e.g., `cat file | bq`), ALL parts must be search/read commands\n * for the whole command to be considered collapsible.\n *\n * Semantic-neutral commands (echo, printf, true, false, :) are skipped in any\n * position, as they're pure output/status commands that don't affect the read/search\n * nature of the pipeline (e.g. `ls dir && echo \"---\" && ls dir2` is still a read).\n */\nexport function isSearchOrReadBashCommand(command: string): {\n  isSearch: boolean\n  isRead: boolean\n  isList: boolean\n} {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    // If we can't parse the command due to malformed syntax,\n    // it's not a search/read command\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  if (partsWithOperators.length === 0) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  let hasSearch = false\n  let hasRead = false\n  let hasList = false\n  let hasNonNeutralCommand = false\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)) {\n      continue\n    }\n\n    hasNonNeutralCommand = true\n\n    const isPartSearch = BASH_SEARCH_COMMANDS.has(baseCommand)\n    const isPartRead = BASH_READ_COMMANDS.has(baseCommand)\n    const isPartList = BASH_LIST_COMMANDS.has(baseCommand)\n\n    if (!isPartSearch && !isPartRead && !isPartList) {\n      return { isSearch: false, isRead: false, isList: false }\n    }\n\n    if (isPartSearch) hasSearch = true\n    if (isPartRead) hasRead = true\n    if (isPartList) hasList = true\n  }\n\n  // Only neutral commands (e.g., just \"echo foo\") -- not collapsible\n  if (!hasNonNeutralCommand) {\n    return { isSearch: false, isRead: false, isList: false }\n  }\n\n  return { isSearch: hasSearch, isRead: hasRead, isList: hasList }\n}\n\n/**\n * Checks if a bash command is expected to produce no stdout on success.\n * Used to show \"Done\" instead of \"(No output)\" in the UI.\n */\nfunction isSilentBashCommand(command: string): boolean {\n  let partsWithOperators: string[]\n  try {\n    partsWithOperators = splitCommandWithOperators(command)\n  } catch {\n    return false\n  }\n\n  if (partsWithOperators.length === 0) {\n    return false\n  }\n\n  let hasNonFallbackCommand = false\n  let lastOperator: string | null = null\n  let skipNextAsRedirectTarget = false\n\n  for (const part of partsWithOperators) {\n    if (skipNextAsRedirectTarget) {\n      skipNextAsRedirectTarget = false\n      continue\n    }\n\n    if (part === '>' || part === '>>' || part === '>&') {\n      skipNextAsRedirectTarget = true\n      continue\n    }\n\n    if (part === '||' || part === '&&' || part === '|' || part === ';') {\n      lastOperator = part\n      continue\n    }\n\n    const baseCommand = part.trim().split(/\\s+/)[0]\n    if (!baseCommand) {\n      continue\n    }\n\n    if (\n      lastOperator === '||' &&\n      BASH_SEMANTIC_NEUTRAL_COMMANDS.has(baseCommand)\n    ) {\n      continue\n    }\n\n    hasNonFallbackCommand = true\n\n    if (!BASH_SILENT_COMMANDS.has(baseCommand)) {\n      return false\n    }\n  }\n\n  return hasNonFallbackCommand\n}\n\n// Commands that should not be auto-backgrounded\nconst DISALLOWED_AUTO_BACKGROUND_COMMANDS = [\n  'sleep', // Sleep should run in foreground unless explicitly backgrounded by user\n]\n\n// Check if background tasks are disabled at module load time\nconst isBackgroundTasksDisabled =\n  // eslint-disable-next-line custom-rules/no-process-env-top-level -- Intentional: schema must be defined at module load\n  isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_BACKGROUND_TASKS)\n\nconst fullInputSchema = lazySchema(() =>\n  z.strictObject({\n    command: z.string().describe('The command to execute'),\n    timeout: semanticNumber(z.number().optional()).describe(\n      `Optional timeout in milliseconds (max ${getMaxTimeoutMs()})`,\n    ),\n    description: z\n      .string()\n      .optional()\n      .describe(`Clear, concise description of what this command does in active voice. Never use words like \"complex\" or \"risk\" in the description - just describe what it does.\n\nFor simple commands (git, npm, standard CLI tools), keep it brief (5-10 words):\n- ls → \"List files in current directory\"\n- git status → \"Show working tree status\"\n- npm install → \"Install package dependencies\"\n\nFor commands that are harder to parse at a glance (piped commands, obscure flags, etc.), add enough context to clarify what it does:\n- find . -name \"*.tmp\" -exec rm {} \\\\; → \"Find and delete all .tmp files recursively\"\n- git reset --hard origin/main → \"Discard all local changes and match remote main\"\n- curl -s url | jq '.data[]' → \"Fetch JSON from URL and extract data array elements\"`),\n    run_in_background: semanticBoolean(z.boolean().optional()).describe(\n      `Set to true to run this command in the background. Use Read to read the output later.`,\n    ),\n    dangerouslyDisableSandbox: semanticBoolean(z.boolean().optional()).describe(\n      'Set this to true to dangerously override sandbox mode and run commands without sandboxing.',\n    ),\n    _simulatedSedEdit: z\n      .object({\n        filePath: z.string(),\n        newContent: z.string(),\n      })\n      .optional()\n      .describe('Internal: pre-computed sed edit result from preview'),\n  }),\n)\n\n// Always omit _simulatedSedEdit from the model-facing schema. It is an internal-only\n// field set by SedEditPermissionRequest after the user approves a sed edit preview.\n// Exposing it in the schema would let the model bypass permission checks and the\n// sandbox by pairing an innocuous command with an arbitrary file write.\n// Also conditionally remove run_in_background when background tasks are disabled.\nconst inputSchema = lazySchema(() =>\n  isBackgroundTasksDisabled\n    ? fullInputSchema().omit({\n        run_in_background: true,\n        _simulatedSedEdit: true,\n      })\n    : fullInputSchema().omit({ _simulatedSedEdit: true }),\n)\ntype InputSchema = ReturnType<typeof inputSchema>\n\n// Use fullInputSchema for the type to always include run_in_background\n// (even when it's omitted from the schema, the code needs to handle it)\nexport type BashToolInput = z.infer<ReturnType<typeof fullInputSchema>>\n\nconst COMMON_BACKGROUND_COMMANDS = [\n  'npm',\n  'yarn',\n  'pnpm',\n  'node',\n  'python',\n  'python3',\n  'go',\n  'cargo',\n  'make',\n  'docker',\n  'terraform',\n  'webpack',\n  'vite',\n  'jest',\n  'pytest',\n  'curl',\n  'wget',\n  'build',\n  'test',\n  'serve',\n  'watch',\n  'dev',\n] as const\n\nfunction getCommandTypeForLogging(\n  command: string,\n): AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0)\n    return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n\n  // Check each part of the command to see if any match common background commands\n  for (const part of parts) {\n    const baseCommand = part.split(' ')[0] || ''\n    if (\n      COMMON_BACKGROUND_COMMANDS.includes(\n        baseCommand as (typeof COMMON_BACKGROUND_COMMANDS)[number],\n      )\n    ) {\n      return baseCommand as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    }\n  }\n\n  return 'other' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n}\n\nconst outputSchema = lazySchema(() =>\n  z.object({\n    stdout: z.string().describe('The standard output of the command'),\n    stderr: z.string().describe('The standard error output of the command'),\n    rawOutputPath: z\n      .string()\n      .optional()\n      .describe('Path to raw output file for large MCP tool outputs'),\n    interrupted: z.boolean().describe('Whether the command was interrupted'),\n    isImage: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if stdout contains image data'),\n    backgroundTaskId: z\n      .string()\n      .optional()\n      .describe(\n        'ID of the background task if command is running in background',\n      ),\n    backgroundedByUser: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if the user manually backgrounded the command with Ctrl+B',\n      ),\n    assistantAutoBackgrounded: z\n      .boolean()\n      .optional()\n      .describe(\n        'True if assistant-mode auto-backgrounded a long-running blocking command',\n      ),\n    dangerouslyDisableSandbox: z\n      .boolean()\n      .optional()\n      .describe('Flag to indicate if sandbox mode was overridden'),\n    returnCodeInterpretation: z\n      .string()\n      .optional()\n      .describe(\n        'Semantic interpretation for non-error exit codes with special meaning',\n      ),\n    noOutputExpected: z\n      .boolean()\n      .optional()\n      .describe(\n        'Whether the command is expected to produce no output on success',\n      ),\n    structuredContent: z\n      .array(z.any())\n      .optional()\n      .describe('Structured content blocks'),\n    persistedOutputPath: z\n      .string()\n      .optional()\n      .describe(\n        'Path to the persisted full output in tool-results dir (set when output is too large for inline)',\n      ),\n    persistedOutputSize: z\n      .number()\n      .optional()\n      .describe(\n        'Total size of the output in bytes (set when output is too large for inline)',\n      ),\n  }),\n)\n\ntype OutputSchema = ReturnType<typeof outputSchema>\nexport type Out = z.infer<OutputSchema>\n\n// Re-export BashProgress from centralized types to break import cycles\nexport type { BashProgress } from '../../types/tools.js'\n\nimport type { BashProgress } from '../../types/tools.js'\n\n/**\n * Checks if a command is allowed to be automatically backgrounded\n * @param command The command to check\n * @returns false for commands that should not be auto-backgrounded (like sleep)\n */\nfunction isAutobackgroundingAllowed(command: string): boolean {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return true\n\n  // Get the first part which should be the base command\n  const baseCommand = parts[0]?.trim()\n  if (!baseCommand) return true\n\n  return !DISALLOWED_AUTO_BACKGROUND_COMMANDS.includes(baseCommand)\n}\n\n/**\n * Detect standalone or leading `sleep N` patterns that should use Monitor\n * instead. Catches `sleep 5`, `sleep 5 && check`, `sleep 5; check` — but\n * not sleep inside pipelines, subshells, or scripts (those are fine).\n */\nexport function detectBlockedSleepPattern(command: string): string | null {\n  const parts = splitCommand_DEPRECATED(command)\n  if (parts.length === 0) return null\n\n  const first = parts[0]?.trim() ?? ''\n  // Bare `sleep N` or `sleep N.N` as the first subcommand.\n  // Float durations (sleep 0.5) are allowed — those are legit pacing, not polls.\n  const m = /^sleep\\s+(\\d+)\\s*$/.exec(first)\n  if (!m) return null\n  const secs = parseInt(m[1]!, 10)\n  if (secs < 2) return null // sub-2s sleeps are fine (rate limiting, pacing)\n\n  // `sleep N` alone → \"what are you waiting for?\"\n  // `sleep N && check` → \"use Monitor { command: check }\"\n  const rest = parts.slice(1).join(' ').trim()\n  return rest\n    ? `sleep ${secs} followed by: ${rest}`\n    : `standalone sleep ${secs}`\n}\n\n/**\n * Checks if a command contains tools that shouldn't run in sandbox\n * This includes:\n * - Dynamic config-based disabled commands and substrings (tengu_sandbox_disabled_commands)\n * - User-configured commands from settings.json (sandbox.excludedCommands)\n *\n * User-configured commands support the same pattern syntax as permission rules:\n * - Exact matches: \"npm run lint\"\n * - Prefix patterns: \"npm run test:*\"\n */\n\ntype SimulatedSedEditResult = {\n  data: Out\n}\n\ntype SimulatedSedEditContext = Pick<\n  ToolUseContext,\n  'readFileState' | 'updateFileHistoryState'\n>\n\n/**\n * Applies a simulated sed edit directly instead of running sed.\n * This is used by the permission dialog to ensure what the user previews\n * is exactly what gets written to the file.\n */\nasync function applySedEdit(\n  simulatedEdit: { filePath: string; newContent: string },\n  toolUseContext: SimulatedSedEditContext,\n  parentMessage?: AssistantMessage,\n): Promise<SimulatedSedEditResult> {\n  const { filePath, newContent } = simulatedEdit\n  const absoluteFilePath = expandPath(filePath)\n  const fs = getFsImplementation()\n\n  // Read original content for VS Code notification\n  const encoding = detectFileEncoding(absoluteFilePath)\n  let originalContent: string\n  try {\n    originalContent = await fs.readFile(absoluteFilePath, { encoding })\n  } catch (e) {\n    if (isENOENT(e)) {\n      return {\n        data: {\n          stdout: '',\n          stderr: `sed: ${filePath}: No such file or directory\\nExit code 1`,\n          interrupted: false,\n        },\n      }\n    }\n    throw e\n  }\n\n  // Track file history before making changes (for undo support)\n  if (fileHistoryEnabled() && parentMessage) {\n    await fileHistoryTrackEdit(\n      toolUseContext.updateFileHistoryState,\n      absoluteFilePath,\n      parentMessage.uuid,\n    )\n  }\n\n  // Detect line endings and write new content\n  const endings = detectLineEndings(absoluteFilePath)\n  writeTextContent(absoluteFilePath, newContent, encoding, endings)\n\n  // Notify VS Code about the file change\n  notifyVscodeFileUpdated(absoluteFilePath, originalContent, newContent)\n\n  // Update read timestamp to invalidate stale writes\n  toolUseContext.readFileState.set(absoluteFilePath, {\n    content: newContent,\n    timestamp: getFileModificationTime(absoluteFilePath),\n    offset: undefined,\n    limit: undefined,\n  })\n\n  // Return success result matching sed output format (sed produces no output on success)\n  return {\n    data: {\n      stdout: '',\n      stderr: '',\n      interrupted: false,\n    },\n  }\n}\n\nexport const BashTool = buildTool({\n  name: BASH_TOOL_NAME,\n  searchHint: 'execute shell commands',\n  // 30K chars - tool result persistence threshold\n  maxResultSizeChars: 30_000,\n  strict: true,\n  async description({ description }) {\n    return description || 'Run shell command'\n  },\n  async prompt() {\n    return getSimplePrompt()\n  },\n  isConcurrencySafe(input) {\n    return this.isReadOnly?.(input) ?? false\n  },\n  isReadOnly(input) {\n    const compoundCommandHasCd = commandHasAnyCd(input.command)\n    const result = checkReadOnlyConstraints(input, compoundCommandHasCd)\n    return result.behavior === 'allow'\n  },\n  toAutoClassifierInput(input) {\n    return input.command\n  },\n  async preparePermissionMatcher({ command }) {\n    // Hook `if` filtering is \"no match → skip hook\" (deny-like semantics), so\n    // compound commands must fire the hook if ANY subcommand matches. Without\n    // splitting, `ls && git push` would bypass a `Bash(git *)` security hook.\n    const parsed = await parseForSecurity(command)\n    if (parsed.kind !== 'simple') {\n      // parse-unavailable / too-complex: fail safe by running the hook.\n      return () => true\n    }\n    // Match on argv (strips leading VAR=val) so `FOO=bar git push` still\n    // matches `Bash(git *)`.\n    const subcommands = parsed.commands.map(c => c.argv.join(' '))\n    return pattern => {\n      const prefix = permissionRuleExtractPrefix(pattern)\n      return subcommands.some(cmd => {\n        if (prefix !== null) {\n          return cmd === prefix || cmd.startsWith(`${prefix} `)\n        }\n        return matchWildcardPattern(pattern, cmd)\n      })\n    }\n  },\n  isSearchOrReadCommand(input) {\n    const parsed = inputSchema().safeParse(input)\n    if (!parsed.success)\n      return { isSearch: false, isRead: false, isList: false }\n    return isSearchOrReadBashCommand(parsed.data.command)\n  },\n  get inputSchema(): InputSchema {\n    return inputSchema()\n  },\n  get outputSchema(): OutputSchema {\n    return outputSchema()\n  },\n  userFacingName(input) {\n    if (!input) {\n      return 'Bash'\n    }\n    // Render sed in-place edits as file edits\n    if (input.command) {\n      const sedInfo = parseSedEditCommand(input.command)\n      if (sedInfo) {\n        return fileEditUserFacingName({\n          file_path: sedInfo.filePath,\n          old_string: 'x',\n        })\n      }\n    }\n    // Env var FIRST: shouldUseSandbox → splitCommand_DEPRECATED → shell-quote's\n    // `new RegExp` per call. userFacingName runs per-render for every bash\n    // message in history; with ~50 msgs + one slow-to-tokenize command, this\n    // exceeds the shimmer tick → transition abort → infinite retry (#21605).\n    return isEnvTruthy(process.env.CLAUDE_CODE_BASH_SANDBOX_SHOW_INDICATOR) &&\n      shouldUseSandbox(input)\n      ? 'SandboxedBash'\n      : 'Bash'\n  },\n  getToolUseSummary(input) {\n    if (!input?.command) {\n      return null\n    }\n    const { command, description } = input\n    if (description) {\n      return description\n    }\n    return truncate(command, TOOL_SUMMARY_MAX_LENGTH)\n  },\n  getActivityDescription(input) {\n    if (!input?.command) {\n      return 'Running command'\n    }\n    const desc =\n      input.description ?? truncate(input.command, TOOL_SUMMARY_MAX_LENGTH)\n    return `Running ${desc}`\n  },\n  async validateInput(input: BashToolInput): Promise<ValidationResult> {\n    if (\n      feature('MONITOR_TOOL') &&\n      !isBackgroundTasksDisabled &&\n      !input.run_in_background\n    ) {\n      const sleepPattern = detectBlockedSleepPattern(input.command)\n      if (sleepPattern !== null) {\n        return {\n          result: false,\n          message: `Blocked: ${sleepPattern}. Run blocking commands in the background with run_in_background: true — you'll get a completion notification when done. For streaming events (watching logs, polling APIs), use the Monitor tool. If you genuinely need a delay (rate limiting, deliberate pacing), keep it under 2 seconds.`,\n          errorCode: 10,\n        }\n      }\n    }\n    return { result: true }\n  },\n  async checkPermissions(input, context): Promise<PermissionResult> {\n    return bashToolHasPermission(input, context)\n  },\n  renderToolUseMessage,\n  renderToolUseProgressMessage,\n  renderToolUseQueuedMessage,\n  renderToolResultMessage,\n  // BashToolResultMessage shows <OutputLine content={stdout}> + stderr.\n  // UI never shows persistedOutputPath wrapper, backgroundInfo — those are\n  // model-facing (mapToolResult... below).\n  extractSearchText({ stdout, stderr }) {\n    return stderr ? `${stdout}\\n${stderr}` : stdout\n  },\n  mapToolResultToToolResultBlockParam(\n    {\n      interrupted,\n      stdout,\n      stderr,\n      isImage,\n      backgroundTaskId,\n      backgroundedByUser,\n      assistantAutoBackgrounded,\n      structuredContent,\n      persistedOutputPath,\n      persistedOutputSize,\n    },\n    toolUseID,\n  ): ToolResultBlockParam {\n    // Handle structured content\n    if (structuredContent && structuredContent.length > 0) {\n      return {\n        tool_use_id: toolUseID,\n        type: 'tool_result',\n        content: structuredContent,\n      }\n    }\n\n    // For image data, format as image content block for Claude\n    if (isImage) {\n      const block = buildImageToolResult(stdout, toolUseID)\n      if (block) return block\n    }\n\n    let processedStdout = stdout\n    if (stdout) {\n      // Replace any leading newlines or lines with only whitespace\n      processedStdout = stdout.replace(/^(\\s*\\n)+/, '')\n      // Still trim the end as before\n      processedStdout = processedStdout.trimEnd()\n    }\n\n    // For large output that was persisted to disk, build <persisted-output>\n    // message for the model. The UI never sees this — it uses data.stdout.\n    if (persistedOutputPath) {\n      const preview = generatePreview(processedStdout, PREVIEW_SIZE_BYTES)\n      processedStdout = buildLargeToolResultMessage({\n        filepath: persistedOutputPath,\n        originalSize: persistedOutputSize ?? 0,\n        isJson: false,\n        preview: preview.preview,\n        hasMore: preview.hasMore,\n      })\n    }\n\n    let errorMessage = stderr.trim()\n    if (interrupted) {\n      if (stderr) errorMessage += EOL\n      errorMessage += '<error>Command was aborted before completion</error>'\n    }\n\n    let backgroundInfo = ''\n    if (backgroundTaskId) {\n      const outputPath = getTaskOutputPath(backgroundTaskId)\n      if (assistantAutoBackgrounded) {\n        backgroundInfo = `Command exceeded the assistant-mode blocking budget (${ASSISTANT_BLOCKING_BUDGET_MS / 1000}s) and was moved to the background with ID: ${backgroundTaskId}. It is still running — you will be notified when it completes. Output is being written to: ${outputPath}. In assistant mode, delegate long-running work to a subagent or use run_in_background to keep this conversation responsive.`\n      } else if (backgroundedByUser) {\n        backgroundInfo = `Command was manually backgrounded by user with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      } else {\n        backgroundInfo = `Command running in background with ID: ${backgroundTaskId}. Output is being written to: ${outputPath}`\n      }\n    }\n\n    return {\n      tool_use_id: toolUseID,\n      type: 'tool_result',\n      content: [processedStdout, errorMessage, backgroundInfo]\n        .filter(Boolean)\n        .join('\\n'),\n      is_error: interrupted,\n    }\n  },\n  async call(\n    input: BashToolInput,\n    toolUseContext,\n    _canUseTool?: CanUseToolFn,\n    parentMessage?: AssistantMessage,\n    onProgress?: ToolCallProgress<BashProgress>,\n  ) {\n    // Handle simulated sed edit - apply directly instead of running sed\n    // This ensures what the user previewed is exactly what gets written\n    if (input._simulatedSedEdit) {\n      return applySedEdit(\n        input._simulatedSedEdit,\n        toolUseContext,\n        parentMessage,\n      )\n    }\n\n    const { abortController, getAppState, setAppState, setToolJSX } =\n      toolUseContext\n\n    const stdoutAccumulator = new EndTruncatingAccumulator()\n    let stderrForShellReset = ''\n    let interpretationResult:\n      | ReturnType<typeof interpretCommandResult>\n      | undefined\n\n    let progressCounter = 0\n    let wasInterrupted = false\n    let result: ExecResult\n\n    const isMainThread = !toolUseContext.agentId\n    const preventCwdChanges = !isMainThread\n\n    try {\n      // Use the new async generator version of runShellCommand\n      const commandGenerator = runShellCommand({\n        input,\n        abortController,\n        // Use the always-shared task channel so async agents' background\n        // bash tasks are actually registered (and killable on agent exit).\n        setAppState: toolUseContext.setAppStateForTasks ?? setAppState,\n        setToolJSX,\n        preventCwdChanges,\n        isMainThread,\n        toolUseId: toolUseContext.toolUseId,\n        agentId: toolUseContext.agentId,\n      })\n\n      // Consume the generator and capture the return value\n      let generatorResult\n      do {\n        generatorResult = await commandGenerator.next()\n        if (!generatorResult.done && onProgress) {\n          const progress = generatorResult.value\n          onProgress({\n            toolUseID: `bash-progress-${progressCounter++}`,\n            data: {\n              type: 'bash_progress',\n              output: progress.output,\n              fullOutput: progress.fullOutput,\n              elapsedTimeSeconds: progress.elapsedTimeSeconds,\n              totalLines: progress.totalLines,\n              totalBytes: progress.totalBytes,\n              taskId: progress.taskId,\n              timeoutMs: progress.timeoutMs,\n            },\n          })\n        }\n      } while (!generatorResult.done)\n\n      // Get the final result from the generator's return value\n      result = generatorResult.value\n\n      trackGitOperations(input.command, result.code, result.stdout)\n\n      const isInterrupt =\n        result.interrupted && abortController.signal.reason === 'interrupt'\n\n      // stderr is interleaved in stdout (merged fd) — result.stdout has both\n      stdoutAccumulator.append((result.stdout || '').trimEnd() + EOL)\n\n      // Interpret the command result using semantic rules\n      interpretationResult = interpretCommandResult(\n        input.command,\n        result.code,\n        result.stdout || '',\n        '',\n      )\n\n      // Check for git index.lock error (stderr is in stdout now)\n      if (\n        result.stdout &&\n        result.stdout.includes(\".git/index.lock': File exists\")\n      ) {\n        logEvent('tengu_git_index_lock_error', {})\n      }\n\n      if (interpretationResult.isError && !isInterrupt) {\n        // Only add exit code if it's actually an error\n        if (result.code !== 0) {\n          stdoutAccumulator.append(`Exit code ${result.code}`)\n        }\n      }\n\n      if (!preventCwdChanges) {\n        const appState = getAppState()\n        if (resetCwdIfOutsideProject(appState.toolPermissionContext)) {\n          stderrForShellReset = stdErrAppendShellResetMessage('')\n        }\n      }\n\n      // Annotate output with sandbox violations if any (stderr is in stdout)\n      const outputWithSbFailures =\n        SandboxManager.annotateStderrWithSandboxFailures(\n          input.command,\n          result.stdout || '',\n        )\n\n      if (result.preSpawnError) {\n        throw new Error(result.preSpawnError)\n      }\n      if (interpretationResult.isError && !isInterrupt) {\n        // stderr is merged into stdout (merged fd); outputWithSbFailures\n        // already has the full output. Pass '' for stdout to avoid\n        // duplication in getErrorParts() and processBashCommand.\n        throw new ShellError(\n          '',\n          outputWithSbFailures,\n          result.code,\n          result.interrupted,\n        )\n      }\n      wasInterrupted = result.interrupted\n    } finally {\n      if (setToolJSX) setToolJSX(null)\n    }\n\n    // Get final string from accumulator\n    const stdout = stdoutAccumulator.toString()\n\n    // Large output: the file on disk has more than getMaxOutputLength() bytes.\n    // stdout already contains the first chunk (from getStdout()). Copy the\n    // output file to the tool-results dir so the model can read it via\n    // FileRead. If > 64 MB, truncate after copying.\n    const MAX_PERSISTED_SIZE = 64 * 1024 * 1024\n    let persistedOutputPath: string | undefined\n    let persistedOutputSize: number | undefined\n    if (result.outputFilePath && result.outputTaskId) {\n      try {\n        const fileStat = await fsStat(result.outputFilePath)\n        persistedOutputSize = fileStat.size\n\n        await ensureToolResultsDir()\n        const dest = getToolResultPath(result.outputTaskId, false)\n        if (fileStat.size > MAX_PERSISTED_SIZE) {\n          await fsTruncate(result.outputFilePath, MAX_PERSISTED_SIZE)\n        }\n        try {\n          await link(result.outputFilePath, dest)\n        } catch {\n          await copyFile(result.outputFilePath, dest)\n        }\n        persistedOutputPath = dest\n      } catch {\n        // File may already be gone — stdout preview is sufficient\n      }\n    }\n\n    const commandType = input.command.split(' ')[0]\n\n    logEvent('tengu_bash_tool_command_executed', {\n      command_type:\n        commandType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      stdout_length: stdout.length,\n      stderr_length: 0,\n      exit_code: result.code,\n      interrupted: wasInterrupted,\n    })\n\n    // Log code indexing tool usage\n    const codeIndexingTool = detectCodeIndexingFromCommand(input.command)\n    if (codeIndexingTool) {\n      logEvent('tengu_code_indexing_tool_used', {\n        tool: codeIndexingTool as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        source:\n          'cli' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        success: result.code === 0,\n      })\n    }\n\n    let strippedStdout = stripEmptyLines(stdout)\n\n    // Claude Code hints protocol: CLIs/SDKs gated on CLAUDECODE=1 emit a\n    // `<claude-code-hint />` tag to stderr (merged into stdout here). Scan,\n    // record for useClaudeCodeHintRecommendation to surface, then strip\n    // so the model never sees the tag — a zero-token side channel.\n    // Stripping runs unconditionally (subagent output must stay clean too);\n    // only the dialog recording is main-thread-only.\n    const extracted = extractClaudeCodeHints(strippedStdout, input.command)\n    strippedStdout = extracted.stripped\n    if (isMainThread && extracted.hints.length > 0) {\n      for (const hint of extracted.hints) maybeRecordPluginHint(hint)\n    }\n\n    let isImage = isImageOutput(strippedStdout)\n\n    // Cap image dimensions + size if present (CC-304 — see\n    // resizeShellImageOutput). Scope the decoded buffer so it can be reclaimed\n    // before we build the output Out object.\n    let compressedStdout = strippedStdout\n    if (isImage) {\n      const resized = await resizeShellImageOutput(\n        strippedStdout,\n        result.outputFilePath,\n        persistedOutputSize,\n      )\n      if (resized) {\n        compressedStdout = resized\n      } else {\n        // Parse failed or file too large (e.g. exceeds MAX_IMAGE_FILE_SIZE).\n        // Keep isImage in sync with what we actually send so the UI label stays\n        // accurate — mapToolResultToToolResultBlockParam's defensive\n        // fallthrough will send text, not an image block.\n        isImage = false\n      }\n    }\n\n    const data: Out = {\n      stdout: compressedStdout,\n      stderr: stderrForShellReset,\n      interrupted: wasInterrupted,\n      isImage,\n      returnCodeInterpretation: interpretationResult?.message,\n      noOutputExpected: isSilentBashCommand(input.command),\n      backgroundTaskId: result.backgroundTaskId,\n      backgroundedByUser: result.backgroundedByUser,\n      assistantAutoBackgrounded: result.assistantAutoBackgrounded,\n      dangerouslyDisableSandbox:\n        'dangerouslyDisableSandbox' in input\n          ? (input.dangerouslyDisableSandbox as boolean | undefined)\n          : undefined,\n      persistedOutputPath,\n      persistedOutputSize,\n    }\n\n    return {\n      data,\n    }\n  },\n  renderToolUseErrorMessage,\n  isResultTruncated(output: Out): boolean {\n    return (\n      isOutputLineTruncated(output.stdout) ||\n      isOutputLineTruncated(output.stderr)\n    )\n  },\n} satisfies ToolDef<InputSchema, Out, BashProgress>)\n\nasync function* runShellCommand({\n  input,\n  abortController,\n  setAppState,\n  setToolJSX,\n  preventCwdChanges,\n  isMainThread,\n  toolUseId,\n  agentId,\n}: {\n  input: BashToolInput\n  abortController: AbortController\n  setAppState: (f: (prev: AppState) => AppState) => void\n  setToolJSX?: SetToolJSXFn\n  preventCwdChanges?: boolean\n  isMainThread?: boolean\n  toolUseId?: string\n  agentId?: AgentId\n}): AsyncGenerator<\n  {\n    type: 'progress'\n    output: string\n    fullOutput: string\n    elapsedTimeSeconds: number\n    totalLines: number\n    totalBytes?: number\n    taskId?: string\n    timeoutMs?: number\n  },\n  ExecResult,\n  void\n> {\n  const { command, description, timeout, run_in_background } = input\n  const timeoutMs = timeout || getDefaultTimeoutMs()\n\n  let fullOutput = ''\n  let lastProgressOutput = ''\n  let lastTotalLines = 0\n  let lastTotalBytes = 0\n  let backgroundShellId: string | undefined = undefined\n  let assistantAutoBackgrounded = false\n\n  // Progress signal: resolved by onProgress callback from the shared poller,\n  // waking the generator to yield a progress update.\n  let resolveProgress: (() => void) | null = null\n  function createProgressSignal(): Promise<null> {\n    return new Promise<null>(resolve => {\n      resolveProgress = () => resolve(null)\n    })\n  }\n\n  // Determine if auto-backgrounding should be enabled\n  // Only enable for commands that are allowed to be auto-backgrounded\n  // and when background tasks are not disabled\n  const shouldAutoBackground =\n    !isBackgroundTasksDisabled && isAutobackgroundingAllowed(command)\n\n  const shellCommand = await exec(command, abortController.signal, 'bash', {\n    timeout: timeoutMs,\n    onProgress(lastLines, allLines, totalLines, totalBytes, isIncomplete) {\n      lastProgressOutput = lastLines\n      fullOutput = allLines\n      lastTotalLines = totalLines\n      lastTotalBytes = isIncomplete ? totalBytes : 0\n      // Wake the generator so it yields the new progress data\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n    },\n    preventCwdChanges,\n    shouldUseSandbox: shouldUseSandbox(input),\n    shouldAutoBackground,\n  })\n\n  // Start the command execution\n  const resultPromise = shellCommand.result\n\n  // Helper to spawn a background task and return its ID\n  async function spawnBackgroundTask(): Promise<string> {\n    const handle = await spawnShellTask(\n      {\n        command,\n        description: description || command,\n        shellCommand,\n        toolUseId,\n        agentId,\n      },\n      {\n        abortController,\n        getAppState: () => {\n          // We don't have direct access to getAppState here, but spawn doesn't\n          // actually use it during the spawn process\n          throw new Error(\n            'getAppState not available in runShellCommand context',\n          )\n        },\n        setAppState,\n      },\n    )\n    return handle.taskId\n  }\n\n  // Helper to start backgrounding with optional logging\n  function startBackgrounding(\n    eventName: string,\n    backgroundFn?: (shellId: string) => void,\n  ): void {\n    // If a foreground task is already registered (via registerForeground in the\n    // progress loop), background it in-place instead of re-spawning. Re-spawning\n    // would overwrite tasks[taskId], emit a duplicate task_started SDK event,\n    // and leak the first cleanup callback.\n    if (foregroundTaskId) {\n      if (\n        !backgroundExistingForegroundTask(\n          foregroundTaskId,\n          shellCommand,\n          description || command,\n          setAppState,\n          toolUseId,\n        )\n      ) {\n        return\n      }\n      backgroundShellId = foregroundTaskId\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n      backgroundFn?.(foregroundTaskId)\n      return\n    }\n\n    // No foreground task registered — spawn a new background task\n    // Note: spawn is essentially synchronous despite being async\n    void spawnBackgroundTask().then(shellId => {\n      backgroundShellId = shellId\n\n      // Wake the generator's Promise.race so it sees backgroundShellId.\n      // Without this, if the poller has stopped ticking for this task\n      // (no output + shared-poller race with sibling stopPolling calls)\n      // and the process is hung on I/O, the race at line ~1357 never\n      // resolves and the generator deadlocks despite being backgrounded.\n      const resolve = resolveProgress\n      if (resolve) {\n        resolveProgress = null\n        resolve()\n      }\n\n      logEvent(eventName, {\n        command_type: getCommandTypeForLogging(command),\n      })\n\n      if (backgroundFn) {\n        backgroundFn(shellId)\n      }\n    })\n  }\n\n  // Set up auto-backgrounding on timeout if enabled\n  // Only background commands that are allowed to be auto-backgrounded (not sleep, etc.)\n  if (shellCommand.onTimeout && shouldAutoBackground) {\n    shellCommand.onTimeout(backgroundFn => {\n      startBackgrounding(\n        'tengu_bash_command_timeout_backgrounded',\n        backgroundFn,\n      )\n    })\n  }\n\n  // In assistant mode, the main agent should stay responsive. Auto-background\n  // blocking commands after ASSISTANT_BLOCKING_BUDGET_MS so the agent can keep\n  // coordinating instead of waiting. The command keeps running — no state loss.\n  if (\n    feature('KAIROS') &&\n    getKairosActive() &&\n    isMainThread &&\n    !isBackgroundTasksDisabled &&\n    run_in_background !== true\n  ) {\n    setTimeout(() => {\n      if (\n        shellCommand.status === 'running' &&\n        backgroundShellId === undefined\n      ) {\n        assistantAutoBackgrounded = true\n        startBackgrounding('tengu_bash_command_assistant_auto_backgrounded')\n      }\n    }, ASSISTANT_BLOCKING_BUDGET_MS).unref()\n  }\n\n  // Handle Claude asking to run it in the background explicitly\n  // When explicitly requested via run_in_background, always honor the request\n  // regardless of the command type (isAutobackgroundingAllowed only applies to automatic backgrounding)\n  // Skip if background tasks are disabled - run in foreground instead\n  if (run_in_background === true && !isBackgroundTasksDisabled) {\n    const shellId = await spawnBackgroundTask()\n\n    logEvent('tengu_bash_command_explicitly_backgrounded', {\n      command_type: getCommandTypeForLogging(command),\n    })\n\n    return {\n      stdout: '',\n      stderr: '',\n      code: 0,\n      interrupted: false,\n      backgroundTaskId: shellId,\n    }\n  }\n\n  // Wait for the initial threshold before showing progress\n  const startTime = Date.now()\n  let foregroundTaskId: string | undefined = undefined\n\n  {\n    const initialResult = await Promise.race([\n      resultPromise,\n      new Promise<null>(resolve => {\n        const t = setTimeout(\n          (r: (v: null) => void) => r(null),\n          PROGRESS_THRESHOLD_MS,\n          resolve,\n        )\n        t.unref()\n      }),\n    ])\n\n    if (initialResult !== null) {\n      shellCommand.cleanup()\n      return initialResult\n    }\n\n    if (backgroundShellId) {\n      return {\n        stdout: '',\n        stderr: '',\n        code: 0,\n        interrupted: false,\n        backgroundTaskId: backgroundShellId,\n        assistantAutoBackgrounded,\n      }\n    }\n  }\n\n  // Start polling the output file for progress. The poller's #tick calls\n  // onProgress every second, which resolves progressSignal below.\n  TaskOutput.startPolling(shellCommand.taskOutput.taskId)\n\n  // Progress loop: wake is driven by the shared poller calling onProgress,\n  // which resolves the progressSignal.\n  try {\n    while (true) {\n      const progressSignal = createProgressSignal()\n      const result = await Promise.race([resultPromise, progressSignal])\n\n      if (result !== null) {\n        // Race: backgrounding fired (15s timer / onTimeout / Ctrl+B) but the\n        // command completed before the next poll tick. #handleExit sets\n        // backgroundTaskId but skips outputFilePath (it assumes the background\n        // message or <task_notification> will carry the path). Strip\n        // backgroundTaskId so the model sees a clean completed command,\n        // reconstruct outputFilePath for large outputs, and suppress the\n        // redundant <task_notification> from the .then() handler.\n        // Check result.backgroundTaskId (not the closure var) to also cover\n        // Ctrl+B, which calls shellCommand.background() directly.\n        if (result.backgroundTaskId !== undefined) {\n          markTaskNotified(result.backgroundTaskId, setAppState)\n          const fixedResult: ExecResult = {\n            ...result,\n            backgroundTaskId: undefined,\n          }\n          // Mirror ShellCommand.#handleExit's large-output branch that was\n          // skipped because #backgroundTaskId was set.\n          const { taskOutput } = shellCommand\n          if (taskOutput.stdoutToFile && !taskOutput.outputFileRedundant) {\n            fixedResult.outputFilePath = taskOutput.path\n            fixedResult.outputFileSize = taskOutput.outputFileSize\n            fixedResult.outputTaskId = taskOutput.taskId\n          }\n          shellCommand.cleanup()\n          return fixedResult\n        }\n        // Command has completed - return the actual result\n        // If we registered as a foreground task, unregister it\n        if (foregroundTaskId) {\n          unregisterForeground(foregroundTaskId, setAppState)\n        }\n        // Clean up stream resources for foreground commands\n        // (backgrounded commands are cleaned up by LocalShellTask)\n        shellCommand.cleanup()\n        return result\n      }\n\n      // Check if command was backgrounded (either via old mechanism or new backgroundAll)\n      if (backgroundShellId) {\n        return {\n          stdout: '',\n          stderr: '',\n          code: 0,\n          interrupted: false,\n          backgroundTaskId: backgroundShellId,\n          assistantAutoBackgrounded,\n        }\n      }\n\n      // Check if this foreground task was backgrounded via backgroundAll()\n      if (foregroundTaskId) {\n        // shellCommand.status becomes 'backgrounded' when background() is called\n        if (shellCommand.status === 'backgrounded') {\n          return {\n            stdout: '',\n            stderr: '',\n            code: 0,\n            interrupted: false,\n            backgroundTaskId: foregroundTaskId,\n            backgroundedByUser: true,\n          }\n        }\n      }\n\n      // Time for a progress update\n      const elapsed = Date.now() - startTime\n      const elapsedSeconds = Math.floor(elapsed / 1000)\n\n      // Show minimal backgrounding UI if available\n      // Skip if background tasks are disabled\n      if (\n        !isBackgroundTasksDisabled &&\n        backgroundShellId === undefined &&\n        elapsedSeconds >= PROGRESS_THRESHOLD_MS / 1000 &&\n        setToolJSX\n      ) {\n        // Register this command as a foreground task so it can be backgrounded via Ctrl+B\n        if (!foregroundTaskId) {\n          foregroundTaskId = registerForeground(\n            {\n              command,\n              description: description || command,\n              shellCommand,\n              agentId,\n            },\n            setAppState,\n            toolUseId,\n          )\n        }\n\n        setToolJSX({\n          jsx: <BackgroundHint />,\n          shouldHidePromptInput: false,\n          shouldContinueAnimation: true,\n          showSpinner: true,\n        })\n      }\n      yield {\n        type: 'progress',\n        fullOutput,\n        output: lastProgressOutput,\n        elapsedTimeSeconds: elapsedSeconds,\n        totalLines: lastTotalLines,\n        totalBytes: lastTotalBytes,\n        taskId: shellCommand.taskOutput.taskId,\n        ...(timeout ? { timeoutMs } : undefined),\n      }\n    }\n  } finally {\n    TaskOutput.stopPolling(shellCommand.taskOutput.taskId)\n  }\n}\n"],"mappings":"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","ignoreList":[]} diff --git a/src/tools/PowerShellTool/PowerShellTool.tsx b/src/tools/PowerShellTool/PowerShellTool.tsx index db35d2aa..1334f831 100644 --- a/src/tools/PowerShellTool/PowerShellTool.tsx +++ b/src/tools/PowerShellTool/PowerShellTool.tsx @@ -396,9 +396,13 @@ export const PowerShellTool = buildTool({ const block = buildImageToolResult(stdout, toolUseID); if (block) return block; } - let processedStdout = stdout; + const normalizedStdout = typeof stdout === 'string' ? stdout : ''; + const normalizedStderr = typeof stderr === 'string' ? stderr : ''; + let processedStdout = normalizedStdout; if (persistedOutputPath) { - const trimmed = stdout ? stdout.replace(/^(\s*\n)+/, '').trimEnd() : ''; + const trimmed = normalizedStdout + ? normalizedStdout.replace(/^(\s*\n)+/, '').trimEnd() + : ''; const preview = generatePreview(trimmed, PREVIEW_SIZE_BYTES); processedStdout = buildLargeToolResultMessage({ filepath: persistedOutputPath, @@ -407,13 +411,13 @@ export const PowerShellTool = buildTool({ preview: preview.preview, hasMore: preview.hasMore }); - } else if (stdout) { - processedStdout = stdout.replace(/^(\s*\n)+/, ''); + } else if (normalizedStdout) { + processedStdout = normalizedStdout.replace(/^(\s*\n)+/, ''); processedStdout = processedStdout.trimEnd(); } - let errorMessage = stderr.trim(); + let errorMessage = normalizedStderr.trim(); if (interrupted) { - if (stderr) errorMessage += EOL; + if (normalizedStderr) errorMessage += EOL; errorMessage += 'Command was aborted before completion'; } let backgroundInfo = ''; @@ -998,4 +1002,4 @@ async function* runPowerShellCommand({ } } } -//# 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":"AAAA,SAASA,OAAO,QAAQ,YAAY;AACpC,cAAcC,oBAAoB,QAAQ,uCAAuC;AACjF,SACEC,QAAQ,EACRC,IAAI,IAAIC,MAAM,EACdC,QAAQ,IAAIC,UAAU,EACtBC,IAAI,QACC,aAAa;AACpB,OAAO,KAAKC,KAAK,MAAM,OAAO;AAC9B,cAAcC,YAAY,QAAQ,4BAA4B;AAC9D,cAAcC,QAAQ,QAAQ,uBAAuB;AACrD,SAASC,CAAC,QAAQ,QAAQ;AAC1B,SAASC,eAAe,QAAQ,0BAA0B;AAC1D,SAASC,uBAAuB,QAAQ,+BAA+B;AACvE,SACE,KAAKC,0DAA0D,EAC/DC,QAAQ,QACH,mCAAmC;AAC1C,cACEC,YAAY,EACZC,IAAI,EACJC,gBAAgB,EAChBC,gBAAgB,QACX,eAAe;AACtB,SAASC,SAAS,EAAE,KAAKC,OAAO,QAAQ,eAAe;AACvD,SACEC,gCAAgC,EAChCC,gBAAgB,EAChBC,kBAAkB,EAClBC,cAAc,EACdC,oBAAoB,QACf,8CAA8C;AACrD,cAAcC,OAAO,QAAQ,oBAAoB;AACjD,cAAcC,gBAAgB,QAAQ,wBAAwB;AAC9D,SAASC,sBAAsB,QAAQ,gCAAgC;AACvE,SAASC,WAAW,QAAQ,yBAAyB;AACrD,SACEC,YAAY,IAAIC,eAAe,EAC/BC,UAAU,QACL,uBAAuB;AAC9B,SAAS5B,QAAQ,QAAQ,uBAAuB;AAChD,SAAS6B,UAAU,QAAQ,2BAA2B;AACtD,SAASC,QAAQ,QAAQ,oBAAoB;AAC7C,cAAcC,gBAAgB,QAAQ,6CAA6C;AACnF,SAASC,WAAW,QAAQ,yBAAyB;AACrD,SAASC,qBAAqB,QAAQ,2CAA2C;AACjF,SAASC,IAAI,QAAQ,sBAAsB;AAC3C,cAAcC,UAAU,QAAQ,6BAA6B;AAC7D,SAASC,cAAc,QAAQ,wCAAwC;AACvE,SAASC,eAAe,QAAQ,gCAAgC;AAChE,SAASC,cAAc,QAAQ,+BAA+B;AAC9D,SAASC,uBAAuB,QAAQ,0CAA0C;AAClF,SAASC,wBAAwB,QAAQ,4BAA4B;AACrE,SAASC,iBAAiB,QAAQ,gCAAgC;AAClE,SAASC,UAAU,QAAQ,gCAAgC;AAC3D,SAASC,qBAAqB,QAAQ,yBAAyB;AAC/D,SACEC,2BAA2B,EAC3BC,oBAAoB,EACpBC,eAAe,EACfC,iBAAiB,EACjBC,kBAAkB,QACb,kCAAkC;AACzC,SAASC,gBAAgB,QAAQ,iCAAiC;AAClE,SAASC,cAAc,QAAQ,mBAAmB;AAClD,SACEC,oBAAoB,EACpBC,aAAa,EACbC,wBAAwB,EACxBC,sBAAsB,EACtBC,6BAA6B,EAC7BC,eAAe,QACV,sBAAsB;AAC7B,SAASC,kBAAkB,QAAQ,mCAAmC;AACtE,SAASC,sBAAsB,QAAQ,uBAAuB;AAC9D,SAASC,2BAA2B,QAAQ,4BAA4B;AACxE,SAASC,mBAAmB,EAAEC,eAAe,EAAEC,SAAS,QAAQ,aAAa;AAC7E,SACEC,uBAAuB,EACvBC,iBAAiB,EACjBC,kBAAkB,QACb,yBAAyB;AAChC,SAASC,oBAAoB,QAAQ,eAAe;AACpD,SACEC,uBAAuB,EACvBC,yBAAyB,EACzBC,oBAAoB,EACpBC,4BAA4B,EAC5BC,0BAA0B,QACrB,SAAS;;AAEhB;AACA,MAAMC,GAAG,GAAG,IAAI;;AAEhB;AACA;AACA;AACA;AACA,MAAMC,kBAAkB,GAAG,IAAIC,GAAG,CAAC,CACjC,eAAe;AAAE;AACjB,eAAe;AAAE;AACjB,SAAS;AAAE;AACX,WAAW,CAAE;AAAA,CACd,CAAC;;AAEF;AACA;AACA;AACA;AACA,MAAMC,gBAAgB,GAAG,IAAID,GAAG,CAAC,CAC/B,aAAa;AAAE;AACf,UAAU;AAAE;AACZ,WAAW;AAAE;AACb,cAAc;AAAE;AAChB,aAAa;AAAE;AACf,aAAa;AAAE;AACf,eAAe;AAAE;AACjB,cAAc;AAAE;AAChB,cAAc;AAAE;AAChB,SAAS;AAAE;AACX,YAAY,CAAE;AAAA,CACf,CAAC;;AAEF;AACA;AACA;AACA,MAAME,4BAA4B,GAAG,IAAIF,GAAG,CAAC,CAC3C,cAAc;AAAE;AAChB,YAAY,CACb,CAAC;;AAEF;AACA;AACA;AACA;AACA,SAASG,+BAA+BA,CAACC,OAAO,EAAE,MAAM,CAAC,EAAE;EACzDC,QAAQ,EAAE,OAAO;EACjBC,MAAM,EAAE,OAAO;AACjB,CAAC,CAAC;EACA,MAAMC,OAAO,GAAGH,OAAO,CAACI,IAAI,CAAC,CAAC;EAC9B,IAAI,CAACD,OAAO,EAAE;IACZ,OAAO;MAAEF,QAAQ,EAAE,KAAK;MAAEC,MAAM,EAAE;IAAM,CAAC;EAC3C;;EAEA;EACA;EACA,MAAMG,KAAK,GAAGF,OAAO,CAACG,KAAK,CAAC,YAAY,CAAC,CAACC,MAAM,CAACC,OAAO,CAAC;EAEzD,IAAIH,KAAK,CAACI,MAAM,KAAK,CAAC,EAAE;IACtB,OAAO;MAAER,QAAQ,EAAE,KAAK;MAAEC,MAAM,EAAE;IAAM,CAAC;EAC3C;EAEA,IAAIQ,SAAS,GAAG,KAAK;EACrB,IAAIC,OAAO,GAAG,KAAK;EACnB,IAAIC,oBAAoB,GAAG,KAAK;EAEhC,KAAK,MAAMC,IAAI,IAAIR,KAAK,EAAE;IACxB,MAAMS,WAAW,GAAGD,IAAI,CAACT,IAAI,CAAC,CAAC,CAACE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC/C,IAAI,CAACQ,WAAW,EAAE;MAChB;IACF;IAEA,MAAMC,SAAS,GAAG5B,kBAAkB,CAAC2B,WAAW,CAAC;IAEjD,IAAIhB,4BAA4B,CAACkB,GAAG,CAACD,SAAS,CAAC,EAAE;MAC/C;IACF;IAEAH,oBAAoB,GAAG,IAAI;IAE3B,MAAMK,YAAY,GAAGtB,kBAAkB,CAACqB,GAAG,CAACD,SAAS,CAAC;IACtD,MAAMG,UAAU,GAAGrB,gBAAgB,CAACmB,GAAG,CAACD,SAAS,CAAC;IAElD,IAAI,CAACE,YAAY,IAAI,CAACC,UAAU,EAAE;MAChC,OAAO;QAAEjB,QAAQ,EAAE,KAAK;QAAEC,MAAM,EAAE;MAAM,CAAC;IAC3C;IAEA,IAAIe,YAAY,EAAEP,SAAS,GAAG,IAAI;IAClC,IAAIQ,UAAU,EAAEP,OAAO,GAAG,IAAI;EAChC;EAEA,IAAI,CAACC,oBAAoB,EAAE;IACzB,OAAO;MAAEX,QAAQ,EAAE,KAAK;MAAEC,MAAM,EAAE;IAAM,CAAC;EAC3C;EAEA,OAAO;IAAED,QAAQ,EAAES,SAAS;IAAER,MAAM,EAAES;EAAQ,CAAC;AACjD;;AAEA;AACA,MAAMQ,qBAAqB,GAAG,IAAI;AAClC,MAAMC,oBAAoB,GAAG,IAAI;AACjC;AACA,MAAMC,4BAA4B,GAAG,MAAM;;AAE3C;AACA;AACA;AACA,MAAMC,mCAAmC,GAAG,CAC1C,aAAa;AAAE;AACf,OAAO,CACR;;AAED;AACA;AACA;AACA;AACA;AACA,SAASC,0BAA0BA,CAACvB,OAAO,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC;EAC5D,MAAMwB,SAAS,GAAGxB,OAAO,CAACI,IAAI,CAAC,CAAC,CAACE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;EAChD,IAAI,CAACkB,SAAS,EAAE,OAAO,IAAI;EAC3B,MAAMT,SAAS,GAAG5B,kBAAkB,CAACqC,SAAS,CAAC;EAC/C,OAAO,CAACF,mCAAmC,CAACG,QAAQ,CAACV,SAAS,CAAC;AACjE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,SAASW,yBAAyBA,CAAC1B,OAAO,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;EACxE;EACA;EACA;EACA;EACA;EACA,MAAM2B,KAAK,GACT3B,OAAO,CACJI,IAAI,CAAC,CAAC,CACNE,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,EACpBF,IAAI,CAAC,CAAC,IAAI,EAAE;EAClB;EACA;EACA,MAAMwB,CAAC,GAAG,0DAA0D,CAACxE,IAAI,CACvEuE,KACF,CAAC;EACD,IAAI,CAACC,CAAC,EAAE,OAAO,IAAI;EACnB,MAAMC,IAAI,GAAGC,QAAQ,CAACF,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;EAChC,IAAIC,IAAI,GAAG,CAAC,EAAE,OAAO,IAAI,EAAC;;EAE1B,MAAME,IAAI,GAAG/B,OAAO,CACjBI,IAAI,CAAC,CAAC,CACN4B,KAAK,CAACL,KAAK,CAAClB,MAAM,CAAC,CACnBwB,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC;EAC3B,OAAOF,IAAI,GACP,eAAeF,IAAI,iBAAiBE,IAAI,EAAE,GAC1C,0BAA0BF,IAAI,EAAE;AACtC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMK,8BAA8B,GAClC,0JAA0J;AAC5J,SAASC,+BAA+BA,CAAA,CAAE,EAAE,OAAO,CAAC;EAClD,OACEjF,WAAW,CAAC,CAAC,KAAK,SAAS,IAC3BI,cAAc,CAAC8E,0BAA0B,CAAC,CAAC,IAC3C,CAAC9E,cAAc,CAAC+E,6BAA6B,CAAC,CAAC;AAEnD;;AAEA;AACA,MAAMC,yBAAyB;AAC7B;AACA3F,WAAW,CAAC4F,OAAO,CAACC,GAAG,CAACC,oCAAoC,CAAC;AAE/D,MAAMC,eAAe,GAAG3F,UAAU,CAAC,MACjCvB,CAAC,CAACmH,YAAY,CAAC;EACb3C,OAAO,EAAExE,CAAC,CAACoH,MAAM,CAAC,CAAC,CAACC,QAAQ,CAAC,mCAAmC,CAAC;EACjEC,OAAO,EAAEtF,cAAc,CAAChC,CAAC,CAACuH,MAAM,CAAC,CAAC,CAACC,QAAQ,CAAC,CAAC,CAAC,CAACH,QAAQ,CACrD,yCAAyC9D,eAAe,CAAC,CAAC,GAC5D,CAAC;EACDkE,WAAW,EAAEzH,CAAC,CACXoH,MAAM,CAAC,CAAC,CACRI,QAAQ,CAAC,CAAC,CACVH,QAAQ,CACP,uEACF,CAAC;EACHK,iBAAiB,EAAE3F,eAAe,CAAC/B,CAAC,CAAC2H,OAAO,CAAC,CAAC,CAACH,QAAQ,CAAC,CAAC,CAAC,CAACH,QAAQ,CACjE,uFACF,CAAC;EACDO,yBAAyB,EAAE7F,eAAe,CAAC/B,CAAC,CAAC2H,OAAO,CAAC,CAAC,CAACH,QAAQ,CAAC,CAAC,CAAC,CAACH,QAAQ,CACzE,4FACF;AACF,CAAC,CACH,CAAC;;AAED;AACA,MAAMQ,WAAW,GAAGtG,UAAU,CAAC,MAC7BuF,yBAAyB,GACrBI,eAAe,CAAC,CAAC,CAACY,IAAI,CAAC;EAAEJ,iBAAiB,EAAE;AAAK,CAAC,CAAC,GACnDR,eAAe,CAAC,CACtB,CAAC;AACD,KAAKa,WAAW,GAAGC,UAAU,CAAC,OAAOH,WAAW,CAAC;;AAEjD;AACA;AACA,OAAO,KAAKI,mBAAmB,GAAGjI,CAAC,CAACkI,KAAK,CAACF,UAAU,CAAC,OAAOd,eAAe,CAAC,CAAC;AAE7E,MAAMiB,YAAY,GAAG5G,UAAU,CAAC,MAC9BvB,CAAC,CAACoI,MAAM,CAAC;EACPC,MAAM,EAAErI,CAAC,CAACoH,MAAM,CAAC,CAAC,CAACC,QAAQ,CAAC,oCAAoC,CAAC;EACjEiB,MAAM,EAAEtI,CAAC,CAACoH,MAAM,CAAC,CAAC,CAACC,QAAQ,CAAC,0CAA0C,CAAC;EACvEkB,WAAW,EAAEvI,CAAC,CAAC2H,OAAO,CAAC,CAAC,CAACN,QAAQ,CAAC,qCAAqC,CAAC;EACxEmB,wBAAwB,EAAExI,CAAC,CACxBoH,MAAM,CAAC,CAAC,CACRI,QAAQ,CAAC,CAAC,CACVH,QAAQ,CACP,uEACF,CAAC;EACHoB,OAAO,EAAEzI,CAAC,CACP2H,OAAO,CAAC,CAAC,CACTH,QAAQ,CAAC,CAAC,CACVH,QAAQ,CAAC,gDAAgD,CAAC;EAC7DqB,mBAAmB,EAAE1I,CAAC,CACnBoH,MAAM,CAAC,CAAC,CACRI,QAAQ,CAAC,CAAC,CACVH,QAAQ,CAAC,yDAAyD,CAAC;EACtEsB,mBAAmB,EAAE3I,CAAC,CACnBuH,MAAM,CAAC,CAAC,CACRC,QAAQ,CAAC,CAAC,CACVH,QAAQ,CAAC,2CAA2C,CAAC;EACxDuB,gBAAgB,EAAE5I,CAAC,CAChBoH,MAAM,CAAC,CAAC,CACRI,QAAQ,CAAC,CAAC,CACVH,QAAQ,CACP,+DACF,CAAC;EACHwB,kBAAkB,EAAE7I,CAAC,CAClB2H,OAAO,CAAC,CAAC,CACTH,QAAQ,CAAC,CAAC,CACVH,QAAQ,CACP,gEACF,CAAC;EACHyB,yBAAyB,EAAE9I,CAAC,CACzB2H,OAAO,CAAC,CAAC,CACTH,QAAQ,CAAC,CAAC,CACVH,QAAQ,CACP,iFACF;AACJ,CAAC,CACH,CAAC;AACD,KAAK0B,YAAY,GAAGf,UAAU,CAAC,OAAOG,YAAY,CAAC;AACnD,OAAO,KAAKa,GAAG,GAAGhJ,CAAC,CAACkI,KAAK,CAACa,YAAY,CAAC;AAEvC,cAAcE,kBAAkB,QAAQ,sBAAsB;AAE9D,cAAcA,kBAAkB,QAAQ,sBAAsB;AAE9D,MAAMC,0BAA0B,GAAG,CACjC,KAAK,EACL,MAAM,EACN,MAAM,EACN,MAAM,EACN,QAAQ,EACR,SAAS,EACT,IAAI,EACJ,OAAO,EACP,MAAM,EACN,QAAQ,EACR,WAAW,EACX,SAAS,EACT,MAAM,EACN,MAAM,EACN,QAAQ,EACR,MAAM,EACN,mBAAmB,EACnB,OAAO,EACP,MAAM,EACN,OAAO,EACP,OAAO,EACP,KAAK,CACN,IAAIC,KAAK;AAEV,SAASC,wBAAwBA,CAC/B5E,OAAO,EAAE,MAAM,CAChB,EAAErE,0DAA0D,CAAC;EAC5D,MAAMwE,OAAO,GAAGH,OAAO,CAACI,IAAI,CAAC,CAAC;EAC9B,MAAMoB,SAAS,GAAGrB,OAAO,CAACG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE;EAE/C,KAAK,MAAMuE,GAAG,IAAIH,0BAA0B,EAAE;IAC5C,IAAIlD,SAAS,CAACsD,WAAW,CAAC,CAAC,KAAKD,GAAG,CAACC,WAAW,CAAC,CAAC,EAAE;MACjD,OAAOD,GAAG,IAAIlJ,0DAA0D;IAC1E;EACF;EAEA,OAAO,OAAO,IAAIA,0DAA0D;AAC9E;AAEA,OAAO,MAAMoJ,cAAc,GAAG9I,SAAS,CAAC;EACtC+I,IAAI,EAAE5F,oBAAoB;EAC1B6F,UAAU,EAAE,qCAAqC;EACjDC,kBAAkB,EAAE,MAAM;EAC1BC,MAAM,EAAE,IAAI;EAEZ,MAAMlC,WAAWA,CAAC;IAChBA;EAC4B,CAA7B,EAAEmC,OAAO,CAAC3B,mBAAmB,CAAC,CAAC,EAAE4B,OAAO,CAAC,MAAM,CAAC,CAAC;IAChD,OAAOpC,WAAW,IAAI,wBAAwB;EAChD,CAAC;EAED,MAAMqC,MAAMA,CAAA,CAAE,EAAED,OAAO,CAAC,MAAM,CAAC,CAAC;IAC9B,OAAOrG,SAAS,CAAC,CAAC;EACpB,CAAC;EAEDuG,iBAAiBA,CAACC,KAAK,EAAE/B,mBAAmB,CAAC,EAAE,OAAO,CAAC;IACrD,OAAO,IAAI,CAACgC,UAAU,GAAGD,KAAK,CAAC,IAAI,KAAK;EAC1C,CAAC;EAEDE,qBAAqBA,CAACF,KAAK,EAAEJ,OAAO,CAAC3B,mBAAmB,CAAC,CAAC,EAAE;IAC1DxD,QAAQ,EAAE,OAAO;IACjBC,MAAM,EAAE,OAAO;EACjB,CAAC,CAAC;IACA,IAAI,CAACsF,KAAK,CAACxF,OAAO,EAAE;MAClB,OAAO;QAAEC,QAAQ,EAAE,KAAK;QAAEC,MAAM,EAAE;MAAM,CAAC;IAC3C;IACA,OAAOH,+BAA+B,CAACyF,KAAK,CAACxF,OAAO,CAAC;EACvD,CAAC;EAEDyF,UAAUA,CAACD,KAAK,EAAE/B,mBAAmB,CAAC,EAAE,OAAO,CAAC;IAC9C;IACA;IACA;IACA;IACA;IACA,IAAIxE,uBAAuB,CAACuG,KAAK,CAACxF,OAAO,CAAC,EAAE;MAC1C,OAAO,KAAK;IACd;IACA;IACA;IACA;IACA;IACA;IACA;IACA,OAAOd,iBAAiB,CAACsG,KAAK,CAACxF,OAAO,CAAC;EACzC,CAAC;EACD2F,qBAAqBA,CAACH,KAAK,EAAE;IAC3B,OAAOA,KAAK,CAACxF,OAAO;EACtB,CAAC;EAED,IAAIqD,WAAWA,CAAA,CAAE,EAAEE,WAAW,CAAC;IAC7B,OAAOF,WAAW,CAAC,CAAC;EACtB,CAAC;EAED,IAAIM,YAAYA,CAAA,CAAE,EAAEY,YAAY,CAAC;IAC/B,OAAOZ,YAAY,CAAC,CAAC;EACvB,CAAC;EAEDiC,cAAcA,CAAA,CAAE,EAAE,MAAM,CAAC;IACvB,OAAO,YAAY;EACrB,CAAC;EAEDC,iBAAiBA,CACfL,KAAK,EAAEJ,OAAO,CAAC3B,mBAAmB,CAAC,GAAG,SAAS,CAChD,EAAE,MAAM,GAAG,IAAI,CAAC;IACf,IAAI,CAAC+B,KAAK,EAAExF,OAAO,EAAE;MACnB,OAAO,IAAI;IACb;IACA,MAAM;MAAEA,OAAO;MAAEiD;IAAY,CAAC,GAAGuC,KAAK;IACtC,IAAIvC,WAAW,EAAE;MACf,OAAOA,WAAW;IACpB;IACA,OAAO/H,QAAQ,CAAC8E,OAAO,EAAEtE,uBAAuB,CAAC;EACnD,CAAC;EAEDoK,sBAAsBA,CACpBN,KAAK,EAAEJ,OAAO,CAAC3B,mBAAmB,CAAC,GAAG,SAAS,CAChD,EAAE,MAAM,CAAC;IACR,IAAI,CAAC+B,KAAK,EAAExF,OAAO,EAAE;MACnB,OAAO,iBAAiB;IAC1B;IACA,MAAM+F,IAAI,GACRP,KAAK,CAACvC,WAAW,IAAI/H,QAAQ,CAACsK,KAAK,CAACxF,OAAO,EAAEtE,uBAAuB,CAAC;IACvE,OAAO,WAAWqK,IAAI,EAAE;EAC1B,CAAC;EAEDC,SAASA,CAAA,CAAE,EAAE,OAAO,CAAC;IACnB,OAAO,IAAI;EACb,CAAC;EAED,MAAMC,aAAaA,CAACT,KAAK,EAAE/B,mBAAmB,CAAC,EAAE4B,OAAO,CAACrJ,gBAAgB,CAAC,CAAC;IACzE;IACA,IAAImG,+BAA+B,CAAC,CAAC,EAAE;MACrC,OAAO;QACL+D,MAAM,EAAE,KAAK;QACbC,OAAO,EAAEjE,8BAA8B;QACvCkE,SAAS,EAAE;MACb,CAAC;IACH;IACA,IACEvL,OAAO,CAAC,cAAc,CAAC,IACvB,CAACyH,yBAAyB,IAC1B,CAACkD,KAAK,CAACtC,iBAAiB,EACxB;MACA,MAAMmD,YAAY,GAAG3E,yBAAyB,CAAC8D,KAAK,CAACxF,OAAO,CAAC;MAC7D,IAAIqG,YAAY,KAAK,IAAI,EAAE;QACzB,OAAO;UACLH,MAAM,EAAE,KAAK;UACbC,OAAO,EAAE,YAAYE,YAAY,+RAA+R;UAChUD,SAAS,EAAE;QACb,CAAC;MACH;IACF;IACA,OAAO;MAAEF,MAAM,EAAE;IAAK,CAAC;EACzB,CAAC;EAED,MAAMI,gBAAgBA,CACpBd,KAAK,EAAE/B,mBAAmB,EAC1B8C,OAAO,EAAEC,UAAU,CAAC1K,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,CAAC,CACjD,EAAEuJ,OAAO,CAACpI,gBAAgB,CAAC,CAAC;IAC3B,OAAO,MAAM4B,2BAA2B,CAAC2G,KAAK,EAAEe,OAAO,CAAC;EAC1D,CAAC;EAEDhH,oBAAoB;EACpBC,4BAA4B;EAC5BC,0BAA0B;EAC1BJ,uBAAuB;EACvBC,yBAAyB;EAEzBmH,mCAAmCA,CACjC;IACE1C,WAAW;IACXF,MAAM;IACNC,MAAM;IACNG,OAAO;IACPC,mBAAmB;IACnBC,mBAAmB;IACnBC,gBAAgB;IAChBC,kBAAkB;IAClBC;EACG,CAAJ,EAAEE,GAAG,EACNkC,SAAS,EAAE,MAAM,CAClB,EAAE5L,oBAAoB,CAAC;IACtB;IACA,IAAImJ,OAAO,EAAE;MACX,MAAM0C,KAAK,GAAGtI,oBAAoB,CAACwF,MAAM,EAAE6C,SAAS,CAAC;MACrD,IAAIC,KAAK,EAAE,OAAOA,KAAK;IACzB;IAEA,IAAIC,eAAe,GAAG/C,MAAM;IAE5B,IAAIK,mBAAmB,EAAE;MACvB,MAAM/D,OAAO,GAAG0D,MAAM,GAAGA,MAAM,CAAC5B,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC4E,OAAO,CAAC,CAAC,GAAG,EAAE;MACvE,MAAMC,OAAO,GAAG9I,eAAe,CAACmC,OAAO,EAAEjC,kBAAkB,CAAC;MAC5D0I,eAAe,GAAG9I,2BAA2B,CAAC;QAC5CiJ,QAAQ,EAAE7C,mBAAmB;QAC7B8C,YAAY,EAAE7C,mBAAmB,IAAI,CAAC;QACtC8C,MAAM,EAAE,KAAK;QACbH,OAAO,EAAEA,OAAO,CAACA,OAAO;QACxBI,OAAO,EAAEJ,OAAO,CAACI;MACnB,CAAC,CAAC;IACJ,CAAC,MAAM,IAAIrD,MAAM,EAAE;MACjB+C,eAAe,GAAG/C,MAAM,CAAC5B,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC;MACjD2E,eAAe,GAAGA,eAAe,CAACC,OAAO,CAAC,CAAC;IAC7C;IAEA,IAAIjK,YAAY,GAAGkH,MAAM,CAAC1D,IAAI,CAAC,CAAC;IAChC,IAAI2D,WAAW,EAAE;MACf,IAAID,MAAM,EAAElH,YAAY,IAAI8C,GAAG;MAC/B9C,YAAY,IAAI,sDAAsD;IACxE;IAEA,IAAIuK,cAAc,GAAG,EAAE;IACvB,IAAI/C,gBAAgB,EAAE;MACpB,MAAMgD,UAAU,GAAGzJ,iBAAiB,CAACyG,gBAAgB,CAAC;MACtD,IAAIE,yBAAyB,EAAE;QAC7B6C,cAAc,GAAG,wDAAwD9F,4BAA4B,GAAG,IAAI,+CAA+C+C,gBAAgB,+FAA+FgD,UAAU,8HAA8H;MACpZ,CAAC,MAAM,IAAI/C,kBAAkB,EAAE;QAC7B8C,cAAc,GAAG,sDAAsD/C,gBAAgB,iCAAiCgD,UAAU,EAAE;MACtI,CAAC,MAAM;QACLD,cAAc,GAAG,0CAA0C/C,gBAAgB,iCAAiCgD,UAAU,EAAE;MAC1H;IACF;IAEA,OAAO;MACLC,WAAW,EAAEX,SAAS;MACtBY,IAAI,EAAE,aAAa,IAAI3C,KAAK;MAC5B4C,OAAO,EAAE,CAACX,eAAe,EAAEhK,YAAY,EAAEuK,cAAc,CAAC,CACrD5G,MAAM,CAACC,OAAO,CAAC,CACfgH,IAAI,CAAC,IAAI,CAAC;MACbC,QAAQ,EAAE1D;IACZ,CAAC;EACH,CAAC;EAED,MAAM2D,IAAIA,CACRlC,KAAK,EAAE/B,mBAAmB,EAC1BkE,cAAc,EAAEnB,UAAU,CAAC1K,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,EAC3C8L,WAA0B,CAAd,EAAEtM,YAAY,EAC1BuM,cAAiC,CAAlB,EAAEpL,gBAAgB,EACjCqL,UAAiD,CAAtC,EAAE/L,gBAAgB,CAAC0I,kBAAkB,CAAC,CAClD,EAAEY,OAAO,CAAC;IAAE0C,IAAI,EAAEvD,GAAG;EAAC,CAAC,CAAC,CAAC;IACxB;IACA;IACA;IACA;IACA,IAAIrC,+BAA+B,CAAC,CAAC,EAAE;MACrC,MAAM,IAAI6F,KAAK,CAAC9F,8BAA8B,CAAC;IACjD;IAEA,MAAM;MAAE+F,eAAe;MAAEC,WAAW;MAAEC;IAAW,CAAC,GAAGR,cAAc;IAEnE,MAAMS,YAAY,GAAG,CAACT,cAAc,CAACU,OAAO;IAE5C,IAAIC,eAAe,GAAG,CAAC;IAEvB,IAAI;MACF,MAAMC,gBAAgB,GAAGC,oBAAoB,CAAC;QAC5ChD,KAAK;QACLyC,eAAe;QACf;QACA;QACAC,WAAW,EAAEP,cAAc,CAACc,mBAAmB,IAAIP,WAAW;QAC9DC,UAAU;QACVO,iBAAiB,EAAE,CAACN,YAAY;QAChCA,YAAY;QACZO,SAAS,EAAEhB,cAAc,CAACgB,SAAS;QACnCN,OAAO,EAAEV,cAAc,CAACU;MAC1B,CAAC,CAAC;MAEF,IAAIO,eAAe;MACnB,GAAG;QACDA,eAAe,GAAG,MAAML,gBAAgB,CAACM,IAAI,CAAC,CAAC;QAC/C,IAAI,CAACD,eAAe,CAACE,IAAI,IAAIhB,UAAU,EAAE;UACvC,MAAMiB,QAAQ,GAAGH,eAAe,CAACI,KAAK;UACtClB,UAAU,CAAC;YACTpB,SAAS,EAAE,eAAe4B,eAAe,EAAE,EAAE;YAC7CP,IAAI,EAAE;cACJT,IAAI,EAAE,qBAAqB;cAC3B2B,MAAM,EAAEF,QAAQ,CAACE,MAAM;cACvBC,UAAU,EAAEH,QAAQ,CAACG,UAAU;cAC/BC,kBAAkB,EAAEJ,QAAQ,CAACI,kBAAkB;cAC/CC,UAAU,EAAEL,QAAQ,CAACK,UAAU;cAC/BC,UAAU,EAAEN,QAAQ,CAACM,UAAU;cAC/BC,SAAS,EAAEP,QAAQ,CAACO,SAAS;cAC7BC,MAAM,EAAER,QAAQ,CAACQ;YACnB;UACF,CAAC,CAAC;QACJ;MACF,CAAC,QAAQ,CAACX,eAAe,CAACE,IAAI;MAE9B,MAAM5C,MAAM,GAAG0C,eAAe,CAACI,KAAK;;MAEpC;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA,MAAMQ,mBAAmB,GACvBtD,MAAM,CAACuD,IAAI,KAAK,CAAC,IACjB,CAACvD,MAAM,CAACrC,MAAM,IACdqC,MAAM,CAACpC,MAAM,IACb,CAACoC,MAAM,CAAC9B,gBAAgB;MAC1B,IAAI,CAACoF,mBAAmB,EAAE;QACxB7K,kBAAkB,CAAC6G,KAAK,CAACxF,OAAO,EAAEkG,MAAM,CAACuD,IAAI,EAAEvD,MAAM,CAACrC,MAAM,CAAC;MAC/D;;MAEA;MACA;MACA;MACA;MACA,MAAM6F,WAAW,GACfxD,MAAM,CAACnC,WAAW,IAAIkE,eAAe,CAAC0B,MAAM,CAACC,MAAM,KAAK,WAAW;;MAErE;MACA;MACA;MACA;MACA;MACA;MACA,IAAIC,mBAAmB,GAAG,EAAE;MAC5B,IAAIzB,YAAY,EAAE;QAChB,MAAM0B,QAAQ,GAAGnC,cAAc,CAACoC,WAAW,CAAC,CAAC;QAC7C,IAAIxL,wBAAwB,CAACuL,QAAQ,CAACE,qBAAqB,CAAC,EAAE;UAC5DH,mBAAmB,GAAGpL,6BAA6B,CAAC,EAAE,CAAC;QACzD;MACF;;MAEA;MACA;MACA;MACA;MACA,IAAIyH,MAAM,CAAC9B,gBAAgB,EAAE;QAC3B,MAAM6F,WAAW,GAAGvN,sBAAsB,CACxCwJ,MAAM,CAACrC,MAAM,IAAI,EAAE,EACnB2B,KAAK,CAACxF,OACR,CAAC;QACD,IAAIoI,YAAY,IAAI6B,WAAW,CAACC,KAAK,CAACzJ,MAAM,GAAG,CAAC,EAAE;UAChD,KAAK,MAAM0J,IAAI,IAAIF,WAAW,CAACC,KAAK,EAAE/M,qBAAqB,CAACgN,IAAI,CAAC;QACnE;QACA,OAAO;UACLpC,IAAI,EAAE;YACJlE,MAAM,EAAEoG,WAAW,CAACG,QAAQ;YAC5BtG,MAAM,EAAE,CAACoC,MAAM,CAACpC,MAAM,IAAI,EAAE,EAAE+F,mBAAmB,CAAC,CAC/CtJ,MAAM,CAACC,OAAO,CAAC,CACfgH,IAAI,CAAC,IAAI,CAAC;YACbzD,WAAW,EAAE,KAAK;YAClBK,gBAAgB,EAAE8B,MAAM,CAAC9B,gBAAgB;YACzCC,kBAAkB,EAAE6B,MAAM,CAAC7B,kBAAkB;YAC7CC,yBAAyB,EAAE4B,MAAM,CAAC5B;UACpC;QACF,CAAC;MACH;MAEA,MAAM+F,iBAAiB,GAAG,IAAI3M,wBAAwB,CAAC,CAAC;MACxD,MAAMkJ,eAAe,GAAG,CAACV,MAAM,CAACrC,MAAM,IAAI,EAAE,EAAEgD,OAAO,CAAC,CAAC;MAEvDwD,iBAAiB,CAACC,MAAM,CAAC1D,eAAe,GAAGlH,GAAG,CAAC;;MAE/C;MACA;MACA;MACA;MACA,MAAM6K,cAAc,GAAG3L,sBAAsB,CAC3C4G,KAAK,CAACxF,OAAO,EACbkG,MAAM,CAACuD,IAAI,EACX7C,eAAe,EACfV,MAAM,CAACpC,MAAM,IAAI,EACnB,CAAC;;MAED;MACA;MACA;MACA;;MAEA,IAAID,MAAM,GAAGnF,eAAe,CAAC2L,iBAAiB,CAACG,QAAQ,CAAC,CAAC,CAAC;;MAE1D;MACA;MACA;MACA;MACA;MACA;MACA,MAAMC,SAAS,GAAG/N,sBAAsB,CAACmH,MAAM,EAAE2B,KAAK,CAACxF,OAAO,CAAC;MAC/D6D,MAAM,GAAG4G,SAAS,CAACL,QAAQ;MAC3B,IAAIhC,YAAY,IAAIqC,SAAS,CAACP,KAAK,CAACzJ,MAAM,GAAG,CAAC,EAAE;QAC9C,KAAK,MAAM0J,IAAI,IAAIM,SAAS,CAACP,KAAK,EAAE/M,qBAAqB,CAACgN,IAAI,CAAC;MACjE;;MAEA;MACA;MACA;MACA;MACA,IAAIjE,MAAM,CAACwE,aAAa,EAAE;QACxB,MAAM,IAAI1C,KAAK,CAAC9B,MAAM,CAACwE,aAAa,CAAC;MACvC;MACA,IAAIH,cAAc,CAACI,OAAO,IAAI,CAACjB,WAAW,EAAE;QAC1C,MAAM,IAAI5M,UAAU,CAClB+G,MAAM,EACNqC,MAAM,CAACpC,MAAM,IAAI,EAAE,EACnBoC,MAAM,CAACuD,IAAI,EACXvD,MAAM,CAACnC,WACT,CAAC;MACH;;MAEA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA;MACA,MAAM6G,kBAAkB,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI;MAC3C,IAAI1G,mBAAmB,EAAE,MAAM,GAAG,SAAS;MAC3C,IAAIC,mBAAmB,EAAE,MAAM,GAAG,SAAS;MAC3C,IAAI+B,MAAM,CAAC2E,cAAc,IAAI3E,MAAM,CAAC4E,YAAY,EAAE;QAChD,IAAI;UACF,MAAMC,QAAQ,GAAG,MAAM9P,MAAM,CAACiL,MAAM,CAAC2E,cAAc,CAAC;UACpD1G,mBAAmB,GAAG4G,QAAQ,CAACC,IAAI;UAEnC,MAAMjN,oBAAoB,CAAC,CAAC;UAC5B,MAAMkN,IAAI,GAAGhN,iBAAiB,CAACiI,MAAM,CAAC4E,YAAY,EAAE,KAAK,CAAC;UAC1D,IAAIC,QAAQ,CAACC,IAAI,GAAGJ,kBAAkB,EAAE;YACtC,MAAMzP,UAAU,CAAC+K,MAAM,CAAC2E,cAAc,EAAED,kBAAkB,CAAC;UAC7D;UACA,IAAI;YACF,MAAMxP,IAAI,CAAC8K,MAAM,CAAC2E,cAAc,EAAEI,IAAI,CAAC;UACzC,CAAC,CAAC,MAAM;YACN,MAAMlQ,QAAQ,CAACmL,MAAM,CAAC2E,cAAc,EAAEI,IAAI,CAAC;UAC7C;UACA/G,mBAAmB,GAAG+G,IAAI;QAC5B,CAAC,CAAC,MAAM;UACN;QAAA;MAEJ;;MAEA;MACA;MACA;MACA,IAAIhH,OAAO,GAAG3F,aAAa,CAACuF,MAAM,CAAC;MACnC,IAAIqH,gBAAgB,GAAGrH,MAAM;MAC7B,IAAII,OAAO,EAAE;QACX,MAAMkH,OAAO,GAAG,MAAM3M,sBAAsB,CAC1CqF,MAAM,EACNqC,MAAM,CAAC2E,cAAc,EACrB1G,mBACF,CAAC;QACD,IAAIgH,OAAO,EAAE;UACXD,gBAAgB,GAAGC,OAAO;QAC5B,CAAC,MAAM;UACL;UACA;UACA;UACA;UACAlH,OAAO,GAAG,KAAK;QACjB;MACF;MAEA,MAAMmH,WAAW,GAAG,CAAClF,MAAM,CAACpC,MAAM,IAAI,EAAE,EAAE+F,mBAAmB,CAAC,CAC3DtJ,MAAM,CAACC,OAAO,CAAC,CACfgH,IAAI,CAAC,IAAI,CAAC;MAEb5L,QAAQ,CAAC,wCAAwC,EAAE;QACjDyP,YAAY,EAAEzG,wBAAwB,CAACY,KAAK,CAACxF,OAAO,CAAC;QACrDsL,aAAa,EAAEJ,gBAAgB,CAACzK,MAAM;QACtC8K,aAAa,EAAEH,WAAW,CAAC3K,MAAM;QACjC+K,SAAS,EAAEtF,MAAM,CAACuD,IAAI;QACtB1F,WAAW,EAAEmC,MAAM,CAACnC;MACtB,CAAC,CAAC;MAEF,OAAO;QACLgE,IAAI,EAAE;UACJlE,MAAM,EAAEqH,gBAAgB;UACxBpH,MAAM,EAAEsH,WAAW;UACnBrH,WAAW,EAAEmC,MAAM,CAACnC,WAAW;UAC/BC,wBAAwB,EAAEuG,cAAc,CAACpE,OAAO;UAChDlC,OAAO;UACPC,mBAAmB;UACnBC;QACF;MACF,CAAC;IACH,CAAC,SAAS;MACR,IAAIgE,UAAU,EAAEA,UAAU,CAAC,IAAI,CAAC;IAClC;EACF,CAAC;EACDsD,iBAAiBA,CAACxC,MAAM,EAAEzE,GAAG,CAAC,EAAE,OAAO,CAAC;IACtC,OACE3G,qBAAqB,CAACoL,MAAM,CAACpF,MAAM,CAAC,IACpChG,qBAAqB,CAACoL,MAAM,CAACnF,MAAM,CAAC;EAExC;AACF,CAAC,WAAW5H,OAAO,CAACqH,WAAW,EAAEiB,GAAG,CAAC,CAAC;AAEtC,gBAAgBgE,oBAAoBA,CAAC;EACnChD,KAAK;EACLyC,eAAe;EACfC,WAAW;EACXC,UAAU;EACVO,iBAAiB;EACjBN,YAAY;EACZO,SAAS;EACTN;AAUF,CATC,EAAE;EACD7C,KAAK,EAAE/B,mBAAmB;EAC1BwE,eAAe,EAAEyD,eAAe;EAChCxD,WAAW,EAAE,CAACyD,CAAC,EAAE,CAACC,IAAI,EAAErQ,QAAQ,EAAE,GAAGA,QAAQ,EAAE,GAAG,IAAI;EACtD4M,UAAU,CAAC,EAAEtM,YAAY;EACzB6M,iBAAiB,CAAC,EAAE,OAAO;EAC3BN,YAAY,CAAC,EAAE,OAAO;EACtBO,SAAS,CAAC,EAAE,MAAM;EAClBN,OAAO,CAAC,EAAE7L,OAAO;AACnB,CAAC,CAAC,EAAEqP,cAAc,CAChB;EACEvE,IAAI,EAAE,UAAU;EAChB2B,MAAM,EAAE,MAAM;EACdC,UAAU,EAAE,MAAM;EAClBC,kBAAkB,EAAE,MAAM;EAC1BC,UAAU,EAAE,MAAM;EAClBC,UAAU,EAAE,MAAM;EAClBE,MAAM,CAAC,EAAE,MAAM;EACfD,SAAS,CAAC,EAAE,MAAM;AACpB,CAAC,EACDjM,UAAU,EACV,IAAI,CACL,CAAC;EACA,MAAM;IACJ2C,OAAO;IACPiD,WAAW;IACXH,OAAO;IACPI,iBAAiB;IACjBE;EACF,CAAC,GAAGoC,KAAK;EACT,MAAM8D,SAAS,GAAGwC,IAAI,CAACC,GAAG,CACxBjJ,OAAO,IAAIhE,mBAAmB,CAAC,CAAC,EAChCC,eAAe,CAAC,CAClB,CAAC;EAED,IAAImK,UAAU,GAAG,EAAE;EACnB,IAAI8C,kBAAkB,GAAG,EAAE;EAC3B,IAAIC,cAAc,GAAG,CAAC;EACtB,IAAIC,cAAc,GAAG,CAAC;EACtB,IAAIC,iBAAiB,EAAE,MAAM,GAAG,SAAS,GAAGC,SAAS;EACrD,IAAIC,6BAA6B,GAAG,KAAK;EACzC,IAAI/H,yBAAyB,GAAG,KAAK;;EAErC;EACA;EACA;EACA,IAAIgI,eAAe,EAAE,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,IAAI,GAAG,IAAI;EAC/C,SAASC,oBAAoBA,CAAA,CAAE,EAAElH,OAAO,CAAC,IAAI,CAAC,CAAC;IAC7C,OAAO,IAAIA,OAAO,CAAC,IAAI,CAAC,CAACmH,OAAO,IAAI;MAClCF,eAAe,GAAGA,CAAA,KAAME,OAAO,CAAC,IAAI,CAAC;IACvC,CAAC,CAAC;EACJ;EAEA,MAAMC,oBAAoB,GACxB,CAACnK,yBAAyB,IAAIf,0BAA0B,CAACvB,OAAO,CAAC;EAEnE,MAAM0M,cAAc,GAAG,MAAMjP,uBAAuB,CAAC,CAAC;EACtD,IAAI,CAACiP,cAAc,EAAE;IACnB;IACA;IACA;IACA,OAAO;MACL7I,MAAM,EAAE,EAAE;MACVC,MAAM,EAAE,6CAA6C;MACrD2F,IAAI,EAAE,CAAC;MACP1F,WAAW,EAAE;IACf,CAAC;EACH;EAEA,IAAI4I,YAAY,EAAEC,OAAO,CAACpJ,UAAU,CAAC,OAAOpG,IAAI,CAAC,CAAC;EAClD,IAAI;IACFuP,YAAY,GAAG,MAAMvP,IAAI,CAAC4C,OAAO,EAAEiI,eAAe,CAAC0B,MAAM,EAAE,YAAY,EAAE;MACvE7G,OAAO,EAAEwG,SAAS;MAClBxB,UAAUA,CAAC+E,SAAS,EAAEC,QAAQ,EAAE1D,UAAU,EAAEC,UAAU,EAAE0D,YAAY,EAAE;QACpEf,kBAAkB,GAAGa,SAAS;QAC9B3D,UAAU,GAAG4D,QAAQ;QACrBb,cAAc,GAAG7C,UAAU;QAC3B8C,cAAc,GAAGa,YAAY,GAAG1D,UAAU,GAAG,CAAC;MAChD,CAAC;MACDX,iBAAiB;MACjB;MACA;MACA;MACA;MACA;MACA;MACAvK,gBAAgB,EACdjB,WAAW,CAAC,CAAC,KAAK,SAAS,GACvB,KAAK,GACLiB,gBAAgB,CAAC;QAAE6B,OAAO;QAAEoD;MAA0B,CAAC,CAAC;MAC9DqJ;IACF,CAAC,CAAC;EACJ,CAAC,CAAC,OAAOO,CAAC,EAAE;IACVhQ,QAAQ,CAACgQ,CAAC,CAAC;IACX;IACA;IACA,OAAO;MACLnJ,MAAM,EAAE,EAAE;MACVC,MAAM,EAAE,yCAAyCjH,eAAe,CAACmQ,CAAC,CAAC,EAAE;MACrEvD,IAAI,EAAE,CAAC;MACP1F,WAAW,EAAE;IACf,CAAC;EACH;EAEA,MAAMkJ,aAAa,GAAGN,YAAY,CAACzG,MAAM;;EAEzC;EACA,eAAegH,mBAAmBA,CAAA,CAAE,EAAE7H,OAAO,CAAC,MAAM,CAAC,CAAC;IACpD,MAAM8H,MAAM,GAAG,MAAM7Q,cAAc,CACjC;MACE0D,OAAO;MACPiD,WAAW,EAAEA,WAAW,IAAIjD,OAAO;MACnC2M,YAAY;MACZhE,SAAS;MACTN;IACF,CAAC,EACD;MACEJ,eAAe;MACf8B,WAAW,EAAEA,CAAA,KAAM;QACjB,MAAM,IAAI/B,KAAK,CACb,2DACF,CAAC;MACH,CAAC;MACDE;IACF,CACF,CAAC;IACD,OAAOiF,MAAM,CAAC5D,MAAM;EACtB;;EAEA;EACA,SAAS6D,kBAAkBA,CACzBC,SAAS,EAAE,MAAM,EACjBC,YAAwC,CAA3B,EAAE,CAACC,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI,CACzC,EAAE,IAAI,CAAC;IACN;IACA;IACA;IACA;IACA,IAAIC,gBAAgB,EAAE;MACpB,IACE,CAACrR,gCAAgC,CAC/BqR,gBAAgB,EAChBb,YAAY,EACZ1J,WAAW,IAAIjD,OAAO,EACtBkI,WAAW,EACXS,SACF,CAAC,EACD;QACA;MACF;MACAwD,iBAAiB,GAAGqB,gBAAgB;MACpC5R,QAAQ,CAACyR,SAAS,EAAE;QAClBhC,YAAY,EAAEzG,wBAAwB,CAAC5E,OAAO;MAChD,CAAC,CAAC;MACFsN,YAAY,GAAGE,gBAAgB,CAAC;MAChC;IACF;;IAEA;IACA;IACA,KAAKN,mBAAmB,CAAC,CAAC,CAACO,IAAI,CAACF,OAAO,IAAI;MACzCpB,iBAAiB,GAAGoB,OAAO;;MAE3B;MACA;MACA;MACA,MAAMf,OAAO,GAAGF,eAAe;MAC/B,IAAIE,OAAO,EAAE;QACXF,eAAe,GAAG,IAAI;QACtBE,OAAO,CAAC,CAAC;MACX;MAEA5Q,QAAQ,CAACyR,SAAS,EAAE;QAClBhC,YAAY,EAAEzG,wBAAwB,CAAC5E,OAAO;MAChD,CAAC,CAAC;MAEF,IAAIsN,YAAY,EAAE;QAChBA,YAAY,CAACC,OAAO,CAAC;MACvB;IACF,CAAC,CAAC;EACJ;;EAEA;EACA,IAAIZ,YAAY,CAACe,SAAS,IAAIjB,oBAAoB,EAAE;IAClDE,YAAY,CAACe,SAAS,CAACJ,YAAY,IAAI;MACrCF,kBAAkB,CAChB,+CAA+C,EAC/CE,YACF,CAAC;IACH,CAAC,CAAC;EACJ;;EAEA;EACA;EACA;EACA,IACEzS,OAAO,CAAC,QAAQ,CAAC,IACjBY,eAAe,CAAC,CAAC,IACjB2M,YAAY,IACZ,CAAC9F,yBAAyB,IAC1BY,iBAAiB,KAAK,IAAI,EAC1B;IACAyK,UAAU,CAAC,MAAM;MACf,IACEhB,YAAY,CAACiB,MAAM,KAAK,SAAS,IACjCzB,iBAAiB,KAAKC,SAAS,EAC/B;QACA9H,yBAAyB,GAAG,IAAI;QAChC8I,kBAAkB,CAChB,sDACF,CAAC;MACH;IACF,CAAC,EAAE/L,4BAA4B,CAAC,CAACwM,KAAK,CAAC,CAAC;EAC1C;;EAEA;EACA;EACA;EACA,IAAI3K,iBAAiB,KAAK,IAAI,IAAI,CAACZ,yBAAyB,EAAE;IAC5D,MAAMiL,OAAO,GAAG,MAAML,mBAAmB,CAAC,CAAC;IAE3CtR,QAAQ,CAAC,kDAAkD,EAAE;MAC3DyP,YAAY,EAAEzG,wBAAwB,CAAC5E,OAAO;IAChD,CAAC,CAAC;IAEF,OAAO;MACL6D,MAAM,EAAE,EAAE;MACVC,MAAM,EAAE,EAAE;MACV2F,IAAI,EAAE,CAAC;MACP1F,WAAW,EAAE,KAAK;MAClBK,gBAAgB,EAAEmJ;IACpB,CAAC;EACH;;EAEA;EACA3P,UAAU,CAACkQ,YAAY,CAACnB,YAAY,CAACoB,UAAU,CAACxE,MAAM,CAAC;;EAEvD;EACA,MAAMyE,SAAS,GAAGC,IAAI,CAACC,GAAG,CAAC,CAAC;EAC5B,IAAIC,gBAAgB,GAAGH,SAAS,GAAG7M,qBAAqB;EACxD,IAAIqM,gBAAgB,EAAE,MAAM,GAAG,SAAS,GAAGpB,SAAS;;EAEpD;EACA;EACA;EACA,IAAI;IACF,OAAO,IAAI,EAAE;MACX,MAAM8B,GAAG,GAAGD,IAAI,CAACC,GAAG,CAAC,CAAC;MACtB,MAAME,qBAAqB,GAAGtC,IAAI,CAACuC,GAAG,CAAC,CAAC,EAAEF,gBAAgB,GAAGD,GAAG,CAAC;MAEjE,MAAMI,cAAc,GAAG/B,oBAAoB,CAAC,CAAC;MAC7C,MAAMrG,MAAM,GAAG,MAAMb,OAAO,CAACkJ,IAAI,CAAC,CAChCtB,aAAa,EACb,IAAI5H,OAAO,CAAC,IAAI,CAAC,CAACmH,OAAO,IACvBmB,UAAU,CAACa,CAAC,IAAIA,CAAC,CAAC,IAAI,CAAC,EAAEJ,qBAAqB,EAAE5B,OAAO,CAAC,CAACqB,KAAK,CAAC,CACjE,CAAC,EACDS,cAAc,CACf,CAAC;MAEF,IAAIpI,MAAM,KAAK,IAAI,EAAE;QACnB;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,IAAIA,MAAM,CAAC9B,gBAAgB,KAAKgI,SAAS,EAAE;UACzChQ,gBAAgB,CAAC8J,MAAM,CAAC9B,gBAAgB,EAAE8D,WAAW,CAAC;UACtD,MAAMuG,WAAW,EAAEpR,UAAU,GAAG;YAC9B,GAAG6I,MAAM;YACT9B,gBAAgB,EAAEgI;UACpB,CAAC;UACD;UACA;UACA,MAAM;YAAE2B;UAAW,CAAC,GAAGpB,YAAY;UACnC,IAAIoB,UAAU,CAACW,YAAY,IAAI,CAACX,UAAU,CAACY,mBAAmB,EAAE;YAC9DF,WAAW,CAAC5D,cAAc,GAAGkD,UAAU,CAACa,IAAI;YAC5CH,WAAW,CAACI,cAAc,GAAGd,UAAU,CAACc,cAAc;YACtDJ,WAAW,CAAC3D,YAAY,GAAGiD,UAAU,CAACxE,MAAM;UAC9C;UACA;UACA;UACA;UACA;UACAoD,YAAY,CAACmC,OAAO,CAAC,CAAC;UACtB,OAAOL,WAAW;QACpB;QACA;QACA,OAAOvI,MAAM;MACf;;MAEA;MACA,IAAIiG,iBAAiB,EAAE;QACrB,OAAO;UACLtI,MAAM,EAAEwI,6BAA6B,GAAGnD,UAAU,GAAG,EAAE;UACvDpF,MAAM,EAAE,EAAE;UACV2F,IAAI,EAAE,CAAC;UACP1F,WAAW,EAAE,KAAK;UAClBK,gBAAgB,EAAE+H,iBAAiB;UACnC7H;QACF,CAAC;MACH;;MAEA;MACA,IACE2D,eAAe,CAAC0B,MAAM,CAACoF,OAAO,IAC9B9G,eAAe,CAAC0B,MAAM,CAACC,MAAM,KAAK,WAAW,IAC7C,CAACyC,6BAA6B,EAC9B;QACAA,6BAA6B,GAAG,IAAI;QACpC,IAAI,CAAC/J,yBAAyB,EAAE;UAC9B8K,kBAAkB,CAAC,iDAAiD,CAAC;UACrE;UACA;UACA;UACA;UACA;QACF;QACAT,YAAY,CAACqC,IAAI,CAAC,CAAC;MACrB;;MAEA;MACA,IAAIxB,gBAAgB,EAAE;QACpB,IAAIb,YAAY,CAACiB,MAAM,KAAK,cAAc,EAAE;UAC1C,OAAO;YACL/J,MAAM,EAAE,EAAE;YACVC,MAAM,EAAE,EAAE;YACV2F,IAAI,EAAE,CAAC;YACP1F,WAAW,EAAE,KAAK;YAClBK,gBAAgB,EAAEoJ,gBAAgB;YAClCnJ,kBAAkB,EAAE;UACtB,CAAC;QACH;MACF;;MAEA;MACA,MAAM4K,OAAO,GAAGhB,IAAI,CAACC,GAAG,CAAC,CAAC,GAAGF,SAAS;MACtC,MAAMkB,cAAc,GAAGpD,IAAI,CAACqD,KAAK,CAACF,OAAO,GAAG,IAAI,CAAC;;MAEjD;MACA,IACE,CAAC3M,yBAAyB,IAC1B6J,iBAAiB,KAAKC,SAAS,IAC/B8C,cAAc,IAAI/N,qBAAqB,GAAG,IAAI,IAC9CgH,UAAU,EACV;QACA,IAAI,CAACqF,gBAAgB,EAAE;UACrBA,gBAAgB,GAAGnR,kBAAkB,CACnC;YACE2D,OAAO;YACPiD,WAAW,EAAEA,WAAW,IAAIjD,OAAO;YACnC2M,YAAY;YACZtE;UACF,CAAC,EACDH,WAAW,EACXS,SACF,CAAC;QACH;QAEAR,UAAU,CAAC;UACTiH,GAAG,EAAE,CAAC,cAAc,GAAG;UACvBC,qBAAqB,EAAE,KAAK;UAC5BC,uBAAuB,EAAE,IAAI;UAC7BC,WAAW,EAAE;QACf,CAAC,CAAC;MACJ;MAEA,MAAM;QACJjI,IAAI,EAAE,UAAU;QAChB4B,UAAU;QACVD,MAAM,EAAE+C,kBAAkB;QAC1B7C,kBAAkB,EAAE+F,cAAc;QAClC9F,UAAU,EAAE6C,cAAc;QAC1B5C,UAAU,EAAE6C,cAAc;QAC1B3C,MAAM,EAAEoD,YAAY,CAACoB,UAAU,CAACxE,MAAM;QACtC,IAAIzG,OAAO,GAAG;UAAEwG;QAAU,CAAC,GAAG8C,SAAS;MACzC,CAAC;MAED+B,gBAAgB,GAAGF,IAAI,CAACC,GAAG,CAAC,CAAC,GAAG9M,oBAAoB;IACtD;EACF,CAAC,SAAS;IACRxD,UAAU,CAAC4R,WAAW,CAAC7C,YAAY,CAACoB,UAAU,CAACxE,MAAM,CAAC;IACtD;IACA;IACA;IACA,IAAI,CAAC4C,iBAAiB,IAAIQ,YAAY,CAACiB,MAAM,KAAK,cAAc,EAAE;MAChE,IAAIJ,gBAAgB,EAAE;QACpBjR,oBAAoB,CAACiR,gBAAgB,EAAEtF,WAAW,CAAC;MACrD;MACAyE,YAAY,CAACmC,OAAO,CAAC,CAAC;IACxB;EACF;AACF","ignoreList":[]} \ No newline at end of file +//# 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":[]} diff --git a/src/tools/shellToolResultMappers.test.ts b/src/tools/shellToolResultMappers.test.ts new file mode 100644 index 00000000..2d4f2fd6 --- /dev/null +++ b/src/tools/shellToolResultMappers.test.ts @@ -0,0 +1,71 @@ +import { expect, test } from 'bun:test' +import { BashTool } from './BashTool/BashTool.js' +import { PowerShellTool } from './PowerShellTool/PowerShellTool.js' + +test('BashTool result mapper tolerates null stderr', () => { + const result = BashTool.mapToolResultToToolResultBlockParam( + { + stdout: 'ok', + stderr: null as unknown as string, + interrupted: false, + }, + 'tool-1', + ) + + expect(result).toMatchObject({ + type: 'tool_result', + tool_use_id: 'tool-1', + content: 'ok', + }) +}) + +test('BashTool result mapper tolerates null stdout', () => { + const result = BashTool.mapToolResultToToolResultBlockParam( + { + stdout: null as unknown as string, + stderr: 'problem', + interrupted: false, + }, + 'tool-2', + ) + + expect(result).toMatchObject({ + type: 'tool_result', + tool_use_id: 'tool-2', + content: 'problem', + }) +}) + +test('PowerShellTool result mapper tolerates null stderr', () => { + const result = PowerShellTool.mapToolResultToToolResultBlockParam( + { + stdout: 'ok', + stderr: null as unknown as string, + interrupted: false, + }, + 'tool-3', + ) + + expect(result).toMatchObject({ + type: 'tool_result', + tool_use_id: 'tool-3', + content: 'ok', + }) +}) + +test('PowerShellTool result mapper tolerates null stdout', () => { + const result = PowerShellTool.mapToolResultToToolResultBlockParam( + { + stdout: null as unknown as string, + stderr: 'problem', + interrupted: false, + }, + 'tool-4', + ) + + expect(result).toMatchObject({ + type: 'tool_result', + tool_use_id: 'tool-4', + content: 'problem', + }) +}) diff --git a/src/utils/promptShellExecution.test.ts b/src/utils/promptShellExecution.test.ts new file mode 100644 index 00000000..bd8f1353 --- /dev/null +++ b/src/utils/promptShellExecution.test.ts @@ -0,0 +1,77 @@ +import { afterEach, expect, test } from 'bun:test' +import { getEmptyToolPermissionContext } from '../Tool.js' +import { BashTool } from '../tools/BashTool/BashTool.js' +import { executeShellCommandsInPrompt } from './promptShellExecution.js' + +const originalCall = BashTool.call +const originalMapToolResultToToolResultBlockParam = + BashTool.mapToolResultToToolResultBlockParam + +afterEach(() => { + BashTool.call = originalCall + BashTool.mapToolResultToToolResultBlockParam = + originalMapToolResultToToolResultBlockParam +}) + +test('executeShellCommandsInPrompt normalizes null shell output', async () => { + let normalizedResult: + | { stdout: string; stderr: string; interrupted: boolean } + | undefined + + BashTool.call = (async () => ({ + data: { + stdout: null, + stderr: null, + interrupted: false, + }, + })) as unknown as typeof BashTool.call + + BashTool.mapToolResultToToolResultBlockParam = (result, toolUseID) => { + normalizedResult = result as { + stdout: string + stderr: string + interrupted: boolean + } + return originalMapToolResultToToolResultBlockParam(result, toolUseID) + } + + await executeShellCommandsInPrompt( + '```!\ngit status\n```', + { + abortController: new AbortController(), + options: { + commands: [], + debug: false, + mainLoopModel: 'sonnet', + tools: new Map(), + verbose: false, + thinkingConfig: { type: 'disabled' }, + mcpClients: [], + mcpResources: {}, + isNonInteractiveSession: false, + agentDefinitions: { + systemDefinitions: [], + projectDefinitions: [], + userDefinitions: [], + }, + }, + readFileState: new Map(), + getAppState() { + return { + toolPermissionContext: { + ...getEmptyToolPermissionContext(), + alwaysAllowRules: { command: ['Bash(*)'] }, + }, + } + }, + setAppState() {}, + } as never, + 'security-review', + ) + + expect(normalizedResult).toEqual({ + stdout: '', + stderr: '', + interrupted: false, + }) +}) diff --git a/src/utils/promptShellExecution.ts b/src/utils/promptShellExecution.ts index 94269946..bbc9d428 100644 --- a/src/utils/promptShellExecution.ts +++ b/src/utils/promptShellExecution.ts @@ -16,7 +16,11 @@ import { processToolResultBlock } from './toolResultStorage.js' // _simulatedSedEdit) that PowerShellTool's does not. // NOTE: call() is invoked directly here, bypassing validateInput — any // load-bearing check must live in call() itself (see PR #23311). -type ShellOut = { stdout: string; stderr: string; interrupted: boolean } +type ShellOut = { + stdout: string | null | undefined + stderr: string | null | undefined + interrupted: boolean +} type PromptShellTool = Tool & { call( input: { command: string }, @@ -113,17 +117,25 @@ export async function executeShellCommandsInPrompt( } const { data } = await shellTool.call({ command }, context) + const normalizedData = { + ...data, + stdout: typeof data.stdout === 'string' ? data.stdout : '', + stderr: typeof data.stderr === 'string' ? data.stderr : '', + } // Reuse the same persistence flow as regular Bash tool calls const toolResultBlock = await processToolResultBlock( shellTool, - data, + normalizedData, randomUUID(), ) // Extract the string content from the block const output = typeof toolResultBlock.content === 'string' ? toolResultBlock.content - : formatBashOutput(data.stdout, data.stderr) + : formatBashOutput( + normalizedData.stdout, + normalizedData.stderr, + ) // Function replacer — String.replace interprets $$, $&, $`, $' in // the replacement string even with a string search pattern. Shell // output (especially PowerShell: $env:PATH, $$, $PSVersionTable) @@ -143,21 +155,23 @@ export async function executeShellCommandsInPrompt( } function formatBashOutput( - stdout: string, - stderr: string, + stdout: string | null | undefined, + stderr: string | null | undefined, inline = false, ): string { + const normalizedStdout = typeof stdout === 'string' ? stdout : '' + const normalizedStderr = typeof stderr === 'string' ? stderr : '' const parts: string[] = [] - if (stdout.trim()) { - parts.push(stdout.trim()) + if (normalizedStdout.trim()) { + parts.push(normalizedStdout.trim()) } - if (stderr.trim()) { + if (normalizedStderr.trim()) { if (inline) { - parts.push(`[stderr: ${stderr.trim()}]`) + parts.push(`[stderr: ${normalizedStderr.trim()}]`) } else { - parts.push(`[stderr]\n${stderr.trim()}`) + parts.push(`[stderr]\n${normalizedStderr.trim()}`) } }