UNPKG

zod

Version:

TypeScript-first schema declaration and validation library with static type inference

177 lines (165 loc) 6.39 kB
import type { $ZodStringFormats } from "../core/checks.js"; import type * as errors from "../core/errors.js"; import * as util from "../core/util.js"; function getRussianPlural(count: number, one: string, few: string, many: string): string { const absCount = Math.abs(count); const lastDigit = absCount % 10; const lastTwoDigits = absCount % 100; if (lastTwoDigits >= 11 && lastTwoDigits <= 19) { return many; } if (lastDigit === 1) { return one; } if (lastDigit >= 2 && lastDigit <= 4) { return few; } return many; } interface RussianSizable { unit: { one: string; few: string; many: string; }; verb: string; } const error: () => errors.$ZodErrorMap = () => { const Sizable: Record<string, RussianSizable> = { string: { unit: { one: "символ", few: "символа", many: "символов", }, verb: "иметь", }, file: { unit: { one: "байт", few: "байта", many: "байт", }, verb: "иметь", }, array: { unit: { one: "элемент", few: "элемента", many: "элементов", }, verb: "иметь", }, set: { unit: { one: "элемент", few: "элемента", many: "элементов", }, verb: "иметь", }, }; function getSizing(origin: string): RussianSizable | null { return Sizable[origin] ?? null; } const FormatDictionary: { [k in $ZodStringFormats | (string & {})]?: string; } = { regex: "ввод", email: "email адрес", url: "URL", emoji: "эмодзи", uuid: "UUID", uuidv4: "UUIDv4", uuidv6: "UUIDv6", nanoid: "nanoid", guid: "GUID", cuid: "cuid", cuid2: "cuid2", ulid: "ULID", xid: "XID", ksuid: "KSUID", datetime: "ISO дата и время", date: "ISO дата", time: "ISO время", duration: "ISO длительность", ipv4: "IPv4 адрес", ipv6: "IPv6 адрес", cidrv4: "IPv4 диапазон", cidrv6: "IPv6 диапазон", base64: "строка в формате base64", base64url: "строка в формате base64url", json_string: "JSON строка", e164: "номер E.164", jwt: "JWT", template_literal: "ввод", }; const TypeDictionary: { [k in errors.$ZodInvalidTypeExpected | (string & {})]?: string; } = { nan: "NaN", number: "число", array: "массив", }; return (issue) => { switch (issue.code) { case "invalid_type": { const expected = TypeDictionary[issue.expected] ?? issue.expected; const receivedType = util.parsedType(issue.input); const received = TypeDictionary[receivedType] ?? receivedType; if (/^[A-Z]/.test(issue.expected)) { return `Неверный ввод: ожидалось instanceof ${issue.expected}, получено ${received}`; } return `Неверный ввод: ожидалось ${expected}, получено ${received}`; } case "invalid_value": if (issue.values.length === 1) return `Неверный ввод: ожидалось ${util.stringifyPrimitive(issue.values[0])}`; return `Неверный вариант: ожидалось одно из ${util.joinValues(issue.values, "|")}`; case "too_big": { const adj = issue.inclusive ? "<=" : "<"; const sizing = getSizing(issue.origin); if (sizing) { const maxValue = Number(issue.maximum); const unit = getRussianPlural(maxValue, sizing.unit.one, sizing.unit.few, sizing.unit.many); return `Слишком большое значение: ожидалось, что ${issue.origin ?? "значение"} будет иметь ${adj}${issue.maximum.toString()} ${unit}`; } return `Слишком большое значение: ожидалось, что ${issue.origin ?? "значение"} будет ${adj}${issue.maximum.toString()}`; } case "too_small": { const adj = issue.inclusive ? ">=" : ">"; const sizing = getSizing(issue.origin); if (sizing) { const minValue = Number(issue.minimum); const unit = getRussianPlural(minValue, sizing.unit.one, sizing.unit.few, sizing.unit.many); return `Слишком маленькое значение: ожидалось, что ${issue.origin} будет иметь ${adj}${issue.minimum.toString()} ${unit}`; } return `Слишком маленькое значение: ожидалось, что ${issue.origin} будет ${adj}${issue.minimum.toString()}`; } case "invalid_format": { const _issue = issue as errors.$ZodStringFormatIssues; if (_issue.format === "starts_with") return `Неверная строка: должна начинаться с "${_issue.prefix}"`; if (_issue.format === "ends_with") return `Неверная строка: должна заканчиваться на "${_issue.suffix}"`; if (_issue.format === "includes") return `Неверная строка: должна содержать "${_issue.includes}"`; if (_issue.format === "regex") return `Неверная строка: должна соответствовать шаблону ${_issue.pattern}`; return `Неверный ${FormatDictionary[_issue.format] ?? issue.format}`; } case "not_multiple_of": return `Неверное число: должно быть кратным ${issue.divisor}`; case "unrecognized_keys": return `Нераспознанн${issue.keys.length > 1 ? "ые" : "ый"} ключ${issue.keys.length > 1 ? "и" : ""}: ${util.joinValues(issue.keys, ", ")}`; case "invalid_key": return `Неверный ключ в ${issue.origin}`; case "invalid_union": return "Неверные входные данные"; case "invalid_element": return `Неверное значение в ${issue.origin}`; default: return `Неверные входные данные`; } }; }; export default function (): { localeError: errors.$ZodErrorMap } { return { localeError: error(), }; }