source dump of claude code
at main 403 lines 51 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import { feature } from 'bun:bundle'; 3import figures from 'figures'; 4import * as React from 'react'; 5import type { z } from 'zod/v4'; 6import { ProgressBar } from '../../components/design-system/ProgressBar.js'; 7import { MessageResponse } from '../../components/MessageResponse.js'; 8import { linkifyUrlsInText, OutputLine } from '../../components/shell/OutputLine.js'; 9import { stringWidth } from '../../ink/stringWidth.js'; 10import { Ansi, Box, Text } from '../../ink.js'; 11import type { ToolProgressData } from '../../Tool.js'; 12import type { ProgressMessage } from '../../types/message.js'; 13import type { MCPProgress } from '../../types/tools.js'; 14import { formatNumber } from '../../utils/format.js'; 15import { createHyperlink } from '../../utils/hyperlink.js'; 16import { getContentSizeEstimate, type MCPToolResult } from '../../utils/mcpValidation.js'; 17import { jsonParse, jsonStringify } from '../../utils/slowOperations.js'; 18import type { inputSchema } from './MCPTool.js'; 19 20// Threshold for displaying warning about large MCP responses 21const MCP_OUTPUT_WARNING_THRESHOLD_TOKENS = 10_000; 22 23// In non-verbose mode, truncate individual input values to keep the header 24// compact. Matches BashTool's philosophy of showing enough to identify the 25// call without dumping the entire payload inline. 26const MAX_INPUT_VALUE_CHARS = 80; 27 28// Max number of top-level keys before we fall back to raw JSON display. 29// Beyond this a flat k:v list is more noise than help. 30const MAX_FLAT_JSON_KEYS = 12; 31 32// Don't attempt flat-object parsing for large blobs. 33const MAX_FLAT_JSON_CHARS = 5_000; 34 35// Don't attempt to parse JSON blobs larger than this (perf safety). 36const MAX_JSON_PARSE_CHARS = 200_000; 37 38// A string value is "dominant text payload" if it has newlines or is 39// long enough that inline display would be worse than unwrapping. 40const UNWRAP_MIN_STRING_LEN = 200; 41export function renderToolUseMessage(input: z.infer<ReturnType<typeof inputSchema>>, { 42 verbose 43}: { 44 verbose: boolean; 45}): React.ReactNode { 46 if (Object.keys(input).length === 0) { 47 return ''; 48 } 49 return Object.entries(input).map(([key, value]) => { 50 let rendered = jsonStringify(value); 51 if (feature('MCP_RICH_OUTPUT') && !verbose && rendered.length > MAX_INPUT_VALUE_CHARS) { 52 rendered = rendered.slice(0, MAX_INPUT_VALUE_CHARS).trimEnd() + '…'; 53 } 54 return `${key}: ${rendered}`; 55 }).join(', '); 56} 57export function renderToolUseProgressMessage(progressMessagesForMessage: ProgressMessage<MCPProgress>[]): React.ReactNode { 58 const lastProgress = progressMessagesForMessage.at(-1); 59 if (!lastProgress?.data) { 60 return <MessageResponse height={1}> 61 <Text dimColor>Running</Text> 62 </MessageResponse>; 63 } 64 const { 65 progress, 66 total, 67 progressMessage 68 } = lastProgress.data; 69 if (progress === undefined) { 70 return <MessageResponse height={1}> 71 <Text dimColor>Running</Text> 72 </MessageResponse>; 73 } 74 if (total !== undefined && total > 0) { 75 const ratio = Math.min(1, Math.max(0, progress / total)); 76 const percentage = Math.round(ratio * 100); 77 return <MessageResponse> 78 <Box flexDirection="column"> 79 {progressMessage && <Text dimColor>{progressMessage}</Text>} 80 <Box flexDirection="row" gap={1}> 81 <ProgressBar ratio={ratio} width={20} /> 82 <Text dimColor>{percentage}%</Text> 83 </Box> 84 </Box> 85 </MessageResponse>; 86 } 87 return <MessageResponse height={1}> 88 <Text dimColor>{progressMessage ?? `Processing… ${progress}`}</Text> 89 </MessageResponse>; 90} 91export function renderToolResultMessage(output: string | MCPToolResult, _progressMessagesForMessage: ProgressMessage<ToolProgressData>[], { 92 verbose, 93 input 94}: { 95 verbose: boolean; 96 input?: unknown; 97}): React.ReactNode { 98 const mcpOutput = output as MCPToolResult; 99 if (!verbose) { 100 const slackSend = trySlackSendCompact(mcpOutput, input); 101 if (slackSend !== null) { 102 return <MessageResponse height={1}> 103 <Text> 104 Sent a message to{' '} 105 <Ansi>{createHyperlink(slackSend.url, slackSend.channel)}</Ansi> 106 </Text> 107 </MessageResponse>; 108 } 109 } 110 const estimatedTokens = getContentSizeEstimate(mcpOutput); 111 const showWarning = estimatedTokens > MCP_OUTPUT_WARNING_THRESHOLD_TOKENS; 112 const warningMessage = showWarning ? `${figures.warning} Large MCP response (~${formatNumber(estimatedTokens)} tokens), this can fill up context quickly` : null; 113 let contentElement: React.ReactNode; 114 if (Array.isArray(mcpOutput)) { 115 const contentBlocks = mcpOutput.map((item, i) => { 116 if (item.type === 'image') { 117 return <Box key={i} justifyContent="space-between" overflowX="hidden" width="100%"> 118 <MessageResponse height={1}> 119 <Text>[Image]</Text> 120 </MessageResponse> 121 </Box>; 122 } 123 // For text blocks and any other block types, extract text if available 124 const textContent = item.type === 'text' && 'text' in item && item.text !== null && item.text !== undefined ? String(item.text) : ''; 125 return feature('MCP_RICH_OUTPUT') ? <MCPTextOutput key={i} content={textContent} verbose={verbose} /> : <OutputLine key={i} content={textContent} verbose={verbose} />; 126 }); 127 128 // Wrap array content in a column layout 129 contentElement = <Box flexDirection="column" width="100%"> 130 {contentBlocks} 131 </Box>; 132 } else if (!mcpOutput) { 133 contentElement = <Box justifyContent="space-between" overflowX="hidden" width="100%"> 134 <MessageResponse height={1}> 135 <Text dimColor>(No content)</Text> 136 </MessageResponse> 137 </Box>; 138 } else { 139 contentElement = feature('MCP_RICH_OUTPUT') ? <MCPTextOutput content={mcpOutput} verbose={verbose} /> : <OutputLine content={mcpOutput} verbose={verbose} />; 140 } 141 if (warningMessage) { 142 return <Box flexDirection="column"> 143 <MessageResponse height={1}> 144 <Text color="warning">{warningMessage}</Text> 145 </MessageResponse> 146 {contentElement} 147 </Box>; 148 } 149 return contentElement; 150} 151 152/** 153 * Render MCP text output. Tries three strategies in order: 154 * 1. If JSON wraps a single dominant text payload (e.g. slack's 155 * {"messages":"line1\nline2..."}), unwrap and let OutputLine truncate. 156 * 2. If JSON is a small flat-ish object, render as aligned key: value. 157 * 3. Otherwise fall through to OutputLine (pretty-print + truncate). 158 */ 159function MCPTextOutput(t0) { 160 const $ = _c(18); 161 const { 162 content, 163 verbose 164 } = t0; 165 let t1; 166 if ($[0] !== content || $[1] !== verbose) { 167 t1 = Symbol.for("react.early_return_sentinel"); 168 bb0: { 169 const unwrapped = tryUnwrapTextPayload(content); 170 if (unwrapped !== null) { 171 const t2 = unwrapped.extras.length > 0 && <Text dimColor={true}>{unwrapped.extras.map(_temp).join(" \xB7 ")}</Text>; 172 let t3; 173 if ($[3] !== unwrapped || $[4] !== verbose) { 174 t3 = <OutputLine content={unwrapped.body} verbose={verbose} linkifyUrls={true} />; 175 $[3] = unwrapped; 176 $[4] = verbose; 177 $[5] = t3; 178 } else { 179 t3 = $[5]; 180 } 181 let t4; 182 if ($[6] !== t2 || $[7] !== t3) { 183 t4 = <MessageResponse><Box flexDirection="column">{t2}{t3}</Box></MessageResponse>; 184 $[6] = t2; 185 $[7] = t3; 186 $[8] = t4; 187 } else { 188 t4 = $[8]; 189 } 190 t1 = t4; 191 break bb0; 192 } 193 } 194 $[0] = content; 195 $[1] = verbose; 196 $[2] = t1; 197 } else { 198 t1 = $[2]; 199 } 200 if (t1 !== Symbol.for("react.early_return_sentinel")) { 201 return t1; 202 } 203 let t2; 204 if ($[9] !== content) { 205 t2 = Symbol.for("react.early_return_sentinel"); 206 bb1: { 207 const flat = tryFlattenJson(content); 208 if (flat !== null) { 209 const maxKeyWidth = Math.max(...flat.map(_temp2)); 210 let t3; 211 if ($[11] !== maxKeyWidth) { 212 t3 = (t4, i) => { 213 const [key, value] = t4; 214 return <Text key={i}><Text dimColor={true}>{key.padEnd(maxKeyWidth)}: </Text><Ansi>{linkifyUrlsInText(value)}</Ansi></Text>; 215 }; 216 $[11] = maxKeyWidth; 217 $[12] = t3; 218 } else { 219 t3 = $[12]; 220 } 221 const t4 = <Box flexDirection="column">{flat.map(t3)}</Box>; 222 let t5; 223 if ($[13] !== t4) { 224 t5 = <MessageResponse>{t4}</MessageResponse>; 225 $[13] = t4; 226 $[14] = t5; 227 } else { 228 t5 = $[14]; 229 } 230 t2 = t5; 231 break bb1; 232 } 233 } 234 $[9] = content; 235 $[10] = t2; 236 } else { 237 t2 = $[10]; 238 } 239 if (t2 !== Symbol.for("react.early_return_sentinel")) { 240 return t2; 241 } 242 let t3; 243 if ($[15] !== content || $[16] !== verbose) { 244 t3 = <OutputLine content={content} verbose={verbose} linkifyUrls={true} />; 245 $[15] = content; 246 $[16] = verbose; 247 $[17] = t3; 248 } else { 249 t3 = $[17]; 250 } 251 return t3; 252} 253 254/** 255 * Parse content as a JSON object and return its entries. Null if content 256 * doesn't parse, isn't an object, is too large, or has 0/too-many keys. 257 */ 258function _temp2(t0) { 259 const [k_0] = t0; 260 return stringWidth(k_0); 261} 262function _temp(t0) { 263 const [k, v] = t0; 264 return `${k}: ${v}`; 265} 266function parseJsonEntries(content: string, { 267 maxChars, 268 maxKeys 269}: { 270 maxChars: number; 271 maxKeys: number; 272}): [string, unknown][] | null { 273 const trimmed = content.trim(); 274 if (trimmed.length === 0 || trimmed.length > maxChars || trimmed[0] !== '{') { 275 return null; 276 } 277 let parsed: unknown; 278 try { 279 parsed = jsonParse(trimmed); 280 } catch { 281 return null; 282 } 283 if (parsed === null || typeof parsed !== 'object' || Array.isArray(parsed)) { 284 return null; 285 } 286 const entries = Object.entries(parsed); 287 if (entries.length === 0 || entries.length > maxKeys) { 288 return null; 289 } 290 return entries; 291} 292 293/** 294 * If content parses as a JSON object where every value is a scalar or a 295 * small nested object, flatten it to [key, displayValue] pairs. Nested 296 * objects get one-line JSON. Returns null if content doesn't qualify. 297 */ 298export function tryFlattenJson(content: string): [string, string][] | null { 299 const entries = parseJsonEntries(content, { 300 maxChars: MAX_FLAT_JSON_CHARS, 301 maxKeys: MAX_FLAT_JSON_KEYS 302 }); 303 if (entries === null) return null; 304 const result: [string, string][] = []; 305 for (const [key, value] of entries) { 306 if (typeof value === 'string') { 307 result.push([key, value]); 308 } else if (value === null || typeof value === 'number' || typeof value === 'boolean') { 309 result.push([key, String(value)]); 310 } else if (typeof value === 'object') { 311 const compact = jsonStringify(value); 312 if (compact.length > 120) return null; 313 result.push([key, compact]); 314 } else { 315 return null; 316 } 317 } 318 return result; 319} 320 321/** 322 * If content is a JSON object where one key holds a dominant string payload 323 * (multiline or long) and all siblings are small scalars, unwrap it. This 324 * handles the common MCP pattern of {"messages":"line1\nline2..."} where 325 * pretty-printing keeps \n escaped but we want real line breaks + truncation. 326 */ 327export function tryUnwrapTextPayload(content: string): { 328 body: string; 329 extras: [string, string][]; 330} | null { 331 const entries = parseJsonEntries(content, { 332 maxChars: MAX_JSON_PARSE_CHARS, 333 maxKeys: 4 334 }); 335 if (entries === null) return null; 336 // Find the one dominant string payload. Trim first: a trailing \n on a 337 // short sibling (e.g. pagination hints) shouldn't make it "dominant". 338 let body: string | null = null; 339 const extras: [string, string][] = []; 340 for (const [key, value] of entries) { 341 if (typeof value === 'string') { 342 const t = value.trimEnd(); 343 const isDominant = t.length > UNWRAP_MIN_STRING_LEN || t.includes('\n') && t.length > 50; 344 if (isDominant) { 345 if (body !== null) return null; // two big strings — ambiguous 346 body = t; 347 continue; 348 } 349 if (t.length > 150) return null; 350 extras.push([key, t.replace(/\s+/g, ' ')]); 351 } else if (value === null || typeof value === 'number' || typeof value === 'boolean') { 352 extras.push([key, String(value)]); 353 } else { 354 return null; // nested object/array — use flat or pretty-print path 355 } 356 } 357 if (body === null) return null; 358 return { 359 body, 360 extras 361 }; 362} 363const SLACK_ARCHIVES_RE = /^https:\/\/[a-z0-9-]+\.slack\.com\/archives\/([A-Z0-9]+)\/p\d+$/; 364 365/** 366 * Detect a Slack send-message result and return a compact {channel, url} pair. 367 * Matches both hosted (claude.ai Slack) and community MCP server shapes — 368 * both return `message_link` in the result. The channel label prefers the 369 * tool input (may be a name like "#foo" or an ID like "C09EVDAN1NK") and 370 * falls back to the ID parsed from the archives URL. 371 */ 372export function trySlackSendCompact(output: string | MCPToolResult, input: unknown): { 373 channel: string; 374 url: string; 375} | null { 376 let text: unknown = output; 377 if (Array.isArray(output)) { 378 const block = output.find(b => b.type === 'text'); 379 text = block && 'text' in block ? block.text : undefined; 380 } 381 if (typeof text !== 'string' || !text.includes('"message_link"')) { 382 return null; 383 } 384 const entries = parseJsonEntries(text, { 385 maxChars: 2000, 386 maxKeys: 6 387 }); 388 const url = entries?.find(([k]) => k === 'message_link')?.[1]; 389 if (typeof url !== 'string') return null; 390 const m = SLACK_ARCHIVES_RE.exec(url); 391 if (!m) return null; 392 const inp = input as { 393 channel_id?: unknown; 394 channel?: unknown; 395 } | undefined; 396 const raw = inp?.channel_id ?? inp?.channel ?? m[1]; 397 const label = typeof raw === 'string' && raw ? raw : 'slack'; 398 return { 399 channel: label.startsWith('#') ? label : `#${label}`, 400 url 401 }; 402} 403//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","figures","React","z","ProgressBar","MessageResponse","linkifyUrlsInText","OutputLine","stringWidth","Ansi","Box","Text","ToolProgressData","ProgressMessage","MCPProgress","formatNumber","createHyperlink","getContentSizeEstimate","MCPToolResult","jsonParse","jsonStringify","inputSchema","MCP_OUTPUT_WARNING_THRESHOLD_TOKENS","MAX_INPUT_VALUE_CHARS","MAX_FLAT_JSON_KEYS","MAX_FLAT_JSON_CHARS","MAX_JSON_PARSE_CHARS","UNWRAP_MIN_STRING_LEN","renderToolUseMessage","input","infer","ReturnType","verbose","ReactNode","Object","keys","length","entries","map","key","value","rendered","slice","trimEnd","join","renderToolUseProgressMessage","progressMessagesForMessage","lastProgress","at","data","progress","total","progressMessage","undefined","ratio","Math","min","max","percentage","round","renderToolResultMessage","output","_progressMessagesForMessage","mcpOutput","slackSend","trySlackSendCompact","url","channel","estimatedTokens","showWarning","warningMessage","warning","contentElement","Array","isArray","contentBlocks","item","i","type","textContent","text","String","MCPTextOutput","t0","$","_c","content","t1","Symbol","for","bb0","unwrapped","tryUnwrapTextPayload","t2","extras","_temp","t3","body","t4","bb1","flat","tryFlattenJson","maxKeyWidth","_temp2","padEnd","t5","k_0","k","v","parseJsonEntries","maxChars","maxKeys","trimmed","trim","parsed","result","push","compact","t","isDominant","includes","replace","SLACK_ARCHIVES_RE","block","find","b","m","exec","inp","channel_id","raw","label","startsWith"],"sources":["UI.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport figures from 'figures'\nimport * as React from 'react'\nimport type { z } from 'zod/v4'\nimport { ProgressBar } from '../../components/design-system/ProgressBar.js'\nimport { MessageResponse } from '../../components/MessageResponse.js'\nimport {\n  linkifyUrlsInText,\n  OutputLine,\n} from '../../components/shell/OutputLine.js'\nimport { stringWidth } from '../../ink/stringWidth.js'\nimport { Ansi, Box, Text } from '../../ink.js'\nimport type { ToolProgressData } from '../../Tool.js'\nimport type { ProgressMessage } from '../../types/message.js'\nimport type { MCPProgress } from '../../types/tools.js'\nimport { formatNumber } from '../../utils/format.js'\nimport { createHyperlink } from '../../utils/hyperlink.js'\nimport {\n  getContentSizeEstimate,\n  type MCPToolResult,\n} from '../../utils/mcpValidation.js'\nimport { jsonParse, jsonStringify } from '../../utils/slowOperations.js'\nimport type { inputSchema } from './MCPTool.js'\n\n// Threshold for displaying warning about large MCP responses\nconst MCP_OUTPUT_WARNING_THRESHOLD_TOKENS = 10_000\n\n// In non-verbose mode, truncate individual input values to keep the header\n// compact. Matches BashTool's philosophy of showing enough to identify the\n// call without dumping the entire payload inline.\nconst MAX_INPUT_VALUE_CHARS = 80\n\n// Max number of top-level keys before we fall back to raw JSON display.\n// Beyond this a flat k:v list is more noise than help.\nconst MAX_FLAT_JSON_KEYS = 12\n\n// Don't attempt flat-object parsing for large blobs.\nconst MAX_FLAT_JSON_CHARS = 5_000\n\n// Don't attempt to parse JSON blobs larger than this (perf safety).\nconst MAX_JSON_PARSE_CHARS = 200_000\n\n// A string value is \"dominant text payload\" if it has newlines or is\n// long enough that inline display would be worse than unwrapping.\nconst UNWRAP_MIN_STRING_LEN = 200\n\nexport function renderToolUseMessage(\n  input: z.infer<ReturnType<typeof inputSchema>>,\n  { verbose }: { verbose: boolean },\n): React.ReactNode {\n  if (Object.keys(input).length === 0) {\n    return ''\n  }\n  return Object.entries(input)\n    .map(([key, value]) => {\n      let rendered = jsonStringify(value)\n      if (\n        feature('MCP_RICH_OUTPUT') &&\n        !verbose &&\n        rendered.length > MAX_INPUT_VALUE_CHARS\n      ) {\n        rendered = rendered.slice(0, MAX_INPUT_VALUE_CHARS).trimEnd() + '…'\n      }\n      return `${key}: ${rendered}`\n    })\n    .join(', ')\n}\n\nexport function renderToolUseProgressMessage(\n  progressMessagesForMessage: ProgressMessage<MCPProgress>[],\n): React.ReactNode {\n  const lastProgress = progressMessagesForMessage.at(-1)\n\n  if (!lastProgress?.data) {\n    return (\n      <MessageResponse height={1}>\n        <Text dimColor>Running…</Text>\n      </MessageResponse>\n    )\n  }\n\n  const { progress, total, progressMessage } = lastProgress.data\n\n  if (progress === undefined) {\n    return (\n      <MessageResponse height={1}>\n        <Text dimColor>Running…</Text>\n      </MessageResponse>\n    )\n  }\n\n  if (total !== undefined && total > 0) {\n    const ratio = Math.min(1, Math.max(0, progress / total))\n    const percentage = Math.round(ratio * 100)\n    return (\n      <MessageResponse>\n        <Box flexDirection=\"column\">\n          {progressMessage && <Text dimColor>{progressMessage}</Text>}\n          <Box flexDirection=\"row\" gap={1}>\n            <ProgressBar ratio={ratio} width={20} />\n            <Text dimColor>{percentage}%</Text>\n          </Box>\n        </Box>\n      </MessageResponse>\n    )\n  }\n\n  return (\n    <MessageResponse height={1}>\n      <Text dimColor>{progressMessage ?? `Processing… ${progress}`}</Text>\n    </MessageResponse>\n  )\n}\n\nexport function renderToolResultMessage(\n  output: string | MCPToolResult,\n  _progressMessagesForMessage: ProgressMessage<ToolProgressData>[],\n  { verbose, input }: { verbose: boolean; input?: unknown },\n): React.ReactNode {\n  const mcpOutput = output as MCPToolResult\n\n  if (!verbose) {\n    const slackSend = trySlackSendCompact(mcpOutput, input)\n    if (slackSend !== null) {\n      return (\n        <MessageResponse height={1}>\n          <Text>\n            Sent a message to{' '}\n            <Ansi>{createHyperlink(slackSend.url, slackSend.channel)}</Ansi>\n          </Text>\n        </MessageResponse>\n      )\n    }\n  }\n\n  const estimatedTokens = getContentSizeEstimate(mcpOutput)\n  const showWarning = estimatedTokens > MCP_OUTPUT_WARNING_THRESHOLD_TOKENS\n  const warningMessage = showWarning\n    ? `${figures.warning} Large MCP response (~${formatNumber(estimatedTokens)} tokens), this can fill up context quickly`\n    : null\n\n  let contentElement: React.ReactNode\n  if (Array.isArray(mcpOutput)) {\n    const contentBlocks = mcpOutput.map((item, i) => {\n      if (item.type === 'image') {\n        return (\n          <Box\n            key={i}\n            justifyContent=\"space-between\"\n            overflowX=\"hidden\"\n            width=\"100%\"\n          >\n            <MessageResponse height={1}>\n              <Text>[Image]</Text>\n            </MessageResponse>\n          </Box>\n        )\n      }\n      // For text blocks and any other block types, extract text if available\n      const textContent =\n        item.type === 'text' &&\n        'text' in item &&\n        item.text !== null &&\n        item.text !== undefined\n          ? String(item.text)\n          : ''\n      return feature('MCP_RICH_OUTPUT') ? (\n        <MCPTextOutput key={i} content={textContent} verbose={verbose} />\n      ) : (\n        <OutputLine key={i} content={textContent} verbose={verbose} />\n      )\n    })\n\n    // Wrap array content in a column layout\n    contentElement = (\n      <Box flexDirection=\"column\" width=\"100%\">\n        {contentBlocks}\n      </Box>\n    )\n  } else if (!mcpOutput) {\n    contentElement = (\n      <Box justifyContent=\"space-between\" overflowX=\"hidden\" width=\"100%\">\n        <MessageResponse height={1}>\n          <Text dimColor>(No content)</Text>\n        </MessageResponse>\n      </Box>\n    )\n  } else {\n    contentElement = feature('MCP_RICH_OUTPUT') ? (\n      <MCPTextOutput content={mcpOutput} verbose={verbose} />\n    ) : (\n      <OutputLine content={mcpOutput} verbose={verbose} />\n    )\n  }\n\n  if (warningMessage) {\n    return (\n      <Box flexDirection=\"column\">\n        <MessageResponse height={1}>\n          <Text color=\"warning\">{warningMessage}</Text>\n        </MessageResponse>\n        {contentElement}\n      </Box>\n    )\n  }\n\n  return contentElement\n}\n\n/**\n * Render MCP text output. Tries three strategies in order:\n * 1. If JSON wraps a single dominant text payload (e.g. slack's\n *    {\"messages\":\"line1\\nline2...\"}), unwrap and let OutputLine truncate.\n * 2. If JSON is a small flat-ish object, render as aligned key: value.\n * 3. Otherwise fall through to OutputLine (pretty-print + truncate).\n */\nfunction MCPTextOutput({\n  content,\n  verbose,\n}: {\n  content: string\n  verbose: boolean\n}): React.ReactNode {\n  const unwrapped = tryUnwrapTextPayload(content)\n  if (unwrapped !== null) {\n    return (\n      <MessageResponse>\n        <Box flexDirection=\"column\">\n          {unwrapped.extras.length > 0 && (\n            <Text dimColor>\n              {unwrapped.extras.map(([k, v]) => `${k}: ${v}`).join(' · ')}\n            </Text>\n          )}\n          <OutputLine content={unwrapped.body} verbose={verbose} linkifyUrls />\n        </Box>\n      </MessageResponse>\n    )\n  }\n  const flat = tryFlattenJson(content)\n  if (flat !== null) {\n    const maxKeyWidth = Math.max(...flat.map(([k]) => stringWidth(k)))\n    return (\n      <MessageResponse>\n        <Box flexDirection=\"column\">\n          {flat.map(([key, value], i) => (\n            <Text key={i}>\n              <Text dimColor>{key.padEnd(maxKeyWidth)}: </Text>\n              <Ansi>{linkifyUrlsInText(value)}</Ansi>\n            </Text>\n          ))}\n        </Box>\n      </MessageResponse>\n    )\n  }\n  return <OutputLine content={content} verbose={verbose} linkifyUrls />\n}\n\n/**\n * Parse content as a JSON object and return its entries. Null if content\n * doesn't parse, isn't an object, is too large, or has 0/too-many keys.\n */\nfunction parseJsonEntries(\n  content: string,\n  { maxChars, maxKeys }: { maxChars: number; maxKeys: number },\n): [string, unknown][] | null {\n  const trimmed = content.trim()\n  if (trimmed.length === 0 || trimmed.length > maxChars || trimmed[0] !== '{') {\n    return null\n  }\n  let parsed: unknown\n  try {\n    parsed = jsonParse(trimmed)\n  } catch {\n    return null\n  }\n  if (parsed === null || typeof parsed !== 'object' || Array.isArray(parsed)) {\n    return null\n  }\n  const entries = Object.entries(parsed)\n  if (entries.length === 0 || entries.length > maxKeys) {\n    return null\n  }\n  return entries\n}\n\n/**\n * If content parses as a JSON object where every value is a scalar or a\n * small nested object, flatten it to [key, displayValue] pairs. Nested\n * objects get one-line JSON. Returns null if content doesn't qualify.\n */\nexport function tryFlattenJson(content: string): [string, string][] | null {\n  const entries = parseJsonEntries(content, {\n    maxChars: MAX_FLAT_JSON_CHARS,\n    maxKeys: MAX_FLAT_JSON_KEYS,\n  })\n  if (entries === null) return null\n  const result: [string, string][] = []\n  for (const [key, value] of entries) {\n    if (typeof value === 'string') {\n      result.push([key, value])\n    } else if (\n      value === null ||\n      typeof value === 'number' ||\n      typeof value === 'boolean'\n    ) {\n      result.push([key, String(value)])\n    } else if (typeof value === 'object') {\n      const compact = jsonStringify(value)\n      if (compact.length > 120) return null\n      result.push([key, compact])\n    } else {\n      return null\n    }\n  }\n  return result\n}\n\n/**\n * If content is a JSON object where one key holds a dominant string payload\n * (multiline or long) and all siblings are small scalars, unwrap it. This\n * handles the common MCP pattern of {\"messages\":\"line1\\nline2...\"} where\n * pretty-printing keeps \\n escaped but we want real line breaks + truncation.\n */\nexport function tryUnwrapTextPayload(\n  content: string,\n): { body: string; extras: [string, string][] } | null {\n  const entries = parseJsonEntries(content, {\n    maxChars: MAX_JSON_PARSE_CHARS,\n    maxKeys: 4,\n  })\n  if (entries === null) return null\n  // Find the one dominant string payload. Trim first: a trailing \\n on a\n  // short sibling (e.g. pagination hints) shouldn't make it \"dominant\".\n  let body: string | null = null\n  const extras: [string, string][] = []\n  for (const [key, value] of entries) {\n    if (typeof value === 'string') {\n      const t = value.trimEnd()\n      const isDominant =\n        t.length > UNWRAP_MIN_STRING_LEN || (t.includes('\\n') && t.length > 50)\n      if (isDominant) {\n        if (body !== null) return null // two big strings — ambiguous\n        body = t\n        continue\n      }\n      if (t.length > 150) return null\n      extras.push([key, t.replace(/\\s+/g, ' ')])\n    } else if (\n      value === null ||\n      typeof value === 'number' ||\n      typeof value === 'boolean'\n    ) {\n      extras.push([key, String(value)])\n    } else {\n      return null // nested object/array — use flat or pretty-print path\n    }\n  }\n  if (body === null) return null\n  return { body, extras }\n}\n\nconst SLACK_ARCHIVES_RE =\n  /^https:\\/\\/[a-z0-9-]+\\.slack\\.com\\/archives\\/([A-Z0-9]+)\\/p\\d+$/\n\n/**\n * Detect a Slack send-message result and return a compact {channel, url} pair.\n * Matches both hosted (claude.ai Slack) and community MCP server shapes —\n * both return `message_link` in the result. The channel label prefers the\n * tool input (may be a name like \"#foo\" or an ID like \"C09EVDAN1NK\") and\n * falls back to the ID parsed from the archives URL.\n */\nexport function trySlackSendCompact(\n  output: string | MCPToolResult,\n  input: unknown,\n): { channel: string; url: string } | null {\n  let text: unknown = output\n  if (Array.isArray(output)) {\n    const block = output.find(b => b.type === 'text')\n    text = block && 'text' in block ? block.text : undefined\n  }\n  if (typeof text !== 'string' || !text.includes('\"message_link\"')) {\n    return null\n  }\n\n  const entries = parseJsonEntries(text, { maxChars: 2000, maxKeys: 6 })\n  const url = entries?.find(([k]) => k === 'message_link')?.[1]\n  if (typeof url !== 'string') return null\n  const m = SLACK_ARCHIVES_RE.exec(url)\n  if (!m) return null\n\n  const inp = input as { channel_id?: unknown; channel?: unknown } | undefined\n  const raw = inp?.channel_id ?? inp?.channel ?? m[1]\n  const label = typeof raw === 'string' && raw ? raw : 'slack'\n  return { channel: label.startsWith('#') ? label : `#${label}`, url }\n}\n"],"mappings":"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","ignoreList":[]}