source dump of claude code
at main 272 lines 24 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import React, { Suspense, use, useState } from 'react'; 3import { Box, Text } from '../../ink.js'; 4import { useKeybinding } from '../../keybindings/useKeybinding.js'; 5import { logEvent } from '../../services/analytics/index.js'; 6import type { Message } from '../../types/message.js'; 7import { generatePermissionExplanation, isPermissionExplainerEnabled, type PermissionExplanation as PermissionExplanationType, type RiskLevel } from '../../utils/permissions/permissionExplainer.js'; 8import { ShimmerChar } from '../Spinner/ShimmerChar.js'; 9import { useShimmerAnimation } from '../Spinner/useShimmerAnimation.js'; 10const LOADING_MESSAGE = 'Loading explanation…'; 11function ShimmerLoadingText() { 12 const $ = _c(7); 13 const [ref, glimmerIndex] = useShimmerAnimation("responding", LOADING_MESSAGE, false); 14 let t0; 15 if ($[0] !== glimmerIndex) { 16 t0 = LOADING_MESSAGE.split("").map((char, index) => <ShimmerChar key={index} char={char} index={index} glimmerIndex={glimmerIndex} messageColor="inactive" shimmerColor="text" />); 17 $[0] = glimmerIndex; 18 $[1] = t0; 19 } else { 20 t0 = $[1]; 21 } 22 let t1; 23 if ($[2] !== t0) { 24 t1 = <Text>{t0}</Text>; 25 $[2] = t0; 26 $[3] = t1; 27 } else { 28 t1 = $[3]; 29 } 30 let t2; 31 if ($[4] !== ref || $[5] !== t1) { 32 t2 = <Box ref={ref}>{t1}</Box>; 33 $[4] = ref; 34 $[5] = t1; 35 $[6] = t2; 36 } else { 37 t2 = $[6]; 38 } 39 return t2; 40} 41function getRiskColor(riskLevel: RiskLevel): 'success' | 'warning' | 'error' { 42 switch (riskLevel) { 43 case 'LOW': 44 return 'success'; 45 case 'MEDIUM': 46 return 'warning'; 47 case 'HIGH': 48 return 'error'; 49 } 50} 51function getRiskLabel(riskLevel: RiskLevel): string { 52 switch (riskLevel) { 53 case 'LOW': 54 return 'Low risk'; 55 case 'MEDIUM': 56 return 'Med risk'; 57 case 'HIGH': 58 return 'High risk'; 59 } 60} 61type PermissionExplanationProps = { 62 toolName: string; 63 toolInput: unknown; 64 toolDescription?: string; 65 messages?: Message[]; 66}; 67type ExplainerState = { 68 visible: boolean; 69 enabled: boolean; 70 promise: Promise<PermissionExplanationType | null> | null; 71}; 72 73/** 74 * Creates an explanation promise that never rejects. 75 * Errors are caught and returned as null. 76 */ 77function createExplanationPromise(props: PermissionExplanationProps): Promise<PermissionExplanationType | null> { 78 return generatePermissionExplanation({ 79 toolName: props.toolName, 80 toolInput: props.toolInput, 81 toolDescription: props.toolDescription, 82 messages: props.messages, 83 signal: new AbortController().signal // Won't abort - request is fast enough 84 }).catch(() => null); 85} 86 87/** 88 * Hook that manages the permission explainer state. 89 * Creates the fetch promise lazily (only when user hits Ctrl+E) 90 * to avoid consuming tokens for explanations users never view. 91 */ 92export function usePermissionExplainerUI(props) { 93 const $ = _c(9); 94 let t0; 95 if ($[0] === Symbol.for("react.memo_cache_sentinel")) { 96 t0 = isPermissionExplainerEnabled(); 97 $[0] = t0; 98 } else { 99 t0 = $[0]; 100 } 101 const enabled = t0; 102 const [visible, setVisible] = useState(false); 103 const [promise, setPromise] = useState(null); 104 let t1; 105 if ($[1] !== promise || $[2] !== props || $[3] !== visible) { 106 t1 = () => { 107 if (!visible) { 108 logEvent("tengu_permission_explainer_shortcut_used", {}); 109 if (!promise) { 110 setPromise(createExplanationPromise(props)); 111 } 112 } 113 setVisible(_temp); 114 }; 115 $[1] = promise; 116 $[2] = props; 117 $[3] = visible; 118 $[4] = t1; 119 } else { 120 t1 = $[4]; 121 } 122 let t2; 123 if ($[5] === Symbol.for("react.memo_cache_sentinel")) { 124 t2 = { 125 context: "Confirmation", 126 isActive: enabled 127 }; 128 $[5] = t2; 129 } else { 130 t2 = $[5]; 131 } 132 useKeybinding("confirm:toggleExplanation", t1, t2); 133 let t3; 134 if ($[6] !== promise || $[7] !== visible) { 135 t3 = { 136 visible, 137 enabled, 138 promise 139 }; 140 $[6] = promise; 141 $[7] = visible; 142 $[8] = t3; 143 } else { 144 t3 = $[8]; 145 } 146 return t3; 147} 148 149/** 150 * Inner component that uses React 19's use() to read the promise. 151 * Suspends while loading, returns null on error. 152 */ 153function _temp(v) { 154 return !v; 155} 156function ExplanationResult(t0) { 157 const $ = _c(21); 158 const { 159 promise 160 } = t0; 161 const explanation = use(promise); 162 if (!explanation) { 163 let t1; 164 if ($[0] === Symbol.for("react.memo_cache_sentinel")) { 165 t1 = <Box marginTop={1}><Text dimColor={true}>Explanation unavailable</Text></Box>; 166 $[0] = t1; 167 } else { 168 t1 = $[0]; 169 } 170 return t1; 171 } 172 let t1; 173 if ($[1] !== explanation.explanation) { 174 t1 = <Text>{explanation.explanation}</Text>; 175 $[1] = explanation.explanation; 176 $[2] = t1; 177 } else { 178 t1 = $[2]; 179 } 180 let t2; 181 if ($[3] !== explanation.reasoning) { 182 t2 = <Box marginTop={1}><Text>{explanation.reasoning}</Text></Box>; 183 $[3] = explanation.reasoning; 184 $[4] = t2; 185 } else { 186 t2 = $[4]; 187 } 188 let t3; 189 if ($[5] !== explanation.riskLevel) { 190 t3 = getRiskColor(explanation.riskLevel); 191 $[5] = explanation.riskLevel; 192 $[6] = t3; 193 } else { 194 t3 = $[6]; 195 } 196 let t4; 197 if ($[7] !== explanation.riskLevel) { 198 t4 = getRiskLabel(explanation.riskLevel); 199 $[7] = explanation.riskLevel; 200 $[8] = t4; 201 } else { 202 t4 = $[8]; 203 } 204 let t5; 205 if ($[9] !== t3 || $[10] !== t4) { 206 t5 = <Text color={t3}>{t4}:</Text>; 207 $[9] = t3; 208 $[10] = t4; 209 $[11] = t5; 210 } else { 211 t5 = $[11]; 212 } 213 let t6; 214 if ($[12] !== explanation.risk) { 215 t6 = <Text> {explanation.risk}</Text>; 216 $[12] = explanation.risk; 217 $[13] = t6; 218 } else { 219 t6 = $[13]; 220 } 221 let t7; 222 if ($[14] !== t5 || $[15] !== t6) { 223 t7 = <Box marginTop={1}><Text>{t5}{t6}</Text></Box>; 224 $[14] = t5; 225 $[15] = t6; 226 $[16] = t7; 227 } else { 228 t7 = $[16]; 229 } 230 let t8; 231 if ($[17] !== t1 || $[18] !== t2 || $[19] !== t7) { 232 t8 = <Box flexDirection="column" marginTop={1}>{t1}{t2}{t7}</Box>; 233 $[17] = t1; 234 $[18] = t2; 235 $[19] = t7; 236 $[20] = t8; 237 } else { 238 t8 = $[20]; 239 } 240 return t8; 241} 242 243/** 244 * Content component - shows loading (via Suspense) or explanation when visible 245 */ 246export function PermissionExplainerContent(t0) { 247 const $ = _c(3); 248 const { 249 visible, 250 promise 251 } = t0; 252 if (!visible || !promise) { 253 return null; 254 } 255 let t1; 256 if ($[0] === Symbol.for("react.memo_cache_sentinel")) { 257 t1 = <Box marginTop={1}><ShimmerLoadingText /></Box>; 258 $[0] = t1; 259 } else { 260 t1 = $[0]; 261 } 262 let t2; 263 if ($[1] !== promise) { 264 t2 = <Suspense fallback={t1}><ExplanationResult promise={promise} /></Suspense>; 265 $[1] = promise; 266 $[2] = t2; 267 } else { 268 t2 = $[2]; 269 } 270 return t2; 271} 272//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","Suspense","use","useState","Box","Text","useKeybinding","logEvent","Message","generatePermissionExplanation","isPermissionExplainerEnabled","PermissionExplanation","PermissionExplanationType","RiskLevel","ShimmerChar","useShimmerAnimation","LOADING_MESSAGE","ShimmerLoadingText","$","_c","ref","glimmerIndex","t0","split","map","char","index","t1","t2","getRiskColor","riskLevel","getRiskLabel","PermissionExplanationProps","toolName","toolInput","toolDescription","messages","ExplainerState","visible","enabled","promise","Promise","createExplanationPromise","props","signal","AbortController","catch","usePermissionExplainerUI","Symbol","for","setVisible","setPromise","_temp","context","isActive","t3","v","ExplanationResult","explanation","reasoning","t4","t5","t6","risk","t7","t8","PermissionExplainerContent"],"sources":["PermissionExplanation.tsx"],"sourcesContent":["import React, { Suspense, use, useState } from 'react'\nimport { Box, Text } from '../../ink.js'\nimport { useKeybinding } from '../../keybindings/useKeybinding.js'\nimport { logEvent } from '../../services/analytics/index.js'\nimport type { Message } from '../../types/message.js'\nimport {\n  generatePermissionExplanation,\n  isPermissionExplainerEnabled,\n  type PermissionExplanation as PermissionExplanationType,\n  type RiskLevel,\n} from '../../utils/permissions/permissionExplainer.js'\nimport { ShimmerChar } from '../Spinner/ShimmerChar.js'\nimport { useShimmerAnimation } from '../Spinner/useShimmerAnimation.js'\n\nconst LOADING_MESSAGE = 'Loading explanation…'\n\nfunction ShimmerLoadingText(): React.ReactNode {\n  const [ref, glimmerIndex] = useShimmerAnimation(\n    'responding',\n    LOADING_MESSAGE,\n    false,\n  )\n\n  return (\n    <Box ref={ref}>\n      <Text>\n        {LOADING_MESSAGE.split('').map((char, index) => (\n          <ShimmerChar\n            key={index}\n            char={char}\n            index={index}\n            glimmerIndex={glimmerIndex}\n            messageColor=\"inactive\"\n            shimmerColor=\"text\"\n          />\n        ))}\n      </Text>\n    </Box>\n  )\n}\n\nfunction getRiskColor(riskLevel: RiskLevel): 'success' | 'warning' | 'error' {\n  switch (riskLevel) {\n    case 'LOW':\n      return 'success'\n    case 'MEDIUM':\n      return 'warning'\n    case 'HIGH':\n      return 'error'\n  }\n}\n\nfunction getRiskLabel(riskLevel: RiskLevel): string {\n  switch (riskLevel) {\n    case 'LOW':\n      return 'Low risk'\n    case 'MEDIUM':\n      return 'Med risk'\n    case 'HIGH':\n      return 'High risk'\n  }\n}\n\ntype PermissionExplanationProps = {\n  toolName: string\n  toolInput: unknown\n  toolDescription?: string\n  messages?: Message[]\n}\n\ntype ExplainerState = {\n  visible: boolean\n  enabled: boolean\n  promise: Promise<PermissionExplanationType | null> | null\n}\n\n/**\n * Creates an explanation promise that never rejects.\n * Errors are caught and returned as null.\n */\nfunction createExplanationPromise(\n  props: PermissionExplanationProps,\n): Promise<PermissionExplanationType | null> {\n  return generatePermissionExplanation({\n    toolName: props.toolName,\n    toolInput: props.toolInput,\n    toolDescription: props.toolDescription,\n    messages: props.messages,\n    signal: new AbortController().signal, // Won't abort - request is fast enough\n  }).catch(() => null)\n}\n\n/**\n * Hook that manages the permission explainer state.\n * Creates the fetch promise lazily (only when user hits Ctrl+E)\n * to avoid consuming tokens for explanations users never view.\n */\nexport function usePermissionExplainerUI(\n  props: PermissionExplanationProps,\n): ExplainerState {\n  const enabled = isPermissionExplainerEnabled()\n  const [visible, setVisible] = useState(false)\n  const [promise, setPromise] =\n    useState<Promise<PermissionExplanationType | null> | null>(null)\n\n  // Use keybinding for ctrl+e toggle (configurable via keybindings.json)\n  useKeybinding(\n    'confirm:toggleExplanation',\n    () => {\n      if (!visible) {\n        logEvent('tengu_permission_explainer_shortcut_used', {})\n        // Only create the promise on first toggle (lazy loading)\n        if (!promise) {\n          setPromise(createExplanationPromise(props))\n        }\n      }\n      setVisible(v => !v)\n    },\n    { context: 'Confirmation', isActive: enabled },\n  )\n\n  return { visible, enabled, promise }\n}\n\n/**\n * Inner component that uses React 19's use() to read the promise.\n * Suspends while loading, returns null on error.\n */\nfunction ExplanationResult({\n  promise,\n}: {\n  promise: Promise<PermissionExplanationType | null>\n}): React.ReactNode {\n  const explanation = use(promise)\n\n  if (!explanation) {\n    return (\n      <Box marginTop={1}>\n        <Text dimColor>Explanation unavailable</Text>\n      </Box>\n    )\n  }\n\n  return (\n    <Box flexDirection=\"column\" marginTop={1}>\n      <Text>{explanation.explanation}</Text>\n      <Box marginTop={1}>\n        <Text>{explanation.reasoning}</Text>\n      </Box>\n      <Box marginTop={1}>\n        <Text>\n          <Text color={getRiskColor(explanation.riskLevel)}>\n            {getRiskLabel(explanation.riskLevel)}:\n          </Text>\n          <Text> {explanation.risk}</Text>\n        </Text>\n      </Box>\n    </Box>\n  )\n}\n\n/**\n * Content component - shows loading (via Suspense) or explanation when visible\n */\nexport function PermissionExplainerContent({\n  visible,\n  promise,\n}: {\n  visible: boolean\n  promise: Promise<PermissionExplanationType | null> | null\n}): React.ReactNode {\n  if (!visible || !promise) {\n    return null\n  }\n\n  return (\n    <Suspense\n      fallback={\n        <Box marginTop={1}>\n          <ShimmerLoadingText />\n        </Box>\n      }\n    >\n      <ExplanationResult promise={promise} />\n    </Suspense>\n  )\n}\n"],"mappings":"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","ignoreList":[]}