/ components / permissions / PermissionExplanation.tsx
PermissionExplanation.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import React, { Suspense, use, useState } from 'react';
  3  import { Box, Text } from '../../ink.js';
  4  import { useKeybinding } from '../../keybindings/useKeybinding.js';
  5  import { logEvent } from '../../services/analytics/index.js';
  6  import type { Message } from '../../types/message.js';
  7  import { generatePermissionExplanation, isPermissionExplainerEnabled, type PermissionExplanation as PermissionExplanationType, type RiskLevel } from '../../utils/permissions/permissionExplainer.js';
  8  import { ShimmerChar } from '../Spinner/ShimmerChar.js';
  9  import { useShimmerAnimation } from '../Spinner/useShimmerAnimation.js';
 10  const LOADING_MESSAGE = 'Loading explanation…';
 11  function 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  }
 41  function 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  }
 51  function 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  }
 61  type PermissionExplanationProps = {
 62    toolName: string;
 63    toolInput: unknown;
 64    toolDescription?: string;
 65    messages?: Message[];
 66  };
 67  type 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   */
 77  function 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   */
 92  export 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   */
153  function _temp(v) {
154    return !v;
155  }
156  function 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   */
246  export 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":[]}