source dump of claude code
at main 118 lines 14 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import * as React from 'react'; 3import { useMemo } from 'react'; 4import { useTerminalSize } from '../../hooks/useTerminalSize.js'; 5import { Ansi, Text } from '../../ink.js'; 6import { createHyperlink } from '../../utils/hyperlink.js'; 7import { jsonParse, jsonStringify } from '../../utils/slowOperations.js'; 8import { renderTruncatedContent } from '../../utils/terminal.js'; 9import { MessageResponse } from '../MessageResponse.js'; 10import { InVirtualListContext } from '../messageActions.js'; 11import { useExpandShellOutput } from './ExpandShellOutputContext.js'; 12export function tryFormatJson(line: string): string { 13 try { 14 const parsed = jsonParse(line); 15 const stringified = jsonStringify(parsed); 16 17 // Check if precision was lost during JSON round-trip 18 // This happens when large integers exceed Number.MAX_SAFE_INTEGER 19 // We normalize both strings by removing whitespace and unnecessary 20 // escapes (\/ is valid but optional in JSON) for comparison 21 const normalizedOriginal = line.replace(/\\\//g, '/').replace(/\s+/g, ''); 22 const normalizedStringified = stringified.replace(/\s+/g, ''); 23 if (normalizedOriginal !== normalizedStringified) { 24 // Precision loss detected - return original line unformatted 25 return line; 26 } 27 return jsonStringify(parsed, null, 2); 28 } catch { 29 return line; 30 } 31} 32const MAX_JSON_FORMAT_LENGTH = 10_000; 33export function tryJsonFormatContent(content: string): string { 34 if (content.length > MAX_JSON_FORMAT_LENGTH) { 35 return content; 36 } 37 const allLines = content.split('\n'); 38 return allLines.map(tryFormatJson).join('\n'); 39} 40 41// Match http(s) URLs inside JSON string values. Conservative: no quotes, 42// no whitespace, no trailing comma/brace that'd be JSON structure. 43const URL_IN_JSON = /https?:\/\/[^\s"'<>\\]+/g; 44export function linkifyUrlsInText(content: string): string { 45 return content.replace(URL_IN_JSON, url => createHyperlink(url)); 46} 47export function OutputLine(t0) { 48 const $ = _c(11); 49 const { 50 content, 51 verbose, 52 isError, 53 isWarning, 54 linkifyUrls 55 } = t0; 56 const { 57 columns 58 } = useTerminalSize(); 59 const expandShellOutput = useExpandShellOutput(); 60 const inVirtualList = React.useContext(InVirtualListContext); 61 const shouldShowFull = verbose || expandShellOutput; 62 let t1; 63 if ($[0] !== columns || $[1] !== content || $[2] !== inVirtualList || $[3] !== linkifyUrls || $[4] !== shouldShowFull) { 64 bb0: { 65 let formatted = tryJsonFormatContent(content); 66 if (linkifyUrls) { 67 formatted = linkifyUrlsInText(formatted); 68 } 69 if (shouldShowFull) { 70 t1 = stripUnderlineAnsi(formatted); 71 break bb0; 72 } 73 t1 = stripUnderlineAnsi(renderTruncatedContent(formatted, columns, inVirtualList)); 74 } 75 $[0] = columns; 76 $[1] = content; 77 $[2] = inVirtualList; 78 $[3] = linkifyUrls; 79 $[4] = shouldShowFull; 80 $[5] = t1; 81 } else { 82 t1 = $[5]; 83 } 84 const formattedContent = t1; 85 const color = isError ? "error" : isWarning ? "warning" : undefined; 86 let t2; 87 if ($[6] !== formattedContent) { 88 t2 = <Ansi>{formattedContent}</Ansi>; 89 $[6] = formattedContent; 90 $[7] = t2; 91 } else { 92 t2 = $[7]; 93 } 94 let t3; 95 if ($[8] !== color || $[9] !== t2) { 96 t3 = <MessageResponse><Text color={color}>{t2}</Text></MessageResponse>; 97 $[8] = color; 98 $[9] = t2; 99 $[10] = t3; 100 } else { 101 t3 = $[10]; 102 } 103 return t3; 104} 105 106/** 107 * Underline ANSI codes in particular tend to leak out for some reason. I wasn't 108 * able to figure out why, or why emitting a reset ANSI code wasn't enough to 109 * prevent them from leaking. I also didn't want to strip all ANSI codes with 110 * stripAnsi(), because we used to do that and people complained about losing 111 * all formatting. So we just strip the underline ANSI codes specifically. 112 */ 113export function stripUnderlineAnsi(content: string): string { 114 return content.replace( 115 // eslint-disable-next-line no-control-regex 116 /\u001b\[([0-9]+;)*4(;[0-9]+)*m|\u001b\[4(;[0-9]+)*m|\u001b\[([0-9]+;)*4m/g, ''); 117} 118//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useMemo","useTerminalSize","Ansi","Text","createHyperlink","jsonParse","jsonStringify","renderTruncatedContent","MessageResponse","InVirtualListContext","useExpandShellOutput","tryFormatJson","line","parsed","stringified","normalizedOriginal","replace","normalizedStringified","MAX_JSON_FORMAT_LENGTH","tryJsonFormatContent","content","length","allLines","split","map","join","URL_IN_JSON","linkifyUrlsInText","url","OutputLine","t0","$","_c","verbose","isError","isWarning","linkifyUrls","columns","expandShellOutput","inVirtualList","useContext","shouldShowFull","t1","bb0","formatted","stripUnderlineAnsi","formattedContent","color","undefined","t2","t3"],"sources":["OutputLine.tsx"],"sourcesContent":["import * as React from 'react'\nimport { useMemo } from 'react'\nimport { useTerminalSize } from '../../hooks/useTerminalSize.js'\nimport { Ansi, Text } from '../../ink.js'\nimport { createHyperlink } from '../../utils/hyperlink.js'\nimport { jsonParse, jsonStringify } from '../../utils/slowOperations.js'\nimport { renderTruncatedContent } from '../../utils/terminal.js'\nimport { MessageResponse } from '../MessageResponse.js'\nimport { InVirtualListContext } from '../messageActions.js'\nimport { useExpandShellOutput } from './ExpandShellOutputContext.js'\n\nexport function tryFormatJson(line: string): string {\n  try {\n    const parsed = jsonParse(line)\n    const stringified = jsonStringify(parsed)\n\n    // Check if precision was lost during JSON round-trip\n    // This happens when large integers exceed Number.MAX_SAFE_INTEGER\n    // We normalize both strings by removing whitespace and unnecessary\n    // escapes (\\/ is valid but optional in JSON) for comparison\n    const normalizedOriginal = line.replace(/\\\\\\//g, '/').replace(/\\s+/g, '')\n    const normalizedStringified = stringified.replace(/\\s+/g, '')\n\n    if (normalizedOriginal !== normalizedStringified) {\n      // Precision loss detected - return original line unformatted\n      return line\n    }\n\n    return jsonStringify(parsed, null, 2)\n  } catch {\n    return line\n  }\n}\n\nconst MAX_JSON_FORMAT_LENGTH = 10_000\n\nexport function tryJsonFormatContent(content: string): string {\n  if (content.length > MAX_JSON_FORMAT_LENGTH) {\n    return content\n  }\n  const allLines = content.split('\\n')\n  return allLines.map(tryFormatJson).join('\\n')\n}\n\n// Match http(s) URLs inside JSON string values. Conservative: no quotes,\n// no whitespace, no trailing comma/brace that'd be JSON structure.\nconst URL_IN_JSON = /https?:\\/\\/[^\\s\"'<>\\\\]+/g\n\nexport function linkifyUrlsInText(content: string): string {\n  return content.replace(URL_IN_JSON, url => createHyperlink(url))\n}\n\nexport function OutputLine({\n  content,\n  verbose,\n  isError,\n  isWarning,\n  linkifyUrls,\n}: {\n  content: string\n  verbose: boolean\n  isError?: boolean\n  isWarning?: boolean\n  linkifyUrls?: boolean\n}): React.ReactNode {\n  const { columns } = useTerminalSize()\n  // Context-based expansion for latest user shell output (from ! commands)\n  const expandShellOutput = useExpandShellOutput()\n  const inVirtualList = React.useContext(InVirtualListContext)\n\n  // Show full output if verbose mode OR if this is the latest user shell output\n  const shouldShowFull = verbose || expandShellOutput\n\n  const formattedContent = useMemo(() => {\n    let formatted = tryJsonFormatContent(content)\n    if (linkifyUrls) {\n      formatted = linkifyUrlsInText(formatted)\n    }\n    if (shouldShowFull) {\n      return stripUnderlineAnsi(formatted)\n    }\n    return stripUnderlineAnsi(\n      renderTruncatedContent(formatted, columns, inVirtualList),\n    )\n  }, [content, shouldShowFull, columns, linkifyUrls, inVirtualList])\n\n  const color = isError ? 'error' : isWarning ? 'warning' : undefined\n\n  return (\n    <MessageResponse>\n      <Text color={color}>\n        <Ansi>{formattedContent}</Ansi>\n      </Text>\n    </MessageResponse>\n  )\n}\n\n/**\n * Underline ANSI codes in particular tend to leak out for some reason. I wasn't\n * able to figure out why, or why emitting a reset ANSI code wasn't enough to\n * prevent them from leaking. I also didn't want to strip all ANSI codes with\n * stripAnsi(), because we used to do that and people complained about losing\n * all formatting. So we just strip the underline ANSI codes specifically.\n */\nexport function stripUnderlineAnsi(content: string): string {\n  return content.replace(\n    // eslint-disable-next-line no-control-regex\n    /\\u001b\\[([0-9]+;)*4(;[0-9]+)*m|\\u001b\\[4(;[0-9]+)*m|\\u001b\\[([0-9]+;)*4m/g,\n    '',\n  )\n}\n"],"mappings":"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","ignoreList":[]}