Migrate from Zod
Migrating from Zod to Valibot is very easy in most cases since both APIs have a lot of similarities. The following guide will help you migrate step by step and also point out important differences.
If anything is unclear or missing, please create an issue on GitHub. We are very interested in making this guide as good as possible.
Replace imports
The first thing to do after installing Valibot is to update your imports. Just change your Zod imports to Valibot's and replace all occurrences of z.
with v.
.
// Change this
import { z } from 'zod';
const Schema = z.object({ key: z.string() });
// To this
import * as v from 'valibot';
const Schema = v.object({ key: v.string() });
Restructure code
One of the biggest differences between Zod and Valibot is the way you further validate a given type. In Zod, you chain methods like .email
and .endsWith
. In Valibot you use pipelines to do the same thing. This is a function that starts with a schema and is followed by up to 19 validation or transformation actions.
// Change this
const Schema = z.string().email().endsWith('@example.com');
// To this
const Schema = v.pipe(v.string(), v.email(), v.endsWith('@example.com'));
Due to the modular design of Valibot, also all other methods like .parse
or .safeParse
have to be used a little bit differently. Instead of chaining them, you usually pass the schema as the first argument and move any existing arguments one position to the right.
// Change this
const value = z.string().parse('foo');
// To this
const value = v.parse(v.string(), 'foo');
Change names
Most of the names are the same as in Zod. However, there are some exceptions. The following table shows all names that have changed.
Zod | Valibot |
---|---|
and | intersect |
catch | fallback |
catchall | objectWithRest |
coerce | pipe , unknown and transform |
datetime | isoDate , isoDateTime |
default | optional |
discriminatedUnion | variant |
element | item |
enum | picklist |
extend | Object merging |
gt , gte | minValue |
infer | InferOutput |
int | integer |
input | InferInput |
instanceof | instance |
intersection | intersect |
lt , lte | maxValue |
max | maxLength , maxSize , maxValue |
min | minLength , minSize , minValue |
nativeEnum | enum |
negative | maxValue |
nonnegative | minValue |
nonpositive | maxValue |
or | union |
output | InferOutput |
passthrough | looseObject |
positive | minValue |
refine | check |
rest | tuple |
safe | safeInteger |
shape | entries |
strict | strictObject |
strip | object |
Other details
Below are some more details that may be helpful when migrating from Zod to Valibot.
Object and tuple
To specify whether objects or tuples should allow or prevent unknown values, Valibot uses different schema functions. Zod uses the methods .passthrough
, .strict
, .strip
, .catchall
and .rest
instead. See the objects and arrays guide for more details.
// Change this
const ObjectSchema = z.object({ key: z.string() }).strict();
// To this
const ObjectSchema = v.strictObject({ key: v.string() });
Error messages
For individual error messages, you can pass a string or an object to Zod. It also allows you to differentiate between an error message for "required" and "invalid_type". With Valibot you just pass a single string instead.
// Change this
const SchemaSchema = z
.string({ invalid_type_error: 'Not a string' })
.min(5, { message: 'Too short' });
// To this
const StringSchema = v.pipe(
v.string('Not a string'),
v.minLength(5, 'Too short')
);
Coerce type
To enforce primitive values, you can use a method of the coerce
object in Zod. There is no such object or function in Valibot. Instead, you use a pipeline with a transform
action as the second argument. This forces you to explicitly define the input, resulting in safer code.
// Change this
const NumberSchema = z.coerce.number();
// To this
const NumberSchema = v.pipe(v.unknown(), v.transform(Number));
Async validation
Similar to Zod, Valibot supports synchronous and asynchronous validation. However, the API is a little bit different. See the async guide for more details.