/ components / messages / UserToolResultMessage / UserToolSuccessMessage.tsx
UserToolSuccessMessage.tsx
  1  import { feature } from 'bun:bundle';
  2  import figures from 'figures';
  3  import * as React from 'react';
  4  import { SentryErrorBoundary } from 'src/components/SentryErrorBoundary.js';
  5  import { Box, Text, useTheme } from '../../../ink.js';
  6  import { useAppState } from '../../../state/AppState.js';
  7  import { filterToolProgressMessages, type Tool, type Tools } from '../../../Tool.js';
  8  import type { NormalizedUserMessage, ProgressMessage } from '../../../types/message.js';
  9  import { deleteClassifierApproval, getClassifierApproval, getYoloClassifierApproval } from '../../../utils/classifierApprovals.js';
 10  import type { buildMessageLookups } from '../../../utils/messages.js';
 11  import { MessageResponse } from '../../MessageResponse.js';
 12  import { HookProgressMessage } from '../HookProgressMessage.js';
 13  type Props = {
 14    message: NormalizedUserMessage;
 15    lookups: ReturnType<typeof buildMessageLookups>;
 16    toolUseID: string;
 17    progressMessagesForMessage: ProgressMessage[];
 18    style?: 'condensed';
 19    tool?: Tool;
 20    tools: Tools;
 21    verbose: boolean;
 22    width: number | string;
 23    isTranscriptMode?: boolean;
 24  };
 25  export function UserToolSuccessMessage({
 26    message,
 27    lookups,
 28    toolUseID,
 29    progressMessagesForMessage,
 30    style,
 31    tool,
 32    tools,
 33    verbose,
 34    width,
 35    isTranscriptMode
 36  }: Props): React.ReactNode {
 37    const [theme] = useTheme();
 38    // Hook stays inside feature() ternary so external builds don't pay a
 39    // per-scrollback-message store subscription — same pattern as
 40    // UserPromptMessage.tsx.
 41    const isBriefOnly = feature('KAIROS') || feature('KAIROS_BRIEF') ?
 42    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 43    useAppState(s => s.isBriefOnly) : false;
 44  
 45    // Capture classifier approval once on mount, then delete from Map to prevent linear growth.
 46    // useState lazy initializer ensures the value persists across re-renders.
 47    const [classifierRule] = React.useState(() => getClassifierApproval(toolUseID));
 48    const [yoloReason] = React.useState(() => getYoloClassifierApproval(toolUseID));
 49    React.useEffect(() => {
 50      deleteClassifierApproval(toolUseID);
 51    }, [toolUseID]);
 52    if (!message.toolUseResult || !tool) {
 53      return null;
 54    }
 55  
 56    // Resumed transcripts deserialize toolUseResult via raw JSON.parse with no
 57    // validation (parseJSONL). A partial/corrupt/old-format result crashes
 58    // renderToolResultMessage on first field access (anthropics/claude-code#39817).
 59    // Validate against outputSchema before rendering — mirrors CollapsedReadSearchContent.
 60    const parsedOutput = tool.outputSchema?.safeParse(message.toolUseResult);
 61    if (parsedOutput && !parsedOutput.success) {
 62      return null;
 63    }
 64    const toolResult = parsedOutput?.data ?? message.toolUseResult;
 65    const renderedMessage = tool.renderToolResultMessage?.(toolResult as never, filterToolProgressMessages(progressMessagesForMessage), {
 66      style,
 67      theme,
 68      tools,
 69      verbose,
 70      isTranscriptMode,
 71      isBriefOnly,
 72      input: lookups.toolUseByToolUseID.get(toolUseID)?.input
 73    }) ?? null;
 74  
 75    // Don't render anything if the tool result message is null
 76    if (renderedMessage === null) {
 77      return null;
 78    }
 79  
 80    // Tools that return '' from userFacingName opt out of tool chrome and
 81    // render like plain assistant text. Skip the tool-result width constraint
 82    // so MarkdownTable's SAFETY_MARGIN=4 (tuned for the assistant-text 2-col
 83    // dot gutter) holds — otherwise tables wrap their box-drawing chars.
 84    const rendersAsAssistantText = tool.userFacingName(undefined) === '';
 85    return <Box flexDirection="column">
 86        <Box flexDirection="column" width={rendersAsAssistantText ? undefined : width}>
 87          {renderedMessage}
 88          {feature('BASH_CLASSIFIER') ? classifierRule && <MessageResponse height={1}>
 89                  <Text dimColor>
 90                    <Text color="success">{figures.tick}</Text>
 91                    {' Auto-approved \u00b7 matched '}
 92                    {`"${classifierRule}"`}
 93                  </Text>
 94                </MessageResponse> : null}
 95          {feature('TRANSCRIPT_CLASSIFIER') ? yoloReason && <MessageResponse height={1}>
 96                  <Text dimColor>Allowed by auto mode classifier</Text>
 97                </MessageResponse> : null}
 98        </Box>
 99        <SentryErrorBoundary>
100          <HookProgressMessage hookEvent="PostToolUse" lookups={lookups} toolUseID={toolUseID} verbose={verbose} isTranscriptMode={isTranscriptMode} />
101        </SentryErrorBoundary>
102      </Box>;
103  }
104  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","figures","React","SentryErrorBoundary","Box","Text","useTheme","useAppState","filterToolProgressMessages","Tool","Tools","NormalizedUserMessage","ProgressMessage","deleteClassifierApproval","getClassifierApproval","getYoloClassifierApproval","buildMessageLookups","MessageResponse","HookProgressMessage","Props","message","lookups","ReturnType","toolUseID","progressMessagesForMessage","style","tool","tools","verbose","width","isTranscriptMode","UserToolSuccessMessage","ReactNode","theme","isBriefOnly","s","classifierRule","useState","yoloReason","useEffect","toolUseResult","parsedOutput","outputSchema","safeParse","success","toolResult","data","renderedMessage","renderToolResultMessage","input","toolUseByToolUseID","get","rendersAsAssistantText","userFacingName","undefined","tick"],"sources":["UserToolSuccessMessage.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport figures from 'figures'\nimport * as React from 'react'\nimport { SentryErrorBoundary } from 'src/components/SentryErrorBoundary.js'\nimport { Box, Text, useTheme } from '../../../ink.js'\nimport { useAppState } from '../../../state/AppState.js'\nimport {\n  filterToolProgressMessages,\n  type Tool,\n  type Tools,\n} from '../../../Tool.js'\nimport type {\n  NormalizedUserMessage,\n  ProgressMessage,\n} from '../../../types/message.js'\nimport {\n  deleteClassifierApproval,\n  getClassifierApproval,\n  getYoloClassifierApproval,\n} from '../../../utils/classifierApprovals.js'\nimport type { buildMessageLookups } from '../../../utils/messages.js'\nimport { MessageResponse } from '../../MessageResponse.js'\nimport { HookProgressMessage } from '../HookProgressMessage.js'\n\ntype Props = {\n  message: NormalizedUserMessage\n  lookups: ReturnType<typeof buildMessageLookups>\n  toolUseID: string\n  progressMessagesForMessage: ProgressMessage[]\n  style?: 'condensed'\n  tool?: Tool\n  tools: Tools\n  verbose: boolean\n  width: number | string\n  isTranscriptMode?: boolean\n}\n\nexport function UserToolSuccessMessage({\n  message,\n  lookups,\n  toolUseID,\n  progressMessagesForMessage,\n  style,\n  tool,\n  tools,\n  verbose,\n  width,\n  isTranscriptMode,\n}: Props): React.ReactNode {\n  const [theme] = useTheme()\n  // Hook stays inside feature() ternary so external builds don't pay a\n  // per-scrollback-message store subscription — same pattern as\n  // UserPromptMessage.tsx.\n  const isBriefOnly =\n    feature('KAIROS') || feature('KAIROS_BRIEF')\n      ? // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant\n        useAppState(s => s.isBriefOnly)\n      : false\n\n  // Capture classifier approval once on mount, then delete from Map to prevent linear growth.\n  // useState lazy initializer ensures the value persists across re-renders.\n  const [classifierRule] = React.useState(() =>\n    getClassifierApproval(toolUseID),\n  )\n  const [yoloReason] = React.useState(() =>\n    getYoloClassifierApproval(toolUseID),\n  )\n  React.useEffect(() => {\n    deleteClassifierApproval(toolUseID)\n  }, [toolUseID])\n\n  if (!message.toolUseResult || !tool) {\n    return null\n  }\n\n  // Resumed transcripts deserialize toolUseResult via raw JSON.parse with no\n  // validation (parseJSONL). A partial/corrupt/old-format result crashes\n  // renderToolResultMessage on first field access (anthropics/claude-code#39817).\n  // Validate against outputSchema before rendering — mirrors CollapsedReadSearchContent.\n  const parsedOutput = tool.outputSchema?.safeParse(message.toolUseResult)\n  if (parsedOutput && !parsedOutput.success) {\n    return null\n  }\n  const toolResult = parsedOutput?.data ?? message.toolUseResult\n\n  const renderedMessage =\n    tool.renderToolResultMessage?.(\n      toolResult as never,\n      filterToolProgressMessages(progressMessagesForMessage),\n      {\n        style,\n        theme,\n        tools,\n        verbose,\n        isTranscriptMode,\n        isBriefOnly,\n        input: lookups.toolUseByToolUseID.get(toolUseID)?.input,\n      },\n    ) ?? null\n\n  // Don't render anything if the tool result message is null\n  if (renderedMessage === null) {\n    return null\n  }\n\n  // Tools that return '' from userFacingName opt out of tool chrome and\n  // render like plain assistant text. Skip the tool-result width constraint\n  // so MarkdownTable's SAFETY_MARGIN=4 (tuned for the assistant-text 2-col\n  // dot gutter) holds — otherwise tables wrap their box-drawing chars.\n  const rendersAsAssistantText = tool.userFacingName(undefined) === ''\n\n  return (\n    <Box flexDirection=\"column\">\n      <Box\n        flexDirection=\"column\"\n        width={rendersAsAssistantText ? undefined : width}\n      >\n        {renderedMessage}\n        {feature('BASH_CLASSIFIER')\n          ? classifierRule && (\n              <MessageResponse height={1}>\n                <Text dimColor>\n                  <Text color=\"success\">{figures.tick}</Text>\n                  {' Auto-approved \\u00b7 matched '}\n                  {`\"${classifierRule}\"`}\n                </Text>\n              </MessageResponse>\n            )\n          : null}\n        {feature('TRANSCRIPT_CLASSIFIER')\n          ? yoloReason && (\n              <MessageResponse height={1}>\n                <Text dimColor>Allowed by auto mode classifier</Text>\n              </MessageResponse>\n            )\n          : null}\n      </Box>\n      <SentryErrorBoundary>\n        <HookProgressMessage\n          hookEvent=\"PostToolUse\"\n          lookups={lookups}\n          toolUseID={toolUseID}\n          verbose={verbose}\n          isTranscriptMode={isTranscriptMode}\n        />\n      </SentryErrorBoundary>\n    </Box>\n  )\n}\n"],"mappings":"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","ignoreList":[]}