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');

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.

ZodValibot
andintersect
catchfallback
catchallobjectWithRest
coercepipe, unknown and transform
datetimeisoDate, isoDateTime
defaultoptional
discriminatedUnionvariant
elementitem
enumpicklist
extendObject merging
gt, gteminValue
inferInferOutput
intinteger
inputInferInput
instanceofinstance
intersectionintersect
lt, ltemaxValue
maxmaxLength, maxSize, maxValue
minminLength, minSize, minValue
nativeEnumenum
negativemaxValue
nonnegativeminValue
nonpositivemaxValue
orunion
outputInferOutput
passthroughlooseObject
positiveminValue
refinecheck
resttuple
safesafeInteger
shapeentries
strictstrictObject
stripobject
superRefinerawCheck, 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 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));

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.

Contributors

Thanks to all the contributors who helped make this page better!

  • GitHub profile picture of fabian-hiller
  • GitHub profile picture of valerii15298
  • GitHub profile picture of morinokami

Partners

Thanks to our partners who support the project ideally and financially.

Sponsors

Thanks to our GitHub sponsors who support the project financially.

  • GitHub profile picture of antfu
  • GitHub profile picture of Thanaen
  • GitHub profile picture of osdiab
  • GitHub profile picture of ruiaraujo012
  • GitHub profile picture of hyunbinseo
  • GitHub profile picture of F0rce
  • GitHub profile picture of Unique-Pixels
  • GitHub profile picture of jdgamble555
  • GitHub profile picture of nickytonline
  • GitHub profile picture of KubaJastrz
  • GitHub profile picture of caegdeveloper
  • GitHub profile picture of dslatkin