Drizzle ORM - Lightweight ORM Made with TypeScript

2025.12.01

What is Drizzle ORM

Drizzle ORM is a lightweight ORM designed TypeScript-first. You can write queries intuitively with SQL-like syntax while providing complete type safety.

Features

✓ TypeScript First
✓ SQL-like Syntax
✓ Zero Dependencies (Lightweight)
✓ Edge Compatible
✓ Serverless Optimized

Schema Definition

Table Definition

// schema.ts
import { pgTable, serial, text, timestamp, integer, boolean } from 'drizzle-orm/pg-core';

export const users = pgTable('users', {
  id: serial('id').primaryKey(),
  name: text('name').notNull(),
  email: text('email').notNull().unique(),
  createdAt: timestamp('created_at').defaultNow()
});

export const posts = pgTable('posts', {
  id: serial('id').primaryKey(),
  title: text('title').notNull(),
  content: text('content'),
  authorId: integer('author_id').references(() => users.id),
  published: boolean('published').default(false),
  createdAt: timestamp('created_at').defaultNow()
});

Relation Definition

import { relations } from 'drizzle-orm';

export const usersRelations = relations(users, ({ many }) => ({
  posts: many(posts)
}));

export const postsRelations = relations(posts, ({ one }) => ({
  author: one(users, {
    fields: [posts.authorId],
    references: [users.id]
  })
}));

Query Builder

Basic CRUD

import { drizzle } from 'drizzle-orm/node-postgres';
import { eq, and, or, gt, like } from 'drizzle-orm';
import * as schema from './schema';

const db = drizzle(pool, { schema });

// INSERT
const newUser = await db.insert(users).values({
  name: 'Alice',
  email: 'alice@example.com'
}).returning();

// SELECT
const allUsers = await db.select().from(users);

// WHERE
const activeUsers = await db.select()
  .from(users)
  .where(eq(users.isActive, true));

// UPDATE
await db.update(users)
  .set({ name: 'Bob' })
  .where(eq(users.id, 1));

// DELETE
await db.delete(users)
  .where(eq(users.id, 1));

Complex Queries

// Multiple conditions
const results = await db.select()
  .from(users)
  .where(
    and(
      gt(users.age, 18),
      or(
        like(users.email, '%@company.com'),
        eq(users.role, 'admin')
      )
    )
  );

// ORDER BY, LIMIT, OFFSET
const pagedUsers = await db.select()
  .from(users)
  .orderBy(desc(users.createdAt))
  .limit(10)
  .offset(20);

// JOIN
const postsWithAuthors = await db.select({
  postTitle: posts.title,
  authorName: users.name
})
  .from(posts)
  .leftJoin(users, eq(posts.authorId, users.id));

Relational Queries

// Get nested data
const usersWithPosts = await db.query.users.findMany({
  with: {
    posts: {
      where: eq(posts.published, true),
      limit: 5
    }
  }
});

// Result
// [
//   {
//     id: 1,
//     name: 'Alice',
//     posts: [
//       { id: 1, title: 'First Post', ... },
//       { id: 2, title: 'Second Post', ... }
//     ]
//   }
// ]

Transactions

await db.transaction(async (tx) => {
  const user = await tx.insert(users)
    .values({ name: 'Alice', email: 'alice@example.com' })
    .returning();

  await tx.insert(posts).values({
    title: 'My First Post',
    authorId: user[0].id
  });
});

Migrations

Drizzle Kit

# Generate migration
npx drizzle-kit generate

# Apply migration
npx drizzle-kit migrate

# Push schema to DB (for development)
npx drizzle-kit push

# Drizzle Studio (GUI)
npx drizzle-kit studio

Configuration File

// drizzle.config.ts
import type { Config } from 'drizzle-kit';

export default {
  schema: './src/schema.ts',
  out: './drizzle',
  dialect: 'postgresql',
  dbCredentials: {
    url: process.env.DATABASE_URL!
  }
} satisfies Config;

Database Support

// PostgreSQL
import { drizzle } from 'drizzle-orm/node-postgres';

// MySQL
import { drizzle } from 'drizzle-orm/mysql2';

// SQLite
import { drizzle } from 'drizzle-orm/better-sqlite3';

// Planetscale
import { drizzle } from 'drizzle-orm/planetscale-serverless';

// Neon
import { drizzle } from 'drizzle-orm/neon-http';

// Turso
import { drizzle } from 'drizzle-orm/libsql';

Comparison with Prisma

AspectDrizzlePrisma
SyntaxSQL-likeProprietary DSL
Bundle sizeSmallLarge
MigrationsCode generationPrisma Migrate
Edge supportNativePrisma Accelerate
Learning curveLow if you know SQLNeed to learn Prisma syntax

Summary

Drizzle ORM is a lightweight ORM that balances SQL-like syntax with complete type safety. It’s optimized for edge environments and serverless, making it suitable for modern TypeScript projects.

← Back to list