从 drizzle-orm@1.0.0-beta.15 开始,drizzle-typebox 被弃用,建议使用 Drizzle ORM 自身内置的一级架构生成支持
你仍然可以使用 drizzle-typebox 包,但是所有新的更新将直接添加到 Drizzle ORM 中
此版本的 typebox 是基于使用 @sinclair/typebox 包的传统版本
从 drizzle-orm@1.0.0-beta.15 开始,drizzle-typebox 被弃用,建议使用 Drizzle ORM 自身内置的一级架构生成支持
你仍然可以使用 drizzle-typebox 包,但是所有新的更新将直接添加到 Drizzle ORM 中
此版本的 typebox 是基于使用 @sinclair/typebox 包的传统版本
npm i drizzle-orm @sinclair/typebox
定义从数据库查询到的数据形状 - 可用于验证 API 响应。
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const users = pgTable('users', {
id: integer().generatedAlwaysAsIdentity().primaryKey(),
name: text().notNull(),
age: integer().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]); // 错误:以上查询中未返回 `age`
const rows = await db.select().from(users).limit(1);
const parsed: { id: number; name: string; age: number } = Value.Parse(userSelectSchema, rows[0]); // 解析成功视图和枚举也受支持。
import { pgEnum } from 'drizzle-orm/pg-core';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const roles = pgEnum('roles', ['admin', 'basic']);
const rolesSchema = createSelectSchema(roles);
const parsed: 'admin' | 'basic' = Value.Parse(rolesSchema, ...);
const usersView = pgView('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, ...);定义插入到数据库中的数据形状 - 可用于验证 API 请求。
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
import { createInsertSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const users = pgTable('users', {
id: integer().generatedAlwaysAsIdentity().primaryKey(),
name: text().notNull(),
age: integer().notNull()
});
const userInsertSchema = createInsertSchema(users);
const user = { name: 'John' };
const parsed: { name: string, age: number } = Value.Parse(userInsertSchema, user); // 错误:未定义 `age`
const user = { name: 'Jane', age: 30 };
const parsed: { name: string, age: number } = Value.Parse(userInsertSchema, user); // 解析成功
await db.insert(users).values(parsed);定义在数据库中更新的数据形状 - 可用于验证 API 请求。
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
import { createUpdateSchema } from 'drizzle-orm/typebox-legacy';
import { Value } from '@sinclair/typebox/value';
const users = pgTable('users', {
id: integer().generatedAlwaysAsIdentity().primaryKey(),
name: text().notNull(),
age: integer().notNull()
});
const userUpdateSchema = createUpdateSchema(users);
const user = { id: 5, name: 'John' };
const parsed: { name?: string | undefined, age?: number | undefined } = Value.Parse(userUpdateSchema, user); // 错误:`id` 是生成列,不能更新
const user = { age: 35 };
const parsed: { name?: string | undefined, age?: number | undefined } = Value.Parse(userUpdateSchema, user); // 解析成功
await db.update(users).set(parsed).where(eq(users.name, 'Jane'));每个创建 schema 的函数都接受一个可选的额外参数,可以用来扩展、修改或完全覆盖字段的 schema。定义回调函数则是扩展或修改,提供 Typebox schema 则是覆盖。
import { pgTable, text, integer, json } from 'drizzle-orm/pg-core';
import { createSelectSchema } from 'drizzle-orm/typebox-legacy';
import { Type } from '@sinclair/typebox';
import { Value } from '@sinclair/typebox/value';
const users = pgTable('users', {
id: integer().generatedAlwaysAsIdentity().primaryKey(),
name: text().notNull(),
bio: text(),
preferences: json()
});
const userSelectSchema = createSelectSchema(users, {
name: (schema) => Type.String({ ...schema, maxLength: 20 }), // 扩展 schema
bio: (schema) => Type.String({ ...schema, maxLength: 1000 }), // 在变为可空/可选之前扩展 schema
preferences: Type.Object({ theme: Type.String() }) // 覆盖字段,包括其可空性
});
const parsed: {
id: number;
name: string,
bio?: string | undefined;
preferences: {
theme: string;
};
} = Value.Parse(userSelectSchema, ...);对于更高级的用例,你可以使用 createSchemaFactory 函数。
用例:使用扩展的 Typebox 实例
import { pgTable, text, integer } from 'drizzle-orm/pg-core';
import { createSchemaFactory } from 'drizzle-orm/typebox';
import { t } from 'elysia'; // 扩展的 Typebox 实例
const users = pgTable('users', {
id: integer().generatedAlwaysAsIdentity().primaryKey(),
name: text().notNull(),
age: integer().notNull()
});
const { createInsertSchema } = createSchemaFactory({ typeboxInstance: t });
const userInsertSchema = createInsertSchema(users, {
// 现在可以使用扩展实例
name: (schema) => t.Number({ ...schema }, { error: '`name` 必须是字符串' })
});pg.boolean();
mysql.boolean();
sqlite.integer({ mode: 'boolean' });
// Schema
Type.Boolean();pg.date({ mode: 'date' });
pg.timestamp({ mode: 'date' });
mysql.date({ mode: 'date' });
mysql.datetime({ mode: 'date' });
mysql.timestamp({ mode: 'date' });
sqlite.integer({ mode: 'timestamp' });
sqlite.integer({ mode: 'timestamp_ms' });
// Schema
Type.Date();pg.date({ mode: 'string' });
pg.timestamp({ mode: 'string' });
pg.cidr();
pg.inet();
pg.interval();
pg.macaddr();
pg.macaddr8();
pg.numeric();
pg.text();
pg.sparsevec();
pg.time();
mysql.binary();
mysql.date({ mode: 'string' });
mysql.datetime({ mode: 'string' });
mysql.decimal();
mysql.time();
mysql.timestamp({ mode: 'string' });
mysql.varbinary();
sqlite.numeric();
sqlite.text({ mode: 'text' });
// Schema
Type.String();pg.bit({ dimensions: ... });
// Schema
t.RegExp(/^[01]+$/, { maxLength: dimensions });pg.uuid();
// Schema
Type.String({ format: 'uuid' });pg.char({ length: ... });
mysql.char({ length: ... });
// Schema
Type.String({ minLength: length, maxLength: length });pg.varchar({ length: ... });
mysql.varchar({ length: ... });
sqlite.text({ mode: 'text', length: ... });
// Schema
Type.String({ maxLength: length });mysql.tinytext();
// Schema
Type.String({ maxLength: 255 }); // 无符号 8 位整数限制mysql.text();
// Schema
Type.String({ maxLength: 65_535 }); // 无符号 16 位整数限制mysql.mediumtext();
// Schema
Type.String({ maxLength: 16_777_215 }); // 无符号 24 位整数限制mysql.longtext();
// Schema
Type.String({ maxLength: 4_294_967_295 }); // 无符号 32 位整数限制pg.text({ enum: ... });
pg.char({ enum: ... });
pg.varchar({ enum: ... });
mysql.tinytext({ enum: ... });
mysql.mediumtext({ enum: ... });
mysql.text({ enum: ... });
mysql.longtext({ enum: ... });
mysql.char({ enum: ... });
mysql.varchar({ enum: ... });
mysql.mysqlEnum(..., ...);
sqlite.text({ mode: 'text', enum: ... });
// Schema
Type.Enum(enum);mysql.tinyint();
// Schema
Type.Integer({ minimum: -128, maximum: 127 }); // 8 位带符号整数上下限mysql.tinyint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 255 }); // 无符号 8 位整数上下限pg.smallint();
pg.smallserial();
mysql.smallint();
// Schema
Type.Integer({ minimum: -32_768, maximum: 32_767 }); // 16 位带符号整数上下限mysql.smallint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 65_535 }); // 无符号 16 位整数上下限pg.real();
mysql.float();
// Schema
Type.Number().min(-8_388_608).max(8_388_607); // 24 位带符号整数上下限mysql.mediumint();
// Schema
Type.Integer({ minimum: -8_388_608, maximum: 8_388_607 }); // 24 位带符号整数上下限mysql.float({ unsigned: true });
// Schema
Type.Number({ minimum: 0, maximum: 16_777_215 }); // 无符号 24 位整数上下限mysql.mediumint({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 16_777_215 }); // 无符号 24 位整数上下限pg.integer();
pg.serial();
mysql.int();
// Schema
Type.Integer({ minimum: -2_147_483_648, maximum: 2_147_483_647 }); // 32 位带符号整数上下限mysql.int({ unsigned: true });
// Schema
Type.Integer({ minimum: 0, maximum: 4_294_967_295 }); // 无符号 32 位整数上下限pg.doublePrecision();
mysql.double();
mysql.real();
sqlite.real();
// Schema
Type.Number({ minimum: -140_737_488_355_328, maximum: 140_737_488_355_327 }); // 48 位带符号整数上下限mysql.double({ unsigned: true });
// Schema
Type.Numer({ minimum: 0, maximum: 281_474_976_710_655 }); // 无符号 48 位整数上下限pg.bigint({ mode: 'number' });
pg.bigserial({ mode: 'number' });
mysql.bigint({ mode: 'number' });
mysql.bigserial({ mode: 'number' });
sqlite.integer({ mode: 'number' });
// Schema
Type.Integer({ minimum: -9_007_199_254_740_991, maximum: 9_007_199_254_740_991 }); // Javascript 最小和最大安全整数mysql.serial();
Type.Integer({ minimum: 0, maximum: 9_007_199_254_740_991 }); // Javascript 最大安全整数pg.bigint({ mode: 'bigint' });
pg.bigserial({ mode: 'bigint' });
mysql.bigint({ mode: 'bigint' });
sqlite.blob({ mode: 'bigint' });
// Schema
Type.BigInt({ minimum: -9_223_372_036_854_775_808n, maximum: 9_223_372_036_854_775_807n }); // 64 位整数上下限mysql.bigint({ mode: 'bigint', unsigned: true });
// Schema
Type.BigInt({ minimum: 0, maximum: 18_446_744_073_709_551_615n }); // 无符号 64 位整数上下限mysql.year();
// Schema
Type.Integer({ minimum: 1_901, maximum: 2_155 });pg.geometry({ type: 'point', mode: 'tuple' });
pg.point({ mode: 'tuple' });
// Schema
Type.Tuple([Type.Number(), Type.Number()]);pg.geometry({ type: 'point', mode: 'xy' });
pg.point({ mode: 'xy' });
// Schema
Type.Object({ x: Type.Number(), y: Type.Number() });pg.halfvec({ dimensions: ... });
pg.vector({ dimensions: ... });
// Schema
Type.Array(Type.Number(), { minItems: dimensions, maxItems: dimensions });pg.line({ mode: 'abc' });
// Schema
Type.Object({ a: Type.Number(), b: Type.Number(), c: Type.Number() });pg.line({ mode: 'tuple' });
// Schema
Type.Tuple([Type.Number(), Type.Number(), Type.Number()]);pg.json();
pg.jsonb();
mysql.json();
sqlite.blob({ mode: 'json' });
sqlite.text({ mode: '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)]));sqlite.blob({ mode: 'buffer' });
// Schema
t.Union([t.Union([t.String(), t.Number(), t.Boolean(), t.Null()]), t.Array(t.Any()), t.Record(t.String(), t.Any())]);pg.dataType().array(...);
// Schema
Type.Array(baseDataTypeSchema, { minItems: size, maxItems: size });