๐Ÿ“„ zod/error-customization

File: error-customization.md | Updated: 11/16/2025

Source: https://zod.dev/error-customization?id=error-precedence

๐Ÿ’Ž Zod 4 is now stable! ย Read the announcement.

Customizing errors

Copy markdown

Edit this page

In Zod, validation errors are surfaced as instances of the z.core.$ZodError class.

The ZodError class in the zod package is a subclass that implements some additional convenience methods.

Instances of $ZodError contain an .issues array. Each issue contains a human-readable message and additional structured metadata about the issue.

ZodZod Mini

import * as z from "zod";
 
const result = z.string().safeParse(12); // { success: false, error: ZodError }
result.error.issues;
// [\
//   {\
//     expected: 'string',\
//     code: 'invalid_type',\
//     path: [],\
//     message: 'Invalid input: expected string, received number'\
//   }\
// ]

Every issue contains a message property with a human-readable error message. Error messages can be customized in a number of ways.

The error param


Virtually every Zod API accepts an optional error message.

z.string("Not a string!");

This custom error will show up as the message property of any validation issues that originate from this schema.

z.string("Not a string!").parse(12);
// โŒ throws ZodError {
//   issues: [\
//     {\
//       expected: 'string',\
//       code: 'invalid_type',\
//       path: [],\
//       message: 'Not a string!'   <-- ๐Ÿ‘€ custom error message\
//     }\
//   ]
// }

All z functions and schema methods accept custom errors.

ZodZod Mini

z.string("Bad!");
z.string().min(5, "Too short!");
z.uuid("Bad UUID!");
z.iso.date("Bad date!");
z.array(z.string(), "Not an array!");
z.array(z.string()).min(5, "Too few items!");
z.set(z.string(), "Bad set!");

If you prefer, you can pass a params object with an error parameter instead.

ZodZod Mini

z.string({ error: "Bad!" });
z.string().min(5, { error: "Too short!" });
z.uuid({ error: "Bad UUID!" });
z.iso.date({ error: "Bad date!" });
z.array(z.string(), { error: "Bad array!" });
z.array(z.string()).min(5, { error: "Too few items!" });
z.set(z.string(), { error: "Bad set!" });

The error param optionally accepts a function. An error customization function is known as an error map in Zod terminology. The error map will run at parse time if a validation error occurs.

z.string({ error: ()=>`[${Date.now()}]: Validation failure.` });

Note โ€” In Zod v3, there were separate params for message (a string) and errorMap (a function). These have been unified in Zod 4 as error.

The error map receives a context object you can use to customize the error message based on the validation issue.

z.string({
  error: (iss) => iss.input === undefined ? "Field is required." : "Invalid input."
});

For advanced cases, the iss object provides additional information you can use to customize the error.

z.string({
  error: (iss) => {
    iss.code; // the issue code
    iss.input; // the input data
    iss.inst; // the schema/check that originated this issue
    iss.path; // the path of the error
  },
});

Depending on the API you are using, there may be additional properties available. Use TypeScript's autocomplete to explore the available properties.

z.string().min(5, {
  error: (iss) => {
    // ...the same as above
    iss.minimum; // the minimum value
    iss.inclusive; // whether the minimum is inclusive
    return `Password must have ${iss.minimum} characters or more`;
  },
});

Return undefined to avoid customizing the error message and fall back to the default message. (More specifically, Zod will yield control to the next error map in the precedence chain .) This is useful for selectively customizing certain error messages but not others.

z.int64({
  error: (issue) => {
    // override too_big error message
    if (issue.code === "too_big") {
      return { message: `Value must be <${issue.maximum}` };
    }
 
    //  defer to default
    return undefined;
  },
});

Per-parse error customization


To customize errors on a per-parse basis, pass an error map into the parse method:

const schema = z.string();
 
schema.parse(12, {
  error: iss => "per-parse custom error"
});

This has lower precedence than any schema-level custom messages.

const schema = z.string({ error: "highest priority" });
const result = schema.safeParse(12, {
  error: (iss) => "lower priority",
});
 
result.error.issues;
// [{ message: "highest priority", ... }]

The iss object is a discriminated union of all possible issue types. Use the code property to discriminate between them.

For a breakdown of all Zod issue codes, see the zod/v4/core documentation.

const result = schema.safeParse(12, {
  error: (iss) => {
    if (iss.code === "invalid_type") {
      return `invalid type, expected ${iss.expected}`;
    }
    if (iss.code === "too_small") {
      return `minimum is ${iss.minimum}`;
    }
    // ...
  }
});

Include input in issues

By default, Zod does not include input data in issues. This is to prevent unintentional logging of potentially sensitive input data. To include the input data in each issue, use the reportInput flag:

z.string().parse(12, {
  reportInput: true
})
 
// ZodError: [\
//   {\
//     "expected": "string",\
//     "code": "invalid_type",\
//     "input": 12, // ๐Ÿ‘€\
//     "path": [],\
//     "message": "Invalid input: expected string, received number"\
//   }\
// ]

Global error customization


To specify a global error map, use z.config() to set Zod's customError configuration setting:

z.config({
  customError: (iss) => {
    return "globally modified error";
  },
});

Global error messages have lower precedence than schema-level or per-parse error messages.

The iss object is a discriminated union of all possible issue types. Use the code property to discriminate between them.

For a breakdown of all Zod issue codes, see the zod/v4/core documentation.

const result = schema.safeParse(12, {
  error: (iss) => {
    if (iss.code === "invalid_type") {
      return `invalid type, expected ${iss.expected}`;
    }
    if (iss.code === "too_small") {
      return `minimum is ${iss.minimum}`;
    }
    // ...
  }
})

Internationalization


To support internationalization of error message, Zod provides several built-in locales. These are exported from the zod/v4/core package.

Note โ€” The regular zod library automatically loads the en locale automatically. Zod Mini does not load any locale by default; instead all error messages default to Invalid input.

ZodZod Mini

import * as z from "zod";
import { en } from "zod/locales"
 
z.config(en());

To lazily load a locale, consider dynamic imports:

import * as z from "zod";
 
async function loadLocale(locale: string) {
  const { default: locale } = await import(`zod/v4/locales/${locale}.js`);
  z.config(locale());
};
 
await loadLocale("fr");

For convenience, all locales are exported as z.locales from "zod". In some bundlers, this may not be tree-shakable.

ZodZod Mini

import * as z from "zod";
 
z.config(z.locales.en());

Locales

The following locales are available:

  • ar โ€” Arabic
  • az โ€” Azerbaijani
  • be โ€” Belarusian
  • bg โ€” Bulgarian
  • ca โ€” Catalan
  • cs โ€” Czech
  • da โ€” Danish
  • de โ€” German
  • en โ€” English
  • eo โ€” Esperanto
  • es โ€” Spanish
  • fa โ€” Farsi
  • fi โ€” Finnish
  • fr โ€” French
  • frCA โ€” Canadian French
  • he โ€” Hebrew
  • hu โ€” Hungarian
  • id โ€” Indonesian
  • is โ€” Icelandic
  • it โ€” Italian
  • ja โ€” Japanese
  • ka โ€” Georgian
  • km โ€” Khmer
  • ko โ€” Korean
  • lt โ€” Lithuanian
  • mk โ€” Macedonian
  • ms โ€” Malay
  • nl โ€” Dutch
  • no โ€” Norwegian
  • ota โ€” Tรผrkรฎ
  • ps โ€” Pashto
  • pl โ€” Polish
  • pt โ€” Portuguese
  • ru โ€” Russian
  • sl โ€” Slovenian
  • sv โ€” Swedish
  • ta โ€” Tamil
  • th โ€” Thai
  • tr โ€” Tรผrkรงe
  • uk โ€” Ukrainian
  • ur โ€” Urdu
  • vi โ€” Tiแบฟng Viแป‡t
  • zhCN โ€” Simplified Chinese
  • zhTW โ€” Traditional Chinese
  • yo โ€” Yorรนbรก

Error precedence


Below is a quick reference for determining error precedence: if multiple error customizations have been defined, which one takes priority? From highest to lowest priority:

  1. Schema-level error โ€” Any error message "hard coded" into a schema definition.

    z.string("Not a string!");

  2. Per-parse error โ€” A custom error map passed into the .parse() method.

    z.string().parse(12, { error: (iss) => "My custom error" });

  3. Global error map โ€”ย A custom error map passed into z.config().

    z.config({ customError: (iss) => "My custom error" });

  4. Locale error map โ€”ย A custom error map passed into z.config().

    z.config(z.locales.en());

Defining schemas

Complete API reference for all Zod schema types, methods, and validation features
Formatting errors

Utilities for formatting and displaying Zod errors

On this page

The error param Per-parse error customization Include input in issues Global error customization Internationalization Locales Error precedence