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.
Official codemod
To make the migration as smoth as possible, we have created an official codemod that automatically migrates your Zod schemas to Valibot. Just copy your schemas into this editor and click play.
The codemod is still in beta and may not cover all edge cases. If you encounter any problems or unexpected behaviour, please create an issue. Alternatively, you can try to fix any issues yourself and create a pull request. You can find the source code here.
You will soon be able to also run the codemod locally in your terminal to migrate your entire codebase at once. Stay tuned!
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');
We recommend that you read our mental model guide to understand how the individual functions of Valibot's modular API work together.
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 | gtValue |
gte | minValue |
infer | InferOutput |
int | integer |
input | InferInput |
instanceof | instance |
intersection | intersect |
lt | ltValue |
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 , forward |
rest | tuple |
safe | safeInteger |
shape | entries |
strict | strictObject |
strip | object |
superRefine | rawCheck , rawTransform |
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 StringSchema = 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));
Instead of unknown
as in the previous example, we usually recommend using a specific schema such as string
to improve type safety. This allows you, for example, to validate the formatting of the string with decimal
before transforming it to a number.
const NumberSchema = v.pipe(v.string(), v.decimal(), 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.