Filemedium importancesource

ValidationErrorsList.tsx

components/ValidationErrorsList.tsx

148
Lines
19567
Bytes
1
Exports
6
Imports
10
Keywords

What this is

This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.

Beginner explanation

This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.

How it is used

Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.

Expert explanation

Architecturally, this file intersects with ui-flow. It contains 148 lines, 6 detected imports, and 1 detected exports.

Important relationships

Detected exports

  • ValidationErrorsList

Keywords

pathtexterrorsinvalidvalueerror_0displayvalueelsesuggestiondoclinkfile

Detected imports

  • react/compiler-runtime
  • lodash-es/setWith.js
  • react
  • ../ink.js
  • ../utils/settings/validation.js
  • ../utils/treeify.js

Source notes

This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.

Open parent directory

Full source

import { c as _c } from "react/compiler-runtime";
import setWith from 'lodash-es/setWith.js';
import * as React from 'react';
import { Box, Text, useTheme } from '../ink.js';
import type { ValidationError } from '../utils/settings/validation.js';
import { type TreeNode, treeify } from '../utils/treeify.js';

/**
 * Builds a nested tree structure from dot-notation paths
 * Uses lodash setWith to avoid automatic array creation
 */
function buildNestedTree(errors: ValidationError[]): TreeNode {
  const tree: TreeNode = {};
  errors.forEach(error => {
    if (!error.path) {
      // Root level error - use empty string as key
      tree[''] = error.message;
      return;
    }

    // Try to enhance the path with meaningful values
    const pathParts = error.path.split('.');
    let modifiedPath = error.path;

    // If we have an invalid value, try to make the path more readable
    if (error.invalidValue !== null && error.invalidValue !== undefined && pathParts.length > 0) {
      const newPathParts: string[] = [];
      for (let i = 0; i < pathParts.length; i++) {
        const part = pathParts[i];
        if (!part) continue;
        const numericPart = parseInt(part, 10);

        // If this is a numeric index and it's the last part where we have the invalid value
        if (!isNaN(numericPart) && i === pathParts.length - 1) {
          // Format the value for display
          let displayValue: string;
          if (typeof error.invalidValue === 'string') {
            displayValue = `"${error.invalidValue}"`;
          } else if (error.invalidValue === null) {
            displayValue = 'null';
          } else if (error.invalidValue === undefined) {
            displayValue = 'undefined';
          } else {
            displayValue = String(error.invalidValue);
          }
          newPathParts.push(displayValue);
        } else {
          // Keep other parts as-is
          newPathParts.push(part);
        }
      }
      modifiedPath = newPathParts.join('.');
    }
    setWith(tree, modifiedPath, error.message, Object);
  });
  return tree;
}

/**
 * Groups and displays validation errors using treeify with deduplication
 */
export function ValidationErrorsList(t0) {
  const $ = _c(9);
  const {
    errors
  } = t0;
  const [themeName] = useTheme();
  if (errors.length === 0) {
    return null;
  }
  let T0;
  let t1;
  let t2;
  if ($[0] !== errors || $[1] !== themeName) {
    const errorsByFile = errors.reduce(_temp, {});
    const sortedFiles = Object.keys(errorsByFile).sort();
    T0 = Box;
    t1 = "column";
    t2 = sortedFiles.map(file_0 => {
      const fileErrors = errorsByFile[file_0] || [];
      fileErrors.sort(_temp2);
      const errorTree = buildNestedTree(fileErrors);
      const suggestionPairs = new Map();
      fileErrors.forEach(error_0 => {
        if (error_0.suggestion || error_0.docLink) {
          const key = `${error_0.suggestion || ""}|${error_0.docLink || ""}`;
          if (!suggestionPairs.has(key)) {
            suggestionPairs.set(key, {
              suggestion: error_0.suggestion,
              docLink: error_0.docLink
            });
          }
        }
      });
      const treeOutput = treeify(errorTree, {
        showValues: true,
        themeName,
        treeCharColors: {
          treeChar: "inactive",
          key: "text",
          value: "inactive"
        }
      });
      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>;
    });
    $[0] = errors;
    $[1] = themeName;
    $[2] = T0;
    $[3] = t1;
    $[4] = t2;
  } else {
    T0 = $[2];
    t1 = $[3];
    t2 = $[4];
  }
  let t3;
  if ($[5] !== T0 || $[6] !== t1 || $[7] !== t2) {
    t3 = <T0 flexDirection={t1}>{t2}</T0>;
    $[5] = T0;
    $[6] = t1;
    $[7] = t2;
    $[8] = t3;
  } else {
    t3 = $[8];
  }
  return t3;
}
function _temp3(pair, index) {
  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>;
}
function _temp2(a, b) {
  if (!a.path && b.path) {
    return -1;
  }
  if (a.path && !b.path) {
    return 1;
  }
  return (a.path || "").localeCompare(b.path || "");
}
function _temp(acc, error) {
  const file = error.file || "(file not specified)";
  if (!acc[file]) {
    acc[file] = [];
  }
  acc[file].push(error);
  return acc;
}
//# 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":[]}