Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
The zod-to-ts npm package is a utility that converts Zod schemas to TypeScript interfaces. This is particularly useful for ensuring type safety and consistency between your Zod validation schemas and TypeScript types.
Convert Zod Schema to TypeScript Interface
This feature allows you to convert a Zod schema into a TypeScript interface. In the example, a Zod schema for a User object is defined and then converted into a TypeScript interface named 'User'.
const { z } = require('zod');
const { zodToTs } = require('zod-to-ts');
const UserSchema = z.object({
id: z.number(),
name: z.string(),
email: z.string().email()
});
const { tsInterface } = zodToTs(UserSchema, 'User');
console.log(tsInterface);
Generate TypeScript Definitions from Zod Schemas
This feature allows you to generate TypeScript definitions from Zod schemas. In the example, a Zod schema for a Product object is defined and then converted into a TypeScript interface named 'Product'.
const { z } = require('zod');
const { zodToTs } = require('zod-to-ts');
const ProductSchema = z.object({
id: z.number(),
name: z.string(),
price: z.number()
});
const { tsInterface } = zodToTs(ProductSchema, 'Product');
console.log(tsInterface);
io-ts is a runtime type system for IO decoding/encoding and validation. It provides similar functionality to zod-to-ts in terms of defining and validating schemas, but it also includes encoding and decoding capabilities. Unlike zod-to-ts, io-ts does not directly convert schemas to TypeScript interfaces.
typescript-json-schema is a tool to generate JSON schema from your TypeScript types. It works in the opposite direction of zod-to-ts, converting TypeScript types to JSON schemas rather than Zod schemas to TypeScript interfaces.
class-validator is a library for validation using TypeScript decorators. It allows you to define validation rules directly in your TypeScript classes. While it does not convert schemas to TypeScript interfaces, it provides a way to enforce validation rules within your TypeScript code.
generate TypeScript types from your Zod schema
npm install zod-to-ts zod typescript
You must be on zod@3 and typescript@4.
import { z } from 'zod'
import { zodToTs } from 'zod-to-ts'
// define your Zod schema
const UserSchema = z.object({
username: z.string(),
age: z.number(),
inventory: z.object({
name: z.string(),
itemId: z.number(),
}).array(),
})
// pass schema and name of type/identifier
const { node } = zodToTs(UserSchema, 'User')
result:
{
username: string
age: number
inventory: {
name: string
itemId: number
}[]
}
You must pass in the identifier User
or it will default to Identifier
. This is necessary to handle cases like recursive types and native enums. zodToTs()
only returns the type value, not the actual type declaration. If you want to add an identifier to the type and create a type declaration, you can use the createTypeAlias()
utility:
import { createTypeAlias, zodToTs } from 'zod-to-ts'
const identifier = 'User'
const { node } = zodToTs(UserSchema, identifier)
const typeAlias = createTypeAlias(
node,
identifier,
// optionally pass a comment
// comment: UserSchema.description
)
result:
type User = {
username: string
}
zodToTs()
and createTypeAlias()
return a TS AST nodes, so if you want to get the node as a string, you can use the printNode()
utility.
zodToTs()
:
import { printNode, zodToTs } from 'zod-to-ts'
const identifier = 'User'
const { node } = zodToTs(UserSchema, identifier)
const nodeString = printNode(node)
result:
"{
username: string
age: number
inventory: {
name: string
itemId: number
}[]
}"
createTypeAlias()
:
import { createTypeAlias, printNode, zodToTs } from 'zod-to-ts'
const identifier = 'User'
const { node } = zodToTs(UserSchema, identifier)
const typeAlias = createTypeAlias(node, identifier)
const nodeString = printNode(typeAlias)
result:
"type User = {
username: string
age: number
inventory: {
name: string
itemId: number
}[]
}"
You can use withGetType
to override a type, which is useful when more information is needed to determine the actual type. Unfortunately, this means working with the TS AST:
import { z } from 'zod'
import { withGetType, zodToTs } from 'zod-to-ts'
const DateSchema = withGetType(
z.instanceof(Date),
(ts) => ts.factory.createIdentifier('Date'),
)
const ItemSchema = z.object({
name: z.string(),
date: DateSchema,
})
const { node } = zodToTs(ItemSchema, 'Item')
result without withGetType
override:
type Item = {
name: string
date: any
}
result with override:
type Item = {
name: string
date: Date
}
TypeScript AST Viewer can help a lot with this if you are having trouble referencing something. It even provides copy-pastable code!
Lazy types default to referencing the root type (User
in the following example). It is impossible to determine what it is referencing otherwise.
// Zod cannot infer types when you use the z.lazy
// so you must define it
import { z } from 'zod'
type User = {
username: string
friends: User[]
}
const UserSchema: z.ZodSchema<User> = z.object({
username: z.string(),
friends: z.lazy(() => UserSchema).array(),
})
const { node } = zodToTs(UserSchema, 'User')
result:
type User = {
username: string
friends: User[]
}
But what happens when the schema looks like this?
type User = {
username: string
item: {
name: string
itemId: string
}
friends: User[]
}
// essentially when you are referencing a different field
// and not the root type
const friendItems = z.lazy(() => UserSchema.item).array()
const UserSchema: z.ZodSchema<User> = z.object({
username: z.string(),
item: z.object({
name: z.string(),
id: z.number(),
}),
friendItems,
})
const { node } = zodToTs(UserSchema, 'User')
result:
{
username: string
item: {
name: string
id: number
}
friendItems: User[]
}
friendItems
will still have the User
type even though it is actually referencing UserSchema["item"]
. You must provide the actual type using withGetType
:
import { z } from 'zod'
import { withGetType } from 'zod-to-ts'
type User = {
username: string
item: {
name: string
id: number
}
friends: User[]
}
const friendItems: z.Schema<User['item'][]> = withGetType(
z.lazy(() => UserSchema.item).array(),
// return a TS AST node
(ts, identifier) =>
ts.factory.createIndexedAccessTypeNode(
ts.factory.createTypeReferenceNode(
ts.factory.createIdentifier(identifier),
undefined,
),
ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral('item')),
),
)
const UserSchema: z.ZodSchema<User> = z.object({
username: z.string(),
item: z.object({
name: z.string(),
id: z.number(),
}),
friendItems,
})
const { node } = zodToTs(UserSchema, 'User')
result:
{
username: string
item: {
name: string
id: number
}
friendItems: User['item'][]
}
z.enum()
is always preferred, but sometimes z.nativeEnum()
is necessary. z.nativeEnum()
works similarly to z.lazy()
in that the identifier of the enum cannot be determined:
import { z } from 'zod'
import { withGetType } from 'zod-to-ts'
enum Fruit {
Apple = 'apple',
Banana = 'banana',
Cantaloupe = 'cantaloupe',
}
const fruitNativeEnum: = z.nativeEnum(
Fruit,
)
const TreeSchema = z.object({
fruit: fruitNativeEnum,
})
result:
{
fruit: unknown
}
There are three ways to solve this: provide an identifier to it or resolve all the enums inside zodToTs()
.
Option 1 - providing an identifier using withGetType()
:
import { z } from 'zod'
import { withGetType, zodToTs } from 'zod-to-ts'
enum Fruit {
Apple = 'apple',
Banana = 'banana',
Cantaloupe = 'cantaloupe',
}
const fruitNativeEnum = withGetType(
z.nativeEnum(
Fruit,
),
// return an identifier that will be used on the enum type
(ts) => ts.factory.createIdentifier('Fruit'),
)
const TreeSchema = z.object({
fruit: fruitNativeEnum,
})
const { node } = zodToTs(TreeSchema)
result:
{
fruit: Fruit
}
Option 2 - resolve enums. This is the same as before, but you just need to pass an option:
const TreeTSType = zodToTs(TreeSchema, undefined, { nativeEnums: 'resolve' })
result:
{
node: {
fruit: Fruit
},
store: {
nativeEnums: [
enum Fruit {
Apple = 'apple',
Banana = 'banana',
Cantaloupe = 'cantaloupe',
}
]
}
}
Note: These are not the actual values, they are TS representation. The actual values are TS AST nodes.
This option allows you to embed the enums before the schema without actually depending on an external enum type.
Option 3 - convert to union. This is the same as how ZodEnum created by z.enum([...]) is handled, but need to pass an option:
const { node } = zodToTs(TreeSchema, undefined, {
nativeEnums: 'union',
})
result:
{
fruit: 'apple' | 'banana' | 'cantaloupe'
}
Note: These are not the actual values, they are TS representation. The actual values are TS AST nodes.
FAQs
generate TypeScript types from your Zod schema
The npm package zod-to-ts receives a total of 278,208 weekly downloads. As such, zod-to-ts popularity was classified as popular.
We found that zod-to-ts demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.