typebox-legacy
Install the dependencies
npm
yarn
pnpm
bun
npm i drizzle-orm@rc @sinclair/typebox
Select schema
Defines the shape of data queried from the database - can be used to validate API responses.
import { int, mysqlTable, text } from 'drizzle-orm/mysql-core';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const users = mysqlTable('users', {
id: int().primaryKey().autoincrement(),
name: text().notNull(),
age: int().notNull()
});
const userSelectSchema = createSelectSchema(users);
const rows = await db.select({ id: users.id, name: users.name }).from(users).limit(1);
const parsed: { id: number; name: string; age: number } = Value.Parse(userSelectSchema, rows[0]); // Error: `age` is not returned in the above query
const rows = await db.select().from(users).limit(1);
const parsed: { id: number; name: string; age: number } = Value.Parse(userSelectSchema, rows[0]); // Will parse successfullyViews are also supported.
import { mysqlView } from 'drizzle-orm/mysql-core';
import { gt } from 'drizzle-orm';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const usersView = mysqlView('users_view').as((qb) => qb.select().from(users).where(gt(users.age, 18)));
const usersViewSchema = createSelectSchema(usersView);
const parsed: { id: number; name: string; age: number } = Value.Parse(usersViewSchema, ...);Insert schema
Defines the shape of data to be inserted into the database - can be used to validate API requests.
import { int, mysqlTable, text } from 'drizzle-orm/mysql-core';
import { createInsertSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const users = mysqlTable('users', {
id: int().primaryKey().autoincrement(),
name: text().notNull(),
age: int().notNull()
});
const userInsertSchema = createInsertSchema(users);
const user = { name: 'John' };
const parsed: { id?: number | undefined, name: string, age: number } = Value.Parse(userInsertSchema, user); // Error: `age` is not defined
const user = { name: 'Jane', age: 30 };
const parsed: { id?: number | undefined, name: string, age: number } = Value.Parse(userInsertSchema, user); // Will parse successfully
await db.insert(users).values(parsed);Update schema
Defines the shape of data to be updated in the database - can be used to validate API requests.
import { int, mysqlTable, text } from 'drizzle-orm/mysql-core';
import { createUpdateSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
import { eq } from "drizzle-orm";
const users = mysqlTable('users', {
id: int().primaryKey().autoincrement(),
name: text().notNull(),
age: int().notNull()
});
const userUpdateSchema = createUpdateSchema(users);
const user = { age: 35 };
const parsed: { id?: number | undefined, name?: string | undefined, age?: number | undefined } = Value.Parse(userUpdateSchema, user); // Will parse successfully
await db.update(users).set(parsed).where(eq(users.name, 'Jane'));Refinements
Each create schema function accepts an additional optional parameter that you can used to extend, modify or completely overwite a field’s schema. Defining a callback function will extend or modify while providing a Typebox schema will overwrite it.
import { int, json, mysqlTable, text } from 'drizzle-orm/mysql-core';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Type } from '@sinclair/typebox';
import { Value } from '@sinclair/typebox/value';
const users = mysqlTable('users', {
id: int().primaryKey().autoincrement(),
name: text().notNull(),
bio: text(),
preferences: json()
});
const userSelectSchema = createSelectSchema(users, {
name: (schema) => Type.String({ ...schema, maxLength: 20 }), // Extends schema
bio: (schema) => Type.String({ ...schema, maxLength: 1000 }), // Extends schema before becoming nullable/optional
preferences: Type.Object({ theme: Type.String() }) // Overwrites the field, including its nullability
});
const parsed: {
id: number;
name: string,
bio: string | null;
preferences: {
theme: string;
};
} = Value.Parse(userSelectSchema, ...);Factory functions
For more advanced use cases, you can use the createSchemaFactory function.
Use case: Using an extended Typebox instance
import { int, mysqlTable, text } from 'drizzle-orm/mysql-core';
import { createSchemaFactory } from 'drizzle-orm/typebox';
import { t } from 'elysia'; // Extended Typebox instance
const users = mysqlTable('users', {
id: int().primaryKey().autoincrement(),
name: text().notNull(),
age: int().notNull()
});
const { createInsertSchema } = createSchemaFactory({ typeboxInstance: t });
const userInsertSchema = createInsertSchema(users, {
// We can now use the extended instance
name: (schema) => t.Number({ ...schema, error: "`name` must be a string" })
});Data type reference
mysql.boolean();
// Schema
Type.Boolean();mysql.mysqlEnum('name', ['val1', 'val2']);
// Schema
Type.Enum({ 'val1': 'val1', 'val2': 'val2' });mysql.date({ mode: 'date' });
mysql.datetime({ mode: 'date' });
mysql.timestamp({ mode: 'date' });
// Schema
Type.Date();mysql.binary();
mysql.date({ mode: 'string' });
mysql.datetime({ mode: 'string' });
mysql.decimal();
mysql.time();
mysql.timestamp({ mode: 'string' });
mysql.varbinary();
// Schema
Type.String();mysql.tinyblob({ mode: 'string' });
mysql.tinyblob();
Type.String({ maxLength: 255 })mysql.blob({ mode: 'string' });
mysql.blob();
Type.String({ maxLength: 65_535 })mysql.mediumblob({ mode: 'string' });
mysql.mediumblob();
Type.String({ maxLength: 16_777_215 })mysql.longblob({ mode: 'string' });
mysql.longblob();
Type.String({ maxLength: 4_294_967_295 })mysql.char({ length: ... });
// Schema
Type.String({ minLength: length, maxLength: length });mysql.varchar({ length: ... });
// Schema
Type.String({ maxLength: length });binary({ length: ... });
// Schema
Type.RegExp(/^[01]*$/, { maxLength: length })mysql.varbinary({ length: ... });
// Schema
Type.RegExp(/^[01]*$/, { maxLength: length })mysql.tinytext();
// Schema
Type.String({ maxLength: 255 }); // unsigned 8-bit integer limitmysql.text();
// Schema
Type.String({ maxLength: 65_535 }); // unsigned 16-bit integer limitmysql.mediumtext();
// Schema
Type.String({ maxLength: 16_777_215 }); // unsigned 24-bit integer limitmysql.longtext();
// Schema
Type.String({ maxLength: 4_294_967_295 }); // unsigned 32-bit integer limitmysql.tinytext({ enum: ... });
mysql.mediumtext({ enum: ... });
mysql.text({ enum: ... });
mysql.longtext({ enum: ... });
mysql.char({ enum: ... });
mysql.varchar({ enum: ... });
// Schema
Type.Enum(enum);mysql.tinyint();
// Schema
Type.Integer({ minimum: -128, maximum: 127 }); // 8-bit integer lower and upper limitmysql.tinyint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 255 }); // unsigned 8-bit integer lower and upper limitmysql.smallint();
// Schema
Type.Integer({ minimum: -32_768, maximum: 32_767 }); // 16-bit integer lower and upper limitmysql.smallint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 65_535 }); // unsigned 16-bit integer lower and upper limitmysql.float();
// Schema
Type.Number().min(-8_388_608).max(8_388_607); // 24-bit integer lower and upper limitmysql.mediumint();
// Schema
Type.Integer({ minimum: -8_388_608, maximum: 8_388_607 }); // 24-bit integer lower and upper limitmysql.float({ unsigned: true });
// Schema
Type.Number({ minimum: 0, maximum: 16_777_215 }); // unsigned 24-bit integer lower and upper limitmysql.mediumint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 16_777_215 }); // unsigned 24-bit integer lower and upper limitmysql.int();
// Schema
Type.Integer({ minimum: -2_147_483_648, maximum: 2_147_483_647 }); // 32-bit integer lower and upper limitmysql.int({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 4_294_967_295 }); // unsgined 32-bit integer lower and upper limitmysql.double();
mysql.real();
// Schema
Type.Number({ minimum: -140_737_488_355_328, maximum: 140_737_488_355_327 }); // 48-bit integer lower and upper limitmysql.double({ unsigned: true });
// Schema
Type.Numer({ minimum: 0, maximum: 281_474_976_710_655 }); // unsigned 48-bit integer lower and upper limitmysql.decimal({ mode: 'number' });
// Schema
Type.Number({ minimum: -9_007_199_254_740_991, maximum: 9_007_199_254_740_991 });mysql.decimal({ mode: 'bigint' });
// Schema
Type.BigInt({ minimum: -9_223_372_036_854_775_808n, maximum: 9_223_372_036_854_775_807n });mysql.decimal({ mode: 'number', unsigned: true });
// Schema
Type.Number({ minimum: 0, maximum: 9_007_199_254_740_991 });mysql.decimal({ mode: 'bigint', unsigned: true });
// Schema
Type.BigInt({ minimum: 0, maximum: 18_446_744_073_709_551_615n });mysql.bigint({ mode: 'number' });
// Schema
Type.Integer({ minimum: -9_007_199_254_740_991, maximum: 9_007_199_254_740_991 }); // Javascript min. and max. safe integersmysql.bigint({ mode: 'number', unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 9_007_199_254_740_991 }); // Javascript min. and max. safe integersmysql.bigint({ mode: 'string' });
// Schema
TypeRegistry.Set('BigIntStringMode', (_, value) => {
if (typeof value !== 'string' || !(/^-?\d+$/.test(value))) {
return false;
}
const bigint = BigInt(value);
if (bigint < -9_223_372_036_854_775_808n || bigint > 9_223_372_036_854_775_807n) {
return false;
}
return true;
});
{ [Kind]: 'BigIntStringMode', type: 'string' }mysql.bigint({ mode: 'string', unsigned: true });
// Schema
TypeRegistry.Set('UnsignedBigIntStringMode', (_, value) => {
if (typeof value !== 'string' || !(/^\d+$/.test(value))) {
return false;
}
const bigint = BigInt(value);
if (bigint < 0 || bigint > 9_223_372_036_854_775_807n) {
return false;
}
return true;
});
{ [Kind]: 'UnsignedBigIntStringMode', type: 'string' }mysql.bigint({ mode: 'bigint' });
// Schema
Type.BigInt({ minimum: -9_223_372_036_854_775_808n, maximum: 9_223_372_036_854_775_807n }); // 64-bit integer lower and upper limitmysql.bigint({ mode: 'bigint', unsigned: true });
// Schema
Type.BigInt({ minimum: 0, maximum: 18_446_744_073_709_551_615n }); // unsigned 64-bit integer lower and upper limitmysql.serial();
Type.Integer({ minimum: 0, maximum: 9_007_199_254_740_991 })mysql.year();
// Schema
Type.Integer({ minimum: 1_901, maximum: 2_155 });mysql.json();
// Schema
Type.Recursive((self) => Type.Union([Type.Union([Type.String(), Type.Number(), Type.Boolean(), Type.Null()]), Type.Array(self), Type.Record(Type.String(), self)]));