/ components / ValidationErrorsList.tsx
ValidationErrorsList.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import setWith from 'lodash-es/setWith.js';
  3  import * as React from 'react';
  4  import { Box, Text, useTheme } from '../ink.js';
  5  import type { ValidationError } from '../utils/settings/validation.js';
  6  import { type TreeNode, treeify } from '../utils/treeify.js';
  7  
  8  /**
  9   * Builds a nested tree structure from dot-notation paths
 10   * Uses lodash setWith to avoid automatic array creation
 11   */
 12  function buildNestedTree(errors: ValidationError[]): TreeNode {
 13    const tree: TreeNode = {};
 14    errors.forEach(error => {
 15      if (!error.path) {
 16        // Root level error - use empty string as key
 17        tree[''] = error.message;
 18        return;
 19      }
 20  
 21      // Try to enhance the path with meaningful values
 22      const pathParts = error.path.split('.');
 23      let modifiedPath = error.path;
 24  
 25      // If we have an invalid value, try to make the path more readable
 26      if (error.invalidValue !== null && error.invalidValue !== undefined && pathParts.length > 0) {
 27        const newPathParts: string[] = [];
 28        for (let i = 0; i < pathParts.length; i++) {
 29          const part = pathParts[i];
 30          if (!part) continue;
 31          const numericPart = parseInt(part, 10);
 32  
 33          // If this is a numeric index and it's the last part where we have the invalid value
 34          if (!isNaN(numericPart) && i === pathParts.length - 1) {
 35            // Format the value for display
 36            let displayValue: string;
 37            if (typeof error.invalidValue === 'string') {
 38              displayValue = `"${error.invalidValue}"`;
 39            } else if (error.invalidValue === null) {
 40              displayValue = 'null';
 41            } else if (error.invalidValue === undefined) {
 42              displayValue = 'undefined';
 43            } else {
 44              displayValue = String(error.invalidValue);
 45            }
 46            newPathParts.push(displayValue);
 47          } else {
 48            // Keep other parts as-is
 49            newPathParts.push(part);
 50          }
 51        }
 52        modifiedPath = newPathParts.join('.');
 53      }
 54      setWith(tree, modifiedPath, error.message, Object);
 55    });
 56    return tree;
 57  }
 58  
 59  /**
 60   * Groups and displays validation errors using treeify with deduplication
 61   */
 62  export function ValidationErrorsList(t0) {
 63    const $ = _c(9);
 64    const {
 65      errors
 66    } = t0;
 67    const [themeName] = useTheme();
 68    if (errors.length === 0) {
 69      return null;
 70    }
 71    let T0;
 72    let t1;
 73    let t2;
 74    if ($[0] !== errors || $[1] !== themeName) {
 75      const errorsByFile = errors.reduce(_temp, {});
 76      const sortedFiles = Object.keys(errorsByFile).sort();
 77      T0 = Box;
 78      t1 = "column";
 79      t2 = sortedFiles.map(file_0 => {
 80        const fileErrors = errorsByFile[file_0] || [];
 81        fileErrors.sort(_temp2);
 82        const errorTree = buildNestedTree(fileErrors);
 83        const suggestionPairs = new Map();
 84        fileErrors.forEach(error_0 => {
 85          if (error_0.suggestion || error_0.docLink) {
 86            const key = `${error_0.suggestion || ""}|${error_0.docLink || ""}`;
 87            if (!suggestionPairs.has(key)) {
 88              suggestionPairs.set(key, {
 89                suggestion: error_0.suggestion,
 90                docLink: error_0.docLink
 91              });
 92            }
 93          }
 94        });
 95        const treeOutput = treeify(errorTree, {
 96          showValues: true,
 97          themeName,
 98          treeCharColors: {
 99            treeChar: "inactive",
100            key: "text",
101            value: "inactive"
102          }
103        });
104        return <Box key={file_0} flexDirection="column"><Text>{file_0}</Text><Box marginLeft={1}><Text dimColor={true}>{treeOutput}</Text></Box>{suggestionPairs.size > 0 && <Box flexDirection="column" marginTop={1}>{Array.from(suggestionPairs.values()).map(_temp3)}</Box>}</Box>;
105      });
106      $[0] = errors;
107      $[1] = themeName;
108      $[2] = T0;
109      $[3] = t1;
110      $[4] = t2;
111    } else {
112      T0 = $[2];
113      t1 = $[3];
114      t2 = $[4];
115    }
116    let t3;
117    if ($[5] !== T0 || $[6] !== t1 || $[7] !== t2) {
118      t3 = <T0 flexDirection={t1}>{t2}</T0>;
119      $[5] = T0;
120      $[6] = t1;
121      $[7] = t2;
122      $[8] = t3;
123    } else {
124      t3 = $[8];
125    }
126    return t3;
127  }
128  function _temp3(pair, index) {
129    return <Box key={`suggestion-pair-${index}`} flexDirection="column" marginBottom={1}>{pair.suggestion && <Text dimColor={true} wrap="wrap">{pair.suggestion}</Text>}{pair.docLink && <Text dimColor={true} wrap="wrap">Learn more: {pair.docLink}</Text>}</Box>;
130  }
131  function _temp2(a, b) {
132    if (!a.path && b.path) {
133      return -1;
134    }
135    if (a.path && !b.path) {
136      return 1;
137    }
138    return (a.path || "").localeCompare(b.path || "");
139  }
140  function _temp(acc, error) {
141    const file = error.file || "(file not specified)";
142    if (!acc[file]) {
143      acc[file] = [];
144    }
145    acc[file].push(error);
146    return acc;
147  }
148  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["setWith","React","Box","Text","useTheme","ValidationError","TreeNode","treeify","buildNestedTree","errors","tree","forEach","error","path","message","pathParts","split","modifiedPath","invalidValue","undefined","length","newPathParts","i","part","numericPart","parseInt","isNaN","displayValue","String","push","join","Object","ValidationErrorsList","t0","$","_c","themeName","T0","t1","t2","errorsByFile","reduce","_temp","sortedFiles","keys","sort","map","file_0","fileErrors","file","_temp2","errorTree","suggestionPairs","Map","error_0","suggestion","docLink","key","has","set","treeOutput","showValues","treeCharColors","treeChar","value","size","Array","from","values","_temp3","t3","pair","index","a","b","localeCompare","acc"],"sources":["ValidationErrorsList.tsx"],"sourcesContent":["import setWith from 'lodash-es/setWith.js'\nimport * as React from 'react'\nimport { Box, Text, useTheme } from '../ink.js'\nimport type { ValidationError } from '../utils/settings/validation.js'\nimport { type TreeNode, treeify } from '../utils/treeify.js'\n\n/**\n * Builds a nested tree structure from dot-notation paths\n * Uses lodash setWith to avoid automatic array creation\n */\nfunction buildNestedTree(errors: ValidationError[]): TreeNode {\n  const tree: TreeNode = {}\n\n  errors.forEach(error => {\n    if (!error.path) {\n      // Root level error - use empty string as key\n      tree[''] = error.message\n      return\n    }\n\n    // Try to enhance the path with meaningful values\n    const pathParts = error.path.split('.')\n    let modifiedPath = error.path\n\n    // If we have an invalid value, try to make the path more readable\n    if (\n      error.invalidValue !== null &&\n      error.invalidValue !== undefined &&\n      pathParts.length > 0\n    ) {\n      const newPathParts: string[] = []\n\n      for (let i = 0; i < pathParts.length; i++) {\n        const part = pathParts[i]\n        if (!part) continue\n\n        const numericPart = parseInt(part, 10)\n\n        // If this is a numeric index and it's the last part where we have the invalid value\n        if (!isNaN(numericPart) && i === pathParts.length - 1) {\n          // Format the value for display\n          let displayValue: string\n          if (typeof error.invalidValue === 'string') {\n            displayValue = `\"${error.invalidValue}\"`\n          } else if (error.invalidValue === null) {\n            displayValue = 'null'\n          } else if (error.invalidValue === undefined) {\n            displayValue = 'undefined'\n          } else {\n            displayValue = String(error.invalidValue)\n          }\n\n          newPathParts.push(displayValue)\n        } else {\n          // Keep other parts as-is\n          newPathParts.push(part)\n        }\n      }\n\n      modifiedPath = newPathParts.join('.')\n    }\n\n    setWith(tree, modifiedPath, error.message, Object)\n  })\n\n  return tree\n}\n\n/**\n * Groups and displays validation errors using treeify with deduplication\n */\nexport function ValidationErrorsList({\n  errors,\n}: {\n  errors: ValidationError[]\n}): React.ReactNode {\n  const [themeName] = useTheme()\n\n  if (errors.length === 0) {\n    return null\n  }\n\n  // Group errors by file\n  const errorsByFile = errors.reduce<Record<string, ValidationError[]>>(\n    (acc, error) => {\n      const file = error.file || '(file not specified)'\n      if (!acc[file]) {\n        acc[file] = []\n      }\n      acc[file]!.push(error)\n      return acc\n    },\n    {},\n  )\n\n  // Sort files alphabetically\n  const sortedFiles = Object.keys(errorsByFile).sort()\n\n  return (\n    <Box flexDirection=\"column\">\n      {sortedFiles.map(file => {\n        const fileErrors = errorsByFile[file] || []\n\n        // Sort errors by path\n        fileErrors.sort((a, b) => {\n          if (!a.path && b.path) return -1\n          if (a.path && !b.path) return 1\n          return (a.path || '').localeCompare(b.path || '')\n        })\n\n        // Build nested tree structure from error paths\n        const errorTree = buildNestedTree(fileErrors)\n\n        // Collect unique suggestion+docLink pairs\n        const suggestionPairs = new Map<\n          string,\n          { suggestion?: string; docLink?: string }\n        >()\n\n        fileErrors.forEach(error => {\n          if (error.suggestion || error.docLink) {\n            // Create a key from suggestion+docLink combination\n            const key = `${error.suggestion || ''}|${error.docLink || ''}`\n            if (!suggestionPairs.has(key)) {\n              suggestionPairs.set(key, {\n                suggestion: error.suggestion,\n                docLink: error.docLink,\n              })\n            }\n          }\n        })\n\n        // Render the tree\n        const treeOutput = treeify(errorTree, {\n          showValues: true,\n          themeName,\n          treeCharColors: {\n            treeChar: 'inactive',\n            key: 'text',\n            value: 'inactive',\n          },\n        })\n\n        return (\n          <Box key={file} flexDirection=\"column\">\n            <Text>{file}</Text>\n            <Box marginLeft={1}>\n              <Text dimColor>{treeOutput}</Text>\n            </Box>\n            {/* Display unique suggestion+docLink pairs */}\n            {suggestionPairs.size > 0 && (\n              <Box flexDirection=\"column\" marginTop={1}>\n                {Array.from(suggestionPairs.values()).map((pair, index) => (\n                  <Box\n                    key={`suggestion-pair-${index}`}\n                    flexDirection=\"column\"\n                    marginBottom={1}\n                  >\n                    {pair.suggestion && (\n                      <Text dimColor wrap=\"wrap\">\n                        {pair.suggestion}\n                      </Text>\n                    )}\n                    {pair.docLink && (\n                      <Text dimColor wrap=\"wrap\">\n                        Learn more: {pair.docLink}\n                      </Text>\n                    )}\n                  </Box>\n                ))}\n              </Box>\n            )}\n          </Box>\n        )\n      })}\n    </Box>\n  )\n}\n"],"mappings":"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","ignoreList":[]}