packages/docs/content/api.mdx
import { Tabs, Tab } from 'fumadocs-ui/components/tabs'; import { Callout } from "fumadocs-ui/components/callout" import { Accordion, Accordions } from 'fumadocs-ui/components/accordion';
To validate data, you must first define a schema. Schemas represent types, from simple primitive values to complex nested objects and arrays.
import * as z from "zod";
// primitive types
z.string();
z.number();
z.bigint();
z.boolean();
z.symbol();
z.undefined();
z.null();
To coerce input data to the appropriate type, use z.coerce instead:
z.coerce.string(); // String(input)
z.coerce.number(); // Number(input)
z.coerce.boolean(); // Boolean(input)
z.coerce.bigint(); // BigInt(input)
The coerced variant of these schemas attempts to convert the input value to the appropriate type.
const schema = z.coerce.string();
schema.parse("tuna"); // => "tuna"
schema.parse(42); // => "42"
schema.parse(true); // => "true"
schema.parse(null); // => "null"
The input type of these coerced schemas is unknown by default. To specify a more specific input type, pass a generic parameter:
const A = z.coerce.number();
type AInput = z.input<typeof A>; // => unknown
const B = z.coerce.number<number>();
type BInput = z.input<typeof B>; // => number
Zod coerces all inputs using the built-in constructors.
| Zod API | Coercion |
|---|---|
z.coerce.string() | String(value) |
z.coerce.number() | Number(value) |
z.coerce.boolean() | Boolean(value) |
z.coerce.bigint() | BigInt(value) |
z.coerce.date() | new Date(value) |
Boolean coercion with z.coerce.boolean() may not work how you expect. Any truthy value is coerced to true, and any falsy value is coerced to false.
const schema = z.coerce.boolean(); // Boolean(input)
schema.parse("tuna"); // => true
schema.parse("true"); // => true
schema.parse("false"); // => true
schema.parse(1); // => true
schema.parse([]); // => true
schema.parse(0); // => false
schema.parse(""); // => false
schema.parse(undefined); // => false
schema.parse(null); // => false
For total control over coercion logic, consider using z.transform() or z.pipe().
By default the input type of any z.coerce schema is unknown. In some cases, it may be preferable for the input type to be more specific. You can specify the input type with a generic parameter.
const regularCoerce = z.coerce.string();
type RegularInput = z.input<typeof regularCoerce>; // => unknown
type RegularOutput = z.output<typeof regularCoerce>; // => string
const customInput = z.coerce.string<string>();
type CustomInput = z.input<typeof customInput>; // => string
type CustomOutput = z.output<typeof customInput>; // => string
Literal schemas represent a literal type, like "hello world" or 5.
const tuna = z.literal("tuna");
const twelve = z.literal(12);
const twobig = z.literal(2n);
const tru = z.literal(true);
To represent the JavaScript literals null and undefined:
z.null();
z.undefined();
z.void(); // equivalent to z.undefined()
To allow multiple literal values:
const colors = z.literal(["red", "green", "blue"]);
colors.parse("green"); // ✅
colors.parse("yellow"); // ❌
To extract the set of allowed values from a literal schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
colors.values; // => Set<"red" | "green" | "blue">
{/* Zod provides a handful of built-in string validation and transform APIs.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.string().startsWith("fourscore")
All of the APIs documented below support the error parameter for customizing the error message.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.string().startsWith("fourscore", {error: "Nice try, buddy"})
Zod provides a handful of built-in string validation and transform APIs. To perform some common string validations:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.string().max(5);
z.string().min(5);
z.string().length(5);
z.string().regex(/^[a-z]+$/);
z.string().startsWith("aaa");
z.string().endsWith("zzz");
z.string().includes("---");
z.string().uppercase();
z.string().lowercase();
To perform some simple string transforms:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.string().trim(); // trim whitespace
z.string().toLowerCase(); // toLowerCase
z.string().toUpperCase(); // toUpperCase
z.string().normalize(); // normalize unicode characters
To validate against some common string formats:
z.email();
z.uuid();
z.url();
z.httpUrl(); // http or https URLs only
z.hostname();
z.e164(); // E.164 phone numbers
z.emoji(); // validates a single emoji character
z.base64();
z.base64url();
z.hex();
z.jwt();
z.nanoid();
z.cuid();
z.cuid2();
z.ulid();
z.ipv4();
z.ipv6();
z.mac();
z.cidrv4(); // ipv4 CIDR block
z.cidrv6(); // ipv6 CIDR block
z.hash("sha256"); // or "sha1", "sha384", "sha512", "md5"
z.iso.date();
z.iso.time();
z.iso.datetime();
z.iso.duration();
To validate email addresses:
z.email();
By default, Zod uses a comparatively strict email regex designed to validate normal email addresses containing common characters. It's roughly equivalent to the rules enforced by Gmail. To learn more about this regex, refer to this post.
/^(?!\.)(?!.*\.\.)([a-z0-9_'+\-\.]*)[a-z0-9_+-]@([a-z0-9][a-z0-9\-]*\.)+[a-z]{2,}$/i
To customize the email validation behavior, you can pass a custom regular expression to the pattern param.
z.email({ pattern: /your regex here/ });
Zod exports several useful regexes you could use.
// Zod's default email regex
z.email();
z.email({ pattern: z.regexes.email }); // equivalent
// the regex used by browsers to validate input[type=email] fields
// https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/email
z.email({ pattern: z.regexes.html5Email });
// the classic emailregex.com regex (RFC 5322)
z.email({ pattern: z.regexes.rfc5322Email });
// a loose regex that allows Unicode (good for intl emails)
z.email({ pattern: z.regexes.unicodeEmail });
To validate UUIDs:
z.uuid();
To specify a particular UUID version:
// supports "v1", "v2", "v3", "v4", "v5", "v6", "v7", "v8"
z.uuid({ version: "v4" });
// for convenience
z.uuidv4();
z.uuidv6();
z.uuidv7();
The RFC 9562/4122 UUID spec requires the first two bits of byte 8 to be 10. Other UUID-like identifiers do not enforce this constraint. To validate any UUID-like identifier:
z.guid();
To validate any WHATWG-compatible URL:
const schema = z.url();
schema.parse("https://example.com"); // ✅
schema.parse("http://localhost"); // ✅
schema.parse("mailto:[email protected]"); // ✅
As you can see this is quite permissive. Internally this uses the new URL() constructor to validate inputs; this behavior may differ across platforms and runtimes but it's the mostly rigorous way to validate URIs/URLs on any given JS runtime/engine.
To validate the hostname against a specific regex:
const schema = z.url({ hostname: /^example\.com$/ });
schema.parse("https://example.com"); // ✅
schema.parse("https://zombo.com"); // ❌
To validate the protocol against a specific regex, use the protocol param.
const schema = z.url({ protocol: /^https$/ });
schema.parse("https://example.com"); // ✅
schema.parse("http://example.com"); // ❌
const httpUrl = z.url({
protocol: /^https?$/,
hostname: z.regexes.domain
});
This restricts the protocol to http/https and ensures the hostname is a valid domain name with the z.regexes.domain regular expression:
/^([a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}$/
To normalize URLs, use the normalize flag. This will overwrite the input value with the normalized URL returned by new URL().
new URL("HTTP://ExAmPle.com:80/./a/../b?X=1#f oo").href
// => "http://example.com/b?X=1#f%20oo"
To validate phone numbers in E.164 format:
const phone = z.e164();
phone.parse("+15555555555"); // ✅
phone.parse("555-555-5555"); // ❌
This schema validates strings with a leading +, a non-zero country code, and 7 to 15 digits total.
As you may have noticed, Zod string includes a few date/time related validations. These validations are regular expression based, so they are not as strict as a full date/time library. However, they are very convenient for validating user input.
The z.iso.datetime() method enforces ISO 8601; by default, no timezone offsets are allowed:
const datetime = z.iso.datetime();
datetime.parse("2020-01-01T06:15:00Z"); // ✅
datetime.parse("2020-01-01T06:15:00.123Z"); // ✅
datetime.parse("2020-01-01T06:15:00.123456Z"); // ✅ (arbitrary precision)
datetime.parse("2020-01-01T06:15:00+02:00"); // ❌ (offsets not allowed)
datetime.parse("2020-01-01T06:15:00"); // ❌ (local not allowed)
To allow timezone offsets:
const datetime = z.iso.datetime({ offset: true });
// allows timezone offsets
datetime.parse("2020-01-01T06:15:00+02:00"); // ✅
// basic offsets not allowed
datetime.parse("2020-01-01T06:15:00+02"); // ❌
datetime.parse("2020-01-01T06:15:00+0200"); // ❌
// Z is still supported
datetime.parse("2020-01-01T06:15:00Z"); // ✅
To allow unqualified (timezone-less) datetimes:
const schema = z.iso.datetime({ local: true });
schema.parse("2020-01-01T06:15:01"); // ✅
schema.parse("2020-01-01T06:15"); // ✅ seconds optional
To constrain the allowable time precision. By default, seconds are optional and arbitrary sub-second precision is allowed.
const a = z.iso.datetime();
a.parse("2020-01-01T06:15Z"); // ✅
a.parse("2020-01-01T06:15:00Z"); // ✅
a.parse("2020-01-01T06:15:00.123Z"); // ✅
const b = z.iso.datetime({ precision: -1 }); // minute precision (no seconds)
b.parse("2020-01-01T06:15Z"); // ✅
b.parse("2020-01-01T06:15:00Z"); // ❌
b.parse("2020-01-01T06:15:00.123Z"); // ❌
const c = z.iso.datetime({ precision: 0 }); // second precision only
c.parse("2020-01-01T06:15Z"); // ❌
c.parse("2020-01-01T06:15:00Z"); // ✅
c.parse("2020-01-01T06:15:00.123Z"); // ❌
const d = z.iso.datetime({ precision: 3 }); // millisecond precision only
d.parse("2020-01-01T06:15Z"); // ❌
d.parse("2020-01-01T06:15:00Z"); // ❌
d.parse("2020-01-01T06:15:00.123Z"); // ✅
The z.iso.date() method validates strings in the format YYYY-MM-DD.
const date = z.iso.date();
date.parse("2020-01-01"); // ✅
date.parse("2020-1-1"); // ❌
date.parse("2020-01-32"); // ❌
The z.iso.time() method validates strings in the format HH:MM[:SS[.s+]]. By default seconds are optional, as are sub-second decimals.
const time = z.iso.time();
time.parse("03:15"); // ✅
time.parse("03:15:00"); // ✅
time.parse("03:15:00.9999999"); // ✅ (arbitrary precision)
No offsets of any kind are allowed.
time.parse("03:15:00Z"); // ❌ (no `Z` allowed)
time.parse("03:15:00+02:00"); // ❌ (no offsets allowed)
Use the precision parameter to constrain the allowable decimal precision.
z.iso.time({ precision: -1 }); // HH:MM (minute precision)
z.iso.time({ precision: 0 }); // HH:MM:SS (second precision)
z.iso.time({ precision: 1 }); // HH:MM:SS.s (decisecond precision)
z.iso.time({ precision: 2 }); // HH:MM:SS.ss (centisecond precision)
z.iso.time({ precision: 3 }); // HH:MM:SS.sss (millisecond precision)
const ipv4 = z.ipv4();
ipv4.parse("192.168.0.0"); // ✅
const ipv6 = z.ipv6();
ipv6.parse("2001:db8:85a3::8a2e:370:7334"); // ✅
Validate IP address ranges specified with CIDR notation.
const cidrv4 = z.cidrv4();
cidrv4.parse("192.168.0.0/24"); // ✅
const cidrv6 = z.cidrv6();
cidrv6.parse("2001:db8::/32"); // ✅
Validate standard 48-bit MAC address IEEE 802.
const mac = z.mac();
mac.parse("00:1A:2B:3C:4D:5E"); // ✅
mac.parse("00-1a-2b-3c-4d-5e"); // ❌ colon-delimited by default
mac.parse("001A:2B3C:4D5E"); // ❌ standard formats only
mac.parse("00:1A:2b:3C:4d:5E"); // ❌ no mixed case
// custom delimiter
const dashMac = z.mac({ delimiter: "-" });
dashMac.parse("00-1A-2B-3C-4D-5E"); // ✅
Validate JSON Web Tokens.
z.jwt();
z.jwt({ alg: "HS256" });
To validate cryptographic hash values:
z.hash("md5");
z.hash("sha1");
z.hash("sha256");
z.hash("sha384");
z.hash("sha512");
By default, z.hash() expects hexadecimal encoding, as is conventional. You can specify a different encoding with the enc parameter:
z.hash("sha256", { enc: "hex" }); // default
z.hash("sha256", { enc: "base64" }); // base64 encoding
z.hash("sha256", { enc: "base64url" }); // base64url encoding (no padding)
| Algorithm / Encoding | "hex" | "base64" | "base64url" |
|---|---|---|---|
"md5" | 32 | 24 (22 + "==") | 22 |
"sha1" | 40 | 28 (27 + "=") | 27 |
"sha256" | 64 | 44 (43 + "=") | 43 |
"sha384" | 96 | 64 (no padding) | 64 |
"sha512" | 128 | 88 (86 + "==") | 86 |
To define your own string formats:
const coolId = z.stringFormat("cool-id", (val)=>{
// arbitrary validation here
return val.length === 100 && val.startsWith("cool-");
});
// a regex is also accepted
z.stringFormat("cool-id", /^cool-[a-z0-9]{95}$/);
This schema will produce "invalid_format" issues, which are more descriptive than the "custom" errors produced by refinements or z.custom().
myFormat.parse("invalid input!");
// ZodError: [
// {
// "code": "invalid_format",
// "format": "cool-id",
// "path": [],
// "message": "Invalid cool-id"
// }
// ]
Introduced in
[email protected].
To define a template literal schema:
const schema = z.templateLiteral([ "hello, ", z.string(), "!" ]);
// `hello, ${string}!`
The z.templateLiteral API can handle any number of string literals (e.g. "hello") and schemas. Any schema with an inferred type that's assignable to string | number | bigint | boolean | null | undefined can be passed.
z.templateLiteral([ "hi there" ]);
// `hi there`
z.templateLiteral([ "email: ", z.string() ]);
// `email: ${string}`
z.templateLiteral([ "high", z.literal(5) ]);
// `high5`
z.templateLiteral([ z.nullable(z.literal("grassy")) ]);
// `grassy` | `null`
z.templateLiteral([ z.number(), z.enum(["px", "em", "rem"]) ]);
// `${number}px` | `${number}em` | `${number}rem`
Use z.number() to validate numbers. It allows any finite number.
const schema = z.number();
schema.parse(3.14); // ✅
schema.parse(NaN); // ❌
schema.parse(Infinity); // ❌
Zod implements a handful of number-specific validations:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.number().gt(5);
z.number().gte(5); // alias .min(5)
z.number().lt(5);
z.number().lte(5); // alias .max(5)
z.number().positive(); // alias .gt(0)
z.number().nonnegative();
z.number().negative();
z.number().nonpositive();
z.number().multipleOf(5); // alias .step(5)
If (for some reason) you want to validate NaN, use z.nan().
z.nan().parse(NaN); // ✅
z.nan().parse("anything else"); // ❌
To validate integers:
z.int(); // restricts to safe integer range
z.int32(); // restrict to int32 range
To validate BigInts:
z.bigint();
Zod includes a handful of bigint-specific validations.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.bigint().gt(5n);
z.bigint().gte(5n); // alias `.min(5n)`
z.bigint().lt(5n);
z.bigint().lte(5n); // alias `.max(5n)`
z.bigint().positive(); // alias `.gt(0n)`
z.bigint().nonnegative();
z.bigint().negative();
z.bigint().nonpositive();
z.bigint().multipleOf(5n); // alias `.step(5n)`
To validate boolean values:
z.boolean().parse(true); // => true
z.boolean().parse(false); // => false
Use z.date() to validate Date instances.
z.date().safeParse(new Date()); // success: true
z.date().safeParse("2022-01-12T06:15:00.000Z"); // success: false
To customize the error message:
z.date({
error: issue => issue.input === undefined ? "Required" : "Invalid date"
});
Zod provides a handful of date-specific validations.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.date().min(new Date("1900-01-01"), { error: "Too old!" });
z.date().max(new Date(), { error: "Too young!" });
Use z.enum to validate inputs against a fixed set of allowable string values.
const FishEnum = z.enum(["Salmon", "Tuna", "Trout"]);
FishEnum.parse("Salmon"); // => "Salmon"
FishEnum.parse("Swordfish"); // => ❌
const fish = ["Salmon", "Tuna", "Trout"];
const FishEnum = z.enum(fish);
type FishEnum = z.infer<typeof FishEnum>; // string
To fix this, always pass the array directly into the z.enum() function, or use as const.
const fish = ["Salmon", "Tuna", "Trout"] as const;
const FishEnum = z.enum(fish);
type FishEnum = z.infer<typeof FishEnum>; // "Salmon" | "Tuna" | "Trout"
Enum-like object literals ({ [key: string]: string | number }) are supported.
const Fish = {
Salmon: 0,
Tuna: 1
} as const
const FishEnum = z.enum(Fish)
FishEnum.parse(Fish.Salmon); // => ✅
FishEnum.parse(0); // => ✅
FishEnum.parse(2); // => ❌
You can also pass in an externally-declared TypeScript enum.
enum Fish {
Salmon = 0,
Tuna = 1
}
const FishEnum = z.enum(Fish);
FishEnum.parse(Fish.Salmon); // => ✅
FishEnum.parse(0); // => ✅
FishEnum.parse(2); // => ❌
Note that using TypeScript's enum keyword is not recommended.
</Callout>
enum Fish {
Salmon = "Salmon",
Tuna = "Tuna",
Trout = "Trout",
}
const FishEnum = z.enum(Fish);
.enumTo extract the schema's values as an enum-like object:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const FishEnum = z.enum(["Salmon", "Tuna", "Trout"]);
FishEnum.enum;
// => { Salmon: "Salmon", Tuna: "Tuna", Trout: "Trout" }
FishEnum.def.entries; // => { Salmon: "Salmon", Tuna: "Tuna", Trout: "Trout" }
</Tab>
</Tabs>
### `.exclude()`
To create a new enum schema, excluding certain values:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const FishEnum = z.enum(["Salmon", "Tuna", "Trout"]);
const TunaOnly = FishEnum.exclude(["Salmon", "Trout"]);
</Tab>
</Tabs>
### `.extract()`
To create a new enum schema, extracting certain values:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const FishEnum = z.enum(["Salmon", "Tuna", "Trout"]);
const SalmonAndTroutOnly = FishEnum.extract(["Salmon", "Trout"]);
</Tab>
</Tabs>
## Stringbools [#stringbool]
> Introduced in `[email protected]`.
In some cases (e.g. parsing environment variables) it's valuable to parse certain string "boolish" values to a plain `boolean` value. Use `z.stringbool()` for this:
```ts
const strbool = z.stringbool();
strbool.parse("true") // => true
strbool.parse("1") // => true
strbool.parse("yes") // => true
strbool.parse("on") // => true
strbool.parse("y") // => true
strbool.parse("enabled") // => true
strbool.parse("false"); // => false
strbool.parse("0"); // => false
strbool.parse("no"); // => false
strbool.parse("off"); // => false
strbool.parse("n"); // => false
strbool.parse("disabled"); // => false
strbool.parse(/* anything else */); // ZodError<[{ code: "invalid_value" }]>
To customize the truthy and falsy values:
// these are the defaults
z.stringbool({
truthy: ["true", "1", "yes", "on", "y", "enabled"],
falsy: ["false", "0", "no", "off", "n", "disabled"],
});
By default the schema is case-insensitive; all inputs are converted to lowercase before comparison to the truthy/falsy values. To make it case-sensitive:
z.stringbool({
case: "sensitive"
});
To make a schema optional (that is, to allow undefined inputs).
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.optional(z.literal("yoda")); // or z.literal("yoda").optional()
This returns a ZodOptional instance that wraps the original schema. To extract the inner schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
optionalYoda.unwrap(); // ZodLiteral<"yoda">
To make a schema nullable (that is, to allow null inputs).
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.nullable(z.literal("yoda")); // or z.literal("yoda").nullable()
This returns a ZodNullable instance that wraps the original schema. To extract the inner schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
nullableYoda.unwrap(); // ZodLiteral<"yoda">
To make a schema nullish (both optional and nullable):
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const nullishYoda = z.nullish(z.literal("yoda"));
Refer to the TypeScript manual for more about the concept of nullish.
Zod aims to mirror TypeScript's type system one-to-one. As such, Zod provides APIs to represent the following special types:
// allows any values
z.any(); // inferred type: `any`
z.unknown(); // inferred type: `unknown`
No value will pass validation.
z.never(); // inferred type: `never`
To define an object type:
// all properties are required by default
const Person = z.object({
name: z.string(),
age: z.number(),
});
type Person = z.infer<typeof Person>;
// => { name: string; age: number; }
By default, all properties are required. To make certain properties optional:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const Dog = z.object({
name: z.string(),
age: z.number().optional(),
});
Dog.parse({ name: "Yeller" }); // ✅
Dog.parse({ name: "Yeller" }); // ✅
</Tab>
</Tabs>
By default, unrecognized keys are *stripped* from the parsed result:
```ts z.object
Dog.parse({ name: "Yeller", extraKey: true });
// => { name: "Yeller" }
z.strictObjectTo define a strict schema that throws an error when unknown keys are found:
const StrictDog = z.strictObject({
name: z.string(),
});
StrictDog.parse({ name: "Yeller", extraKey: true });
// ❌ throws
z.looseObjectTo define a loose schema that allows unknown keys to pass through:
const LooseDog = z.looseObject({
name: z.string(),
});
LooseDog.parse({ name: "Yeller", extraKey: true });
// => { name: "Yeller", extraKey: true }
.catchall()To define a catchall schema that will be used to validate any unrecognized keys:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const DogWithStrings = z
.object({
name: z.string(),
age: z.number().optional(),
})
.catchall(z.string());
DogWithStrings.parse({ name: "Yeller", extraKey: "extraValue" }); // ✅
DogWithStrings.parse({ name: "Yeller", extraKey: 42 }); // ❌
DogWithStrings.parse({ name: "Yeller", extraKey: "extraValue" }); // ✅ DogWithStrings.parse({ name: "Yeller", extraKey: 42 }); // ❌
</Tab>
</Tabs>
### `.shape`
To access the internal schemas:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
Dog.shape.name; // => string schema
Dog.shape.age; // => number schema
.keyof()To create a ZodEnum schema from the keys of an object schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const keySchema = Dog.keyof();
// => ZodEnum<["name", "age"]>
.extend()To add additional fields to an object schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const DogWithBreed = Dog.extend({
breed: z.string(),
});
This API can be used to overwrite existing fields! Be careful with this power! If the two schemas share keys, B will override A.
<Callout> **Alternative: spread syntax** — You can alternatively avoid `.extend()` altogether by creating a new object schema entirely. This makes the strictness level of the resulting schema visually obvious.const DogWithBreed = z.object({ // or z.strictObject() or z.looseObject()...
...Dog.shape,
breed: z.string(),
});
You can also use this to merge multiple objects in one go.
const DogWithBreed = z.object({
...Animal.shape,
...Pet.shape,
breed: z.string(),
});
This approach has a few advantages:
tsc-efficient — the .extend() method can be expensive on large schemas, and due to a TypeScript limitation it gets quadratically more expensive when calls are chainedz.strictObject() or z.looseObject()
</Callout>
.safeExtend()The .safeExtend() method works similarly to .extend(), but it won't let you overwrite an existing property with a non-assignable schema. In other words, the result of .safeExtend() will have an inferred type that extends the original (in the TypeScript sense).
z.object({ a: z.string() }).safeExtend({ a: z.string().min(5) }); // ✅
z.object({ a: z.string() }).safeExtend({ a: z.any() }); // ✅
z.object({ a: z.string() }).safeExtend({ a: z.number() });
// ^ ❌ ZodNumber is not assignable
Use .safeExtend() to extend schemas that contain refinements. (Regular .extend() will throw an error when used on schemas with refinements.)
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const Base = z.object({
a: z.string(),
b: z.string()
}).refine(user => user.a === user.b);
// Extended inherits the refinements of Base
const Extended = Base.safeExtend({
a: z.string().min(10)
});
// Extended inherits the refinements of Base const Extended = z.safeExtend(Base, { a: z.string().min(10) });
</Tab>
</Tabs>
### `.pick()`
Inspired by TypeScript's built-in `Pick` and `Omit` utility types, Zod provides dedicated APIs for picking and omitting certain keys from an object schema.
Starting from this initial schema:
```ts z.object
const Recipe = z.object({
title: z.string(),
description: z.string().optional(),
ingredients: z.array(z.string()),
});
// { title: string; description?: string | undefined; ingredients: string[] }
To pick certain keys:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const JustTheTitle = Recipe.pick({ title: true });
.omit()To omit certain keys:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const RecipeNoId = Recipe.omit({ id: true });
.partial()For convenience, Zod provides a dedicated API for making some or all properties optional, inspired by the built-in TypeScript utility type Partial.
To make all fields optional:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const PartialRecipe = Recipe.partial();
// { title?: string | undefined; description?: string | undefined; ingredients?: string[] | undefined }
To make certain properties optional:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const RecipeOptionalIngredients = Recipe.partial({
ingredients: true,
});
// { title: string; description?: string | undefined; ingredients?: string[] | undefined }
.required()Zod provides an API for making some or all properties required, inspired by TypeScript's Required utility type.
To make all properties required:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const RequiredRecipe = Recipe.required();
// { title: string; description: string; ingredients: string[] }
To make certain properties required:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const RecipeRequiredDescription = Recipe.required({description: true});
// { title: string; description: string; ingredients: string[] }
To define a self-referential type, use a getter on the key. This lets JavaScript resolve the cyclical schema at runtime.
const Category = z.object({
name: z.string(),
get subcategories(){
return z.array(Category)
}
});
type Category = z.infer<typeof Category>;
// { name: string; subcategories: Category[] }
You can also represent mutually recursive types:
const User = z.object({
email: z.email(),
get posts(){
return z.array(Post)
}
});
const Post = z.object({
title: z.string(),
get author(){
return User
}
});
All object APIs (.pick(), .omit(), .required(), .partial(), etc.) work as you'd expect.
Due to TypeScript limitations, recursive type inference can be finicky, and it only works in certain scenarios. Some more complicated types may trigger recursive type errors like this:
const Activity = z.object({
name: z.string(),
get subactivities() {
// ^ ❌ 'subactivities' implicitly has return type 'any' because it does not
// have a return type annotation and is referenced directly or indirectly
// in one of its return expressions.ts(7023)
return z.nullable(z.array(Activity));
},
});
In these cases, you can resolve the error with a type annotation on the offending getter:
const Activity = z.object({
name: z.string(),
get subactivities(): z.ZodNullable<z.ZodArray<typeof Activity>> {
return z.nullable(z.array(Activity));
},
});
{/* <Accordions> <Accordion title="Resolving type errors in recursive schemas"> Recursive type inference can by mysterious. TypeScript is capable of it in certain limited scenarios. Depending on what you're trying to do, you may encounter errors like this:
```ts
export const Activity = z.object({
name: z.string(),
get children() {
// ^ ❌ 'children' implicitly has return type 'any' because it does not
// have a return type annotation and is referenced directly or indirectly
// in one of its return expressions.ts(7023)
return z.optional(z.array(Activity)); //.optional();
},
});
```
Here are a couple rules of thumb:
### Object types only
Generally speaking, recursive inference only works with object types that are referencing each other. TypeScript has special handling for resolving getter-based recursive objects, which is what Zod relies on. If you try to add non-object types into the mix, you'll likely encounter errors.
```ts
const Activity = z.object({
name: z.string(),
get children() { // ❌ type error
return z.optional(ActivityArray);
},
});
const ActivityArray = z.array(Activity);
```
Sometimes you can get around this limitation by defining
### Avoid nesting function calls
Functions like `z.array()` and `z.optional()` accept Zod schemas, so when you use them TypeScript will do some type-checking on their inputs to make sure they are valid. But type checking is the enemy of recursive type inference—it's hard for TypeScript to *check* and *infer* types at the same time. Methods do not have this problem, so prefer methods over functions when possible (sorry Zod Mini users).
```ts
const Activity = z.object({
name: z.string(),
get subactivities() {
// ^ ❌ 'subactivities' implicitly has return type 'any' because it does not
// have a return type annotation and is referenced directly or indirectly
// in one of its return expressions.ts(7023)
return z.union([z.null(), Activity]);
},
});
```
### Fall back to type annotations on your getters
When in doubt, you can generally sidestep these issues with some carefully deployed type annotations on your getters. Due to the limitations described above, this is particularly necessary when using Zod Mini.
```ts
import * as z from "zod";
const Activity = z.object({
name: z.string(),
get subactivities(): z.ZodMiniDefault<z.ZodMiniArray<typeof Activity>> {
return z._default(z.array(Activity), []);
},
});
```
To define an array schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const stringArray = z.array(z.string()); // or z.string().array()
To access the inner schema for an element of the array.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
stringArray.unwrap(); // => string schema
Zod implements a number of array-specific validations:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.array(z.string()).min(5); // must contain 5 or more items
z.array(z.string()).max(5); // must contain 5 or fewer items
z.array(z.string()).length(5); // must contain 5 items exactly
Unlike arrays, tuples are typically fixed-length arrays that specify different schemas for each index.
const MyTuple = z.tuple([
z.string(),
z.number(),
z.boolean()
]);
type MyTuple = z.infer<typeof MyTuple>;
// [string, number, boolean]
To add a variadic ("rest") argument:
const variadicTuple = z.tuple([z.string()], z.number());
// => [string, ...number[]];
Union types (A | B) represent a logical "OR". Zod union schemas will check the input against each option in order. The first value that validates successfully is returned.
const stringOrNumber = z.union([z.string(), z.number()]);
// string | number
stringOrNumber.parse("foo"); // passes
stringOrNumber.parse(14); // passes
To extract the internal option schemas:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
stringOrNumber.options; // [ZodString, ZodNumber]
{/* For convenience, you can also use the .or method:
const stringOrNumber = z.string().or(z.number());
``` */}
{/* **Optional string validation:**
To validate an optional form input, you can union the desired string validation with an empty string [literal](#literals).
This example validates an input that is optional but needs to contain a [valid URL](#strings):
```ts
const optionalUrl = z.union([z.string().url().nullish(), z.literal("")]);
console.log(optionalUrl.safeParse(undefined).success); // true
console.log(optionalUrl.safeParse(null).success); // true
console.log(optionalUrl.safeParse("").success); // true
console.log(optionalUrl.safeParse("https://zod.dev").success); // true
console.log(optionalUrl.safeParse("not a valid url").success); // false
*/}
An exclusive union (XOR) is a union where exactly one option must match. Unlike regular unions that succeed when any option matches, z.xor() fails if zero options match OR if multiple options match.
const schema = z.xor([z.string(), z.number()]);
schema.parse("hello"); // ✅ passes
schema.parse(42); // ✅ passes
schema.parse(true); // ❌ fails (zero matches)
This is useful when you want to ensure mutual exclusivity between options:
// Validate that exactly ONE of these matches
const payment = z.xor([
z.object({ type: z.literal("card"), cardNumber: z.string() }),
z.object({ type: z.literal("bank"), accountNumber: z.string() }),
]);
payment.parse({ type: "card", cardNumber: "1234" }); // ✅ passes
If the input could match multiple options, z.xor() will fail:
const overlapping = z.xor([z.string(), z.any()]);
overlapping.parse("hello"); // ❌ fails (matches both string and any)
A discriminated union is a special kind of union in which a) all the options are object schemas that b) share a particular key (the "discriminator"). Based on the value of the discriminator key, TypeScript is able to "narrow" the type signature as you'd expect.
type MyResult =
| { status: "success"; data: string }
| { status: "failed"; error: string };
function handleResult(result: MyResult){
if(result.status === "success"){
result.data; // string
} else {
result.error; // string
}
}
You could represent it with a regular z.union(). But regular unions are naive—they check the input against each option in order and return the first one that passes. This can be slow for large unions.
So Zod provides a z.discriminatedUnion() API that uses a discriminator key to make parsing more efficient.
const MyResult = z.discriminatedUnion("status", [
z.object({ status: z.literal("success"), data: z.string() }),
z.object({ status: z.literal("failed"), error: z.string() }),
]);
Each option should be an object schema whose discriminator prop (status in the example above) corresponds to some literal value or set of values, usually z.enum(), z.literal(), z.null(), or z.undefined().
For advanced use cases, discriminated unions can be nested. Zod will figure out the optimal parsing strategy to leverage the discriminators at each level.
const BaseError = { status: z.literal("failed"), message: z.string() };
const MyErrors = z.discriminatedUnion("code", [
z.object({ ...BaseError, code: z.literal(400) }),
z.object({ ...BaseError, code: z.literal(401) }),
z.object({ ...BaseError, code: z.literal(500) }),
]);
const MyResult = z.discriminatedUnion("status", [
z.object({ status: z.literal("success"), data: z.string() }),
MyErrors
]);
Intersection types (A & B) represent a logical "AND".
const a = z.union([z.number(), z.string()]);
const b = z.union([z.number(), z.boolean()]);
const c = z.intersection(a, b);
type c = z.infer<typeof c>; // => number
This can be useful for intersecting two object types.
const Person = z.object({ name: z.string() });
type Person = z.infer<typeof Person>;
const Employee = z.object({ role: z.string() });
type Employee = z.infer<typeof Employee>;
const EmployedPerson = z.intersection(Person, Employee);
type EmployedPerson = z.infer<typeof EmployedPerson>;
// Person & Employee
Record schemas are used to validate types such as Record<string, string>.
z.recordconst IdCache = z.record(z.string(), z.string());
type IdCache = z.infer<typeof IdCache>; // Record<string, string>
IdCache.parse({
carlotta: "77d2586b-9e8e-4ecf-8b21-ea7e0530eadd",
jimmie: "77d2586b-9e8e-4ecf-8b21-ea7e0530eadd",
});
The key schema can be any Zod schema that is assignable to string | number | symbol.
const Keys = z.union([z.string(), z.number(), z.symbol()]);
const AnyObject = z.record(Keys, z.unknown());
// Record<string | number | symbol, unknown>
To create an object schemas containing keys defined by an enum:
const Keys = z.enum(["id", "name", "email"]);
const Person = z.record(Keys, z.string());
// { id: string; name: string; email: string }
Zod supports numeric keys inside records in a way that closely mirrors TypeScript itself. A number schema, when used as a record key, will validate that the key is a valid "numeric string". Additional numerical constraints (min, max, step, etc.) will also be validated.
const numberKeys = z.record(z.number(), z.string());
numberKeys.parse({
1: "one", // ✅
2: "two", // ✅
"1.5": "one", // ✅
"-3": "two", // ✅
abc: "one" // ❌
});
// further validation is also supported
const intKeys = z.record(z.int().step(1).min(0).max(10), z.string());
intKeys.parse({
0: "zero", // ✅
1: "one", // ✅
2: "two", // ✅
12: "twelve", // ❌
abc: "one" // ❌
});
z.partialRecordtype MyRecord = Record<"a" | "b", string>;
const myRecord: MyRecord = { a: "foo", b: "bar" }; // ✅
const myRecord: MyRecord = { a: "foo" }; // ❌ missing required key `b`
For partial key sets, use z.partialRecord().
If you want a partial record type, use z.partialRecord(). This skips the special exhaustiveness checks Zod normally runs with z.enum() and z.literal() key schemas.
const Keys = z.enum(["id", "name", "email"]).or(z.never());
const Person = z.partialRecord(Keys, z.string());
// { id?: string; name?: string; email?: string }
z.looseRecordBy default, z.record() errors on keys that don't match the key schema. Use z.looseRecord() to pass through non-matching keys unchanged. This is particularly useful when combined with intersections to model multiple pattern properties:
const schema = z
.object({ name: z.string() })
.and(z.looseRecord(z.string().regex(/_phone$/), z.e164()));
type schema = z.infer<typeof schema>;
// => { name: string } & Record<string, string>
schema.parse({
name: "John",
home_phone: "+12345678900", // validated as phone number
work_phone: "+12345678900", // validated as phone number
});
const StringNumberMap = z.map(z.string(), z.number());
type StringNumberMap = z.infer<typeof StringNumberMap>; // Map<string, number>
const myMap: StringNumberMap = new Map();
myMap.set("one", 1);
myMap.set("two", 2);
StringNumberMap.parse(myMap);
const NumberSet = z.set(z.number());
type NumberSet = z.infer<typeof NumberSet>; // Set<number>
const mySet: NumberSet = new Set();
mySet.add(1);
mySet.add(2);
NumberSet.parse(mySet);
Set schemas can be further constrained with the following utility methods.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.set(z.string()).min(5); // must contain 5 or more items
z.set(z.string()).max(5); // must contain 5 or fewer items
z.set(z.string()).size(5); // must contain 5 items exactly
To validate File instances:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const fileSchema = z.file();
fileSchema.min(10_000); // minimum .size (bytes)
fileSchema.max(1_000_000); // maximum .size (bytes)
fileSchema.mime("image/png"); // MIME type
fileSchema.mime(["image/png", "image/jpeg"]); // multiple MIME types
fileSchema.check(z.minSize(10_000)); // minimum .size (bytes) fileSchema.check(z.maxSize(1_000_000)); // maximum .size (bytes) fileSchema.check(z.mime("image/png")); // MIME type fileSchema.check(z.mime(["image/png", "image/jpeg"])); // multiple MIME types
</Tab>
</Tabs>
## Promises
<Callout type="warn">
**Deprecated** — `z.promise()` is deprecated. There are vanishingly few valid uses cases for a `Promise` schema. If you suspect a value might be a `Promise`, simply `await` it before parsing it with Zod.
</Callout>
<Accordions type="single"><Accordion title="See z.promise() documentation">
```ts
const numberPromise = z.promise(z.number());
"Parsing" works a little differently with promise schemas. Validation happens in two parts:
.then and .catch methods.)..then to attach an additional validation step onto the existing Promise. You'll have to use .catch on the returned Promise to handle validation failures.numberPromise.parse("tuna");
// ZodError: Non-Promise type: string
numberPromise.parse(Promise.resolve("tuna"));
// => Promise<number>
const test = async () => {
await numberPromise.parse(Promise.resolve("tuna"));
// ZodError: Non-number type: string
await numberPromise.parse(Promise.resolve(3.14));
// => 3.14
};
</Accordion></Accordions>
You can use z.instanceof to check that the input is an instance of a class. This is useful to validate inputs against classes that are exported from third-party libraries.
class Test {
name: string;
}
const TestSchema = z.instanceof(Test);
TestSchema.parse(new Test()); // ✅
TestSchema.parse("whatever"); // ❌
To validate a particular property of a class instance against a Zod schema:
const blobSchema = z.instanceof(URL).check(
z.property("protocol", z.literal("https:" as string, "Only HTTPS allowed"))
);
blobSchema.parse(new URL("https://example.com")); // ✅
blobSchema.parse(new URL("http://example.com")); // ❌
The z.property() API works with any data type (but it's most useful when used in conjunction with z.instanceof()).
const blobSchema = z.string().check(
z.property("length", z.number().min(10))
);
blobSchema.parse("hello there!"); // ✅
blobSchema.parse("hello."); // ❌
Every Zod schema stores an array of refinements. Refinements are a way to perform custom validation that Zod doesn't provide a native API for.
.refine(){/* <Callout> Checks do not (in fact, cannot) change the inferred type of the schema. </Callout>
.refine() */}<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const myString = z.string().refine((val) => val.length <= 255);
errorTo customize the error message:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const myString = z.string().refine((val) => val.length > 8, {
error: "Too short!"
});
abortBy default, validation issues from checks are considered continuable; that is, Zod will execute all checks in sequence, even if one of them causes a validation error. This is usually desirable, as it means Zod can surface as many errors as possible in one go.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const myString = z.string()
.refine((val) => val.length > 8, { error: "Too short!" })
.refine((val) => val === val.toLowerCase(), { error: "Must be lowercase" });
const result = myString.safeParse("OH NO");
result.error?.issues;
/* [
{ "code": "custom", "message": "Too short!" },
{ "code": "custom", "message": "Must be lowercase" }
] */
const result = z.safeParse(myString, "OH NO"); result.error?.issues; /* [ { "code": "custom", "message": "Too short!" }, { "code": "custom", "message": "Must be lowercase" } ] */
</Tab>
</Tabs>
To mark a particular refinement as *non-continuable*, use the `abort` parameter. Validation will terminate if the check fails.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const myString = z.string()
.refine((val) => val.length > 8, { error: "Too short!", abort: true })
.refine((val) => val === val.toLowerCase(), { error: "Must be lowercase", abort: true });
const result = myString.safeParse("OH NO");
result.error?.issues;
// => [{ "code": "custom", "message": "Too short!" }]
const result = z.safeParse(myString, "OH NO"); result.error?.issues; // [ { "code": "custom", "message": "Too short!" }]
</Tab>
</Tabs>
#### `path`
To customize the error path, use the `path` parameter. This is typically only useful in the context of object schemas.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const passwordForm = z
.object({
password: z.string(),
confirm: z.string(),
})
.refine((data) => data.password === data.confirm, {
error: "Passwords don't match",
path: ["confirm"], // path of error
});
This will set the path parameter in the associated issue:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const result = passwordForm.safeParse({ password: "asdf", confirm: "qwer" });
result.error.issues;
/* [{
"code": "custom",
"path": [ "confirm" ],
"message": "Passwords don't match"
}] */
To define an asynchronous refinement, just pass an async function:
const userId = z.string().refine(async (id) => {
// verify that ID exists in database
return true;
});
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const result = await userId.parseAsync("abc123");
whenNote — This is a power user feature and can absolutely be abused in ways that will increase the probability of uncaught errors originating from inside your refinements.
By default, refinements don't run if any non-continuable issues have already been encountered. Zod is careful to ensure the type signature of the value is correct before passing it into any refinement functions.
const schema = z.string().refine((val) => {
return val.length > 8
});
schema.parse(1234); // invalid_type: refinement won't be executed
In some cases, you want finer control over when refinements run. For instance consider this "password confirm" check:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const schema = z
.object({
password: z.string().min(8),
confirmPassword: z.string(),
anotherField: z.string(),
})
.refine((data) => data.password === data.confirmPassword, {
message: "Passwords do not match",
path: ["confirmPassword"],
});
schema.parse({
password: "asdf",
confirmPassword: "asdf",
anotherField: 1234 // ❌ this error will prevent the password check from running
});
schema.parse({ password: "asdf", confirmPassword: "asdf", anotherField: 1234 // ❌ this error will prevent the password check from running });
</Tab>
</Tabs>
An error on `anotherField` will prevent the password confirmation check from executing, even though the check doesn't depend on `anotherField`. To control when a refinement will run, use the `when` parameter:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const baseSchema = z.object({
password: z.string().min(8),
confirmPassword: z.string(),
anotherField: z.string(),
});
const schema = baseSchema
.refine((data) => data.password === data.confirmPassword, {
message: "Passwords do not match",
path: ["confirmPassword"],
// run if password & confirmPassword are valid
when(payload) { // [!code ++]
return baseSchema // [!code ++]
.pick({ password: true, confirmPassword: true }) // [!code ++]
.safeParse(payload.value).success; // [!code ++]
}, // [!code ++]
});
schema.parse({
password: "asdf",
confirmPassword: "asdf",
anotherField: 1234 // ❌ this error will not prevent the password check from running
});
when(payload) { // [!code ++]
// no issues with `password` or `confirmPassword` // [!code ++]
return payload.issues.every((iss) => { // [!code ++]
const firstPathEl = iss.path?.[0]; // [!code ++]
return firstPathEl !== "password" && firstPathEl !== "confirmPassword"; // [!code ++]
}); // [!code ++]
}, // [!code ++]
}));
schema.parse({ password: "asdf", confirmPassword: "asdf", anotherField: 1234 // ❌ this error will prevent the password check from running });
</Tab>
</Tabs>
### `.superRefine()`
The regular `.refine` API only generates a single issue with a `"custom"` error code, but `.superRefine()` makes it possible to create multiple issues using any of Zod's [internal issue types](https://github.com/colinhacks/zod/blob/main/packages/zod/src/v4/core/errors.ts).
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const UniqueStringArray = z.array(z.string()).superRefine((val, ctx) => {
if (val.length > 3) {
ctx.addIssue({
code: "too_big",
maximum: 3,
origin: "array",
inclusive: true,
message: "Too many items 😡",
input: val,
});
}
if (val.length !== new Set(val).size) {
ctx.addIssue({
code: "custom",
message: `No duplicates allowed.`,
input: val,
});
}
});
if (val.length !== new Set(val).size) {
ctx.addIssue({
code: "custom",
message: `No duplicates allowed.`,
input: val,
});
}
}) );
</Tab>
</Tabs>
### `.check()`
<Callout>
**Note** — The `.check()` API is a more low-level API that's generally more complex than `.superRefine()`. It can be faster in performance-sensitive code paths, but it's also more verbose.
</Callout>
<Accordions>
<Accordion title="View example">
The `.refine()` API is syntactic sugar atop a more versatile (and verbose) API called `.check()`. You can use this API to create multiple issues in a single refinement or have full control of the generated issue objects.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const UniqueStringArray = z.array(z.string()).check((ctx) => {
if (ctx.value.length > 3) {
// full control of issue objects
ctx.issues.push({
code: "too_big",
maximum: 3,
origin: "array",
inclusive: true,
message: "Too many items 😡",
input: ctx.value
});
}
// create multiple issues in one refinement
if (ctx.value.length !== new Set(ctx.value).size) {
ctx.issues.push({
code: "custom",
message: `No duplicates allowed.`,
input: ctx.value,
continue: true // make this issue continuable (default: false)
});
}
});
// create multiple issues in one refinement
if (ctx.value.length !== new Set(ctx.value).size) {
ctx.issues.push({
code: "custom",
message: No duplicates allowed.,
input: ctx.value,
continue: true // make this issue continuable (default: false)
});
}
});
</Tab>
</Tabs>
</Accordion>
</Accordions>
## Codecs
> Introduced in `[email protected]`. Refer to the dedicated [Codecs](/codecs) page for more information.
Codecs are a special kind of schema that implement *bidirectional transformations* between two other schemas.
```ts
const stringToDate = z.codec(
z.iso.datetime(), // input schema: ISO date string
z.date(), // output schema: Date object
{
decode: (isoString) => new Date(isoString), // ISO string → Date
encode: (date) => date.toISOString(), // Date → ISO string
}
);
A regular .parse() operations performs the forward transform. It calls the codec's decode function.
stringToDate.parse("2024-01-15T10:30:00.000Z"); // => Date
You can alternatively use the top-level z.decode() function. Unlike .parse() (which accepts unknown input), z.decode() expects a strongly-typed input (string in this example).
z.decode(stringToDate, "2024-01-15T10:30:00.000Z"); // => Date
To perform the reverse transform, use the inverse: z.encode().
z.encode(stringToDate, new Date("2024-01-15")); // => "2024-01-15T00:00:00.000Z"
Use z.invertCodec() to derive a new codec with the input and output schemas swapped.
const dateToString = z.invertCodec(stringToDate);
z.decode(dateToString, new Date("2024-01-15")); // => string
z.encode(dateToString, "2024-01-15T00:00:00.000Z"); // => Date
Refer to the dedicated Codecs page for more information. That page contains implementations for commonly-needed codecs that you can copy/paste into your project:
stringToNumberstringToIntstringToBigIntnumberToBigIntisoDatetimeToDateepochSecondsToDateepochMillisToDatejsonCodecutf8ToBytesbytesToUtf8base64ToBytesbase64urlToByteshexToBytesstringToURLstringToHttpURLuriComponentstringToBooleanSchemas can be chained together into "pipes". Pipes are primarily useful when used in conjunction with Transforms.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const stringToLength = z.string().pipe(z.transform(val => val.length));
stringToLength.parse("hello"); // => 5
z.parse(stringToLength, "hello"); // => 5
</Tab>
</Tabs>
## Transforms
> **Note** — For bi-directional transforms, use [codecs](/codecs).
Transforms are a special kind of schema that perform a unidirectional transformation. Instead of validating input, they accept anything and perform some transformation on the data. To define a transform:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const castToString = z.transform((val) => String(val));
castToString.parse("asdf"); // => "asdf"
castToString.parse(123); // => "123"
castToString.parse(true); // => "true"
z.parse(castToString, "asdf"); // => "asdf" z.parse(castToString, 123); // => "123" z.parse(castToString, true); // => "true"
</Tab>
</Tabs>
<Callout type="warn">
Transform functions should never throw. Thrown errors are not caught by Zod.
</Callout>
{/* The output type of the schema is inferred from the transform function:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const castToString = z.transform((val) => String(val));
type CastToString = z.infer<typeof castToString>; // string
type CastToString = z.infer<typeof castToString>; // string
</Tab>
</Tabs> */}
To perform validation logic inside a transform, use `ctx`. To report a validation issue, push a new issue onto `ctx.issues` (similar to the [`.check()`](#check) API).
```ts
const coercedInt = z.transform((val, ctx) => {
try {
const parsed = Number.parseInt(String(val));
return parsed;
} catch (e) {
ctx.issues.push({
code: "custom",
message: "Not a number",
input: val,
});
// this is a special constant with type `never`
// returning it lets you exit the transform without impacting the inferred return type
return z.NEVER;
}
});
Most commonly, transforms are used in conjunction with Pipes. This combination is useful for performing some initial validation, then transforming the parsed data into another form.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const stringToLength = z.string().pipe(z.transform(val => val.length));
stringToLength.parse("hello"); // => 5
z.parse(stringToLength, "hello"); // => 5
</Tab>
</Tabs>
### `.transform()`
Piping some schema into a transform is a common pattern, so Zod provides a convenience `.transform()` method.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const stringToLength = z.string().transform(val => val.length);
Transforms can also be async:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const idToUser = z
.string()
.transform(async (id) => {
// fetch user from database
return db.getUserById(id);
});
const user = await idToUser.parseAsync("abc123");
const user = await idToUser.parse("abc123");
</Tab>
</Tabs>
<Callout>
If you use async transforms, you must use a `.parseAsync` or `.safeParseAsync` when parsing data! Otherwise Zod will throw an error.
</Callout>
### `.preprocess()`
Piping a transform into another schema is another common pattern, so Zod provides a convenience `z.preprocess()` function.
```ts
const coercedInt = z.preprocess((val) => {
if (typeof val === "string") {
return Number.parseInt(val);
}
return val;
}, z.int());
By default, the input type of a z.preprocess() schema is unknown, since the preprocessor is expected to handle arbitrary input. To narrow the input type, annotate the preprocessor's parameter directly:
const trimmed = z.preprocess(
(val: string | null | undefined) => val?.trim() ?? "",
z.string()
);
type Input = z.input<typeof trimmed>; // string | null | undefined
type Output = z.output<typeof trimmed>; // string
This is useful when integrating with libraries like react-hook-form that derive their form value type from z.input<>.
To set a default value for a schema:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const defaultTuna = z.string().default("tuna");
defaultTuna.parse(undefined); // => "tuna"
defaultTuna.parse(undefined); // => "tuna"
</Tab>
</Tabs>
Alternatively, you can pass a function which will be re-executed whenever a default value needs to be generated:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const randomDefault = z.number().default(Math.random);
randomDefault.parse(undefined); // => 0.4413456736055323
randomDefault.parse(undefined); // => 0.1871840107401901
randomDefault.parse(undefined); // => 0.7223408162401552
z.parse(randomDefault, undefined); // => 0.4413456736055323 z.parse(randomDefault, undefined); // => 0.1871840107401901 z.parse(randomDefault, undefined); // => 0.7223408162401552
</Tab>
</Tabs>
## Prefaults
In Zod, setting a *default* value will short-circuit the parsing process. If the input is `undefined`, the default value is eagerly returned. As such, the default value must be assignable to the *output type* of the schema.
```ts
const schema = z.string().transform(val => val.length).default(0);
schema.parse(undefined); // => 0
Sometimes, it's useful to define a prefault ("pre-parse default") value. If the input is undefined, the prefault value will be parsed instead. The parsing process is not short circuited. As such, the prefault value must be assignable to the input type of the schema.
z.string().transform(val => val.length).prefault("tuna");
schema.parse(undefined); // => 4
This is also useful if you want to pass some input value through some mutating refinements.
const a = z.string().trim().toUpperCase().prefault(" tuna ");
a.parse(undefined); // => "TUNA"
const b = z.string().trim().toUpperCase().default(" tuna ");
b.parse(undefined); // => " tuna "
Use .catch() to define a fallback value to be returned in the event of a validation error:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const numberWithCatch = z.number().catch(42);
numberWithCatch.parse(5); // => 5
numberWithCatch.parse("tuna"); // => 42
numberWithCatch.parse(5); // => 5 numberWithCatch.parse("tuna"); // => 42
</Tab>
</Tabs>
Alternatively, you can pass a function which will be re-executed whenever a catch value needs to be generated.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}>
<Tab value="Zod">
```ts
const numberWithRandomCatch = z.number().catch((ctx) => {
ctx.error; // the caught ZodError
return Math.random();
});
numberWithRandomCatch.parse("sup"); // => 0.4413456736055323
numberWithRandomCatch.parse("sup"); // => 0.1871840107401901
numberWithRandomCatch.parse("sup"); // => 0.7223408162401552
z.parse(numberWithRandomCatch, "sup"); // => 0.4413456736055323 z.parse(numberWithRandomCatch, "sup"); // => 0.1871840107401901 z.parse(numberWithRandomCatch, "sup"); // => 0.7223408162401552
</Tab>
</Tabs>
## Branded types
TypeScript's type system is [structural](https://www.typescriptlang.org/docs/handbook/type-compatibility.html), meaning that two types that are structurally equivalent are considered the same.
```ts
type Cat = { name: string };
type Dog = { name: string };
const pluto: Dog = { name: "pluto" };
const simba: Cat = pluto; // works fine
In some cases, it can be desirable to simulate nominal typing inside TypeScript. This can be achieved with branded types (also known as "opaque types").
const Cat = z.object({ name: z.string() }).brand<"Cat">();
const Dog = z.object({ name: z.string() }).brand<"Dog">();
type Cat = z.infer<typeof Cat>; // { name: string } & z.$brand<"Cat">
type Dog = z.infer<typeof Dog>; // { name: string } & z.$brand<"Dog">
const pluto = Dog.parse({ name: "pluto" });
const simba: Cat = pluto; // ❌ not allowed
Under the hood, this works by attaching a "brand" to the schema's inferred type.
const Cat = z.object({ name: z.string() }).brand<"Cat">();
type Cat = z.output<typeof Cat>; // { name: string } & z.$brand<"Cat">
With this brand, plain (unbranded) data structures are not assignable to the inferred type. You have to parse some data with the schema to get branded data.
Note that branded types do not affect the runtime result of
.parse. It is a static-only construct.
By default, only the output type is branded.
const USD = z.string().brand<"USD">();
type USDOutput = z.output<typeof USD>; // string & z.$brand<"USD">
type USDInput = z.input<typeof USD>; // string
To customize this, pass a second generic to .brand() to specify the direction of the brand.
// requires Zod 4.2+
z.string().brand<"Cat", "out">(); // output is branded (default)
z.string().brand<"Cat", "in">(); // input is branded
z.string().brand<"Cat", "inout">(); // both are branded
To mark a schema as readonly:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const ReadonlyUser = z.object({ name: z.string() }).readonly();
type ReadonlyUser = z.infer<typeof ReadonlyUser>;
// Readonly<{ name: string }>
The inferred type is marked as readonly. Note that in TypeScript, this only affects objects, arrays, tuples, Set, and Map:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
z.object({ name: z.string() }).readonly(); // { readonly name: string }
z.array(z.string()).readonly(); // readonly string[]
z.tuple([z.string(), z.number()]).readonly(); // readonly [string, number]
z.map(z.string(), z.date()).readonly(); // ReadonlyMap<string, Date>
z.set(z.string()).readonly(); // ReadonlySet<string>
Inputs will be parsed like normal, then the result will be frozen with Object.freeze() to prevent modifications.
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
const result = ReadonlyUser.parse({ name: "fido" });
result.name = "simba"; // throws TypeError
To validate any JSON-encodable value:
const jsonSchema = z.json();
This is a convenience API that returns the following union schema:
const jsonSchema = z.lazy(() => {
return z.union([
z.string(params),
z.number(),
z.boolean(),
z.null(),
z.array(jsonSchema),
z.record(z.string(), jsonSchema)
]);
});
Zod provides a z.function() utility for defining Zod-validated functions. This way, you can avoid intermixing validation code with your business logic.
const MyFunction = z.function({
input: [z.string()], // parameters (must be an array or a ZodTuple)
output: z.number() // return type
});
type MyFunction = z.infer<typeof MyFunction>;
// (input: string) => number
Function schemas have an .implement() method which accepts a function and returns a new function that automatically validates its inputs and outputs.
const computeTrimmedLength = MyFunction.implement((input) => {
// TypeScript knows input is a string!
return input.trim().length;
});
computeTrimmedLength("sandwich"); // => 8
computeTrimmedLength(" asdf "); // => 4
This function will throw a ZodError if the input is invalid:
computeTrimmedLength(42); // throws ZodError
If you only care about validating inputs, you can omit the output field.
const MyFunction = z.function({
input: [z.string()], // parameters (must be an array or a ZodTuple)
});
const computeTrimmedLength = MyFunction.implement((input) => input.trim.length);
Use the .implementAsync() method to create an async function.
const computeTrimmedLengthAsync = MyFunction.implementAsync(
async (input) => input.trim().length
);
computeTrimmedLengthAsync("sandwich"); // => Promise<8>
You can create a Zod schema for any TypeScript type by using z.custom(). This is useful for validating types from third-party libraries or any other type that isn't covered by a built-in schema. For class instances, prefer z.instanceof(); for template literal types, prefer z.templateLiteral().
import { Decimal } from "decimal.js";
const decimalSchema = z.custom<Decimal>((val) => Decimal.isDecimal(val));
decimalSchema.parse(new Decimal("1.5")); // passes
decimalSchema.parse("1.5"); // throws
If you don't provide a validation function, Zod will allow any value. This can be dangerous!
z.custom<{ arg: string }>(); // performs no validation
You can customize the error message and other options by passing a second argument. This parameter works the same way as the params parameter of .refine.
z.custom<...>((val) => ..., "custom error message");
Use .apply() to incorporate external functions into Zod's method chain:
<Tabs groupId="lib" items={["Zod", "Zod Mini"]}> <Tab value="Zod">
function setCommonNumberChecks<T extends z.ZodNumber>(schema: T) {
return schema
.min(0)
.max(100);
}
const schema = z.number()
.apply(setCommonNumberChecks)
.nullable();
schema.parse(0); // => 0
schema.parse(-1); // ❌ throws
schema.parse(101); // ❌ throws
schema.parse(null); // => null
const schema = z.nullable( z.number().apply(setCommonNumberChecks) );
z.parse(schema, 0); // => 0 z.parse(schema, -1); // ❌ throws z.parse(schema, 101); // ❌ throws z.parse(schema, null); // => null
</Tab>
</Tabs>