/ components / permissions / PermissionRuleExplanation.tsx
PermissionRuleExplanation.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import { feature } from 'bun:bundle';
  3  import chalk from 'chalk';
  4  import React from 'react';
  5  import { Ansi, Box, Text } from '../../ink.js';
  6  import { useAppState } from '../../state/AppState.js';
  7  import type { PermissionDecision, PermissionDecisionReason } from '../../utils/permissions/PermissionResult.js';
  8  import { permissionRuleValueToString } from '../../utils/permissions/permissionRuleParser.js';
  9  import type { Theme } from '../../utils/theme.js';
 10  import ThemedText from '../design-system/ThemedText.js';
 11  export type PermissionRuleExplanationProps = {
 12    permissionResult: PermissionDecision;
 13    toolType: 'tool' | 'command' | 'edit' | 'read';
 14  };
 15  type DecisionReasonStrings = {
 16    reasonString: string;
 17    configString?: string;
 18    /** When set, reasonString is plain text rendered with this theme color instead of <Ansi>. */
 19    themeColor?: keyof Theme;
 20  };
 21  function stringsForDecisionReason(reason: PermissionDecisionReason | undefined, toolType: 'tool' | 'command' | 'edit' | 'read'): DecisionReasonStrings | null {
 22    if (!reason) {
 23      return null;
 24    }
 25    if ((feature('BASH_CLASSIFIER') || feature('TRANSCRIPT_CLASSIFIER')) && reason.type === 'classifier') {
 26      if (reason.classifier === 'auto-mode') {
 27        return {
 28          reasonString: `Auto mode classifier requires confirmation for this ${toolType}.\n${reason.reason}`,
 29          configString: undefined,
 30          themeColor: 'error'
 31        };
 32      }
 33      return {
 34        reasonString: `Classifier ${chalk.bold(reason.classifier)} requires confirmation for this ${toolType}.\n${reason.reason}`,
 35        configString: undefined
 36      };
 37    }
 38    switch (reason.type) {
 39      case 'rule':
 40        return {
 41          reasonString: `Permission rule ${chalk.bold(permissionRuleValueToString(reason.rule.ruleValue))} requires confirmation for this ${toolType}.`,
 42          configString: reason.rule.source === 'policySettings' ? undefined : '/permissions to update rules'
 43        };
 44      case 'hook':
 45        {
 46          const hookReasonString = reason.reason ? `:\n${reason.reason}` : '.';
 47          const sourceLabel = reason.hookSource ? ` ${chalk.dim(`[${reason.hookSource}]`)}` : '';
 48          return {
 49            reasonString: `Hook ${chalk.bold(reason.hookName)} requires confirmation for this ${toolType}${hookReasonString}${sourceLabel}`,
 50            configString: '/hooks to update'
 51          };
 52        }
 53      case 'safetyCheck':
 54      case 'other':
 55        return {
 56          reasonString: reason.reason,
 57          configString: undefined
 58        };
 59      case 'workingDir':
 60        return {
 61          reasonString: reason.reason,
 62          configString: '/permissions to update rules'
 63        };
 64      default:
 65        return null;
 66    }
 67  }
 68  export function PermissionRuleExplanation(t0) {
 69    const $ = _c(11);
 70    const {
 71      permissionResult,
 72      toolType
 73    } = t0;
 74    const permissionMode = useAppState(_temp);
 75    const t1 = permissionResult?.decisionReason;
 76    let t2;
 77    if ($[0] !== t1 || $[1] !== toolType) {
 78      t2 = stringsForDecisionReason(t1, toolType);
 79      $[0] = t1;
 80      $[1] = toolType;
 81      $[2] = t2;
 82    } else {
 83      t2 = $[2];
 84    }
 85    const strings = t2;
 86    if (!strings) {
 87      return null;
 88    }
 89    const themeColor = strings.themeColor ?? (permissionResult?.decisionReason?.type === "hook" && permissionMode === "auto" ? "warning" : undefined);
 90    let t3;
 91    if ($[3] !== strings.reasonString || $[4] !== themeColor) {
 92      t3 = themeColor ? <ThemedText color={themeColor}>{strings.reasonString}</ThemedText> : <Text><Ansi>{strings.reasonString}</Ansi></Text>;
 93      $[3] = strings.reasonString;
 94      $[4] = themeColor;
 95      $[5] = t3;
 96    } else {
 97      t3 = $[5];
 98    }
 99    let t4;
100    if ($[6] !== strings.configString) {
101      t4 = strings.configString && <Text dimColor={true}>{strings.configString}</Text>;
102      $[6] = strings.configString;
103      $[7] = t4;
104    } else {
105      t4 = $[7];
106    }
107    let t5;
108    if ($[8] !== t3 || $[9] !== t4) {
109      t5 = <Box marginBottom={1} flexDirection="column">{t3}{t4}</Box>;
110      $[8] = t3;
111      $[9] = t4;
112      $[10] = t5;
113    } else {
114      t5 = $[10];
115    }
116    return t5;
117  }
118  function _temp(s) {
119    return s.toolPermissionContext.mode;
120  }
121  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","chalk","React","Ansi","Box","Text","useAppState","PermissionDecision","PermissionDecisionReason","permissionRuleValueToString","Theme","ThemedText","PermissionRuleExplanationProps","permissionResult","toolType","DecisionReasonStrings","reasonString","configString","themeColor","stringsForDecisionReason","reason","type","classifier","undefined","bold","rule","ruleValue","source","hookReasonString","sourceLabel","hookSource","dim","hookName","PermissionRuleExplanation","t0","$","_c","permissionMode","_temp","t1","decisionReason","t2","strings","t3","t4","t5","s","toolPermissionContext","mode"],"sources":["PermissionRuleExplanation.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport chalk from 'chalk'\nimport React from 'react'\nimport { Ansi, Box, Text } from '../../ink.js'\nimport { useAppState } from '../../state/AppState.js'\nimport type {\n  PermissionDecision,\n  PermissionDecisionReason,\n} from '../../utils/permissions/PermissionResult.js'\nimport { permissionRuleValueToString } from '../../utils/permissions/permissionRuleParser.js'\nimport type { Theme } from '../../utils/theme.js'\nimport ThemedText from '../design-system/ThemedText.js'\n\nexport type PermissionRuleExplanationProps = {\n  permissionResult: PermissionDecision\n  toolType: 'tool' | 'command' | 'edit' | 'read'\n}\n\ntype DecisionReasonStrings = {\n  reasonString: string\n  configString?: string\n  /** When set, reasonString is plain text rendered with this theme color instead of <Ansi>. */\n  themeColor?: keyof Theme\n}\n\nfunction stringsForDecisionReason(\n  reason: PermissionDecisionReason | undefined,\n  toolType: 'tool' | 'command' | 'edit' | 'read',\n): DecisionReasonStrings | null {\n  if (!reason) {\n    return null\n  }\n  if (\n    (feature('BASH_CLASSIFIER') || feature('TRANSCRIPT_CLASSIFIER')) &&\n    reason.type === 'classifier'\n  ) {\n    if (reason.classifier === 'auto-mode') {\n      return {\n        reasonString: `Auto mode classifier requires confirmation for this ${toolType}.\\n${reason.reason}`,\n        configString: undefined,\n        themeColor: 'error',\n      }\n    }\n    return {\n      reasonString: `Classifier ${chalk.bold(reason.classifier)} requires confirmation for this ${toolType}.\\n${reason.reason}`,\n      configString: undefined,\n    }\n  }\n  switch (reason.type) {\n    case 'rule':\n      return {\n        reasonString: `Permission rule ${chalk.bold(\n          permissionRuleValueToString(reason.rule.ruleValue),\n        )} requires confirmation for this ${toolType}.`,\n        configString:\n          reason.rule.source === 'policySettings'\n            ? undefined\n            : '/permissions to update rules',\n      }\n    case 'hook': {\n      const hookReasonString = reason.reason ? `:\\n${reason.reason}` : '.'\n      const sourceLabel = reason.hookSource\n        ? ` ${chalk.dim(`[${reason.hookSource}]`)}`\n        : ''\n      return {\n        reasonString: `Hook ${chalk.bold(reason.hookName)} requires confirmation for this ${toolType}${hookReasonString}${sourceLabel}`,\n        configString: '/hooks to update',\n      }\n    }\n    case 'safetyCheck':\n    case 'other':\n      return {\n        reasonString: reason.reason,\n        configString: undefined,\n      }\n    case 'workingDir':\n      return {\n        reasonString: reason.reason,\n        configString: '/permissions to update rules',\n      }\n    default:\n      return null\n  }\n}\n\nexport function PermissionRuleExplanation({\n  permissionResult,\n  toolType,\n}: PermissionRuleExplanationProps): React.ReactNode {\n  const permissionMode = useAppState(s => s.toolPermissionContext.mode)\n  const strings = stringsForDecisionReason(\n    permissionResult?.decisionReason,\n    toolType,\n  )\n  if (!strings) {\n    return null\n  }\n\n  const themeColor =\n    strings.themeColor ??\n    (permissionResult?.decisionReason?.type === 'hook' &&\n    permissionMode === 'auto'\n      ? 'warning'\n      : undefined)\n\n  return (\n    <Box marginBottom={1} flexDirection=\"column\">\n      {themeColor ? (\n        <ThemedText color={themeColor}>{strings.reasonString}</ThemedText>\n      ) : (\n        <Text>\n          <Ansi>{strings.reasonString}</Ansi>\n        </Text>\n      )}\n      {strings.configString && <Text dimColor>{strings.configString}</Text>}\n    </Box>\n  )\n}\n"],"mappings":"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","ignoreList":[]}