src/content/docs/rqb.mdx
import Tab from '@mdx/Tab.astro'; import Tabs from '@mdx/Tabs.astro'; import Callout from '@mdx/Callout.astro'; import CodeTabs from '@mdx/CodeTabs.astro'; import CodeTab from '@mdx/CodeTab.astro'; import Section from '@mdx/Section.astro'; import IsSupportedChipGroup from '@mdx/IsSupportedChipGroup.astro';
<IsSupportedChipGroup chips={{ 'PostgreSQL': true, 'SQLite': true, 'MySQL': true, 'SingleStore': true }} />
Drizzle ORM is designed to be a thin typed layer on top of SQL. We truly believe we've designed the best way to operate an SQL database from TypeScript and it's time to make it better.
Relational queries are meant to provide you with a great developer experience for querying nested relational data from an SQL database, avoiding multiple joins and complex data mappings.
It is an extension to the existing schema definition and query builder. You can opt-in to use it based on your needs. We've made sure you have both the best-in-class developer experience and performance.
<CodeTabs items={["index.ts", "schema.ts"]}> <CodeTab> ```typescript copy /schema/3 import * as schema from './schema'; import { drizzle } from 'drizzle-orm/...';
const db = drizzle({ schema });
const result = await db._query.users.findMany({
with: {
posts: true
},
});
```
```ts
[{
id: 10,
name: "Dan",
posts: [
{
id: 1,
content: "SQL is awesome",
authorId: 10,
},
{
id: 2,
content: "But check relational queries",
authorId: 10,
}
]
}]
```
</CodeTab>
```typescript copy
import { integer, serial, text, pgTable } from 'drizzle-orm/pg-core';
import { relations } from 'drizzle-orm';
export const users = pgTable('users', {
id: serial('id').primaryKey(),
name: text('name').notNull(),
});
export const usersRelations = relations(users, ({ many }) => ({
posts: many(posts),
}));
export const posts = pgTable('posts', {
id: serial('id').primaryKey(),
content: text('content').notNull(),
authorId: integer('author_id').notNull(),
});
export const postsRelations = relations(posts, ({ one }) => ({
author: one(users, { fields: [posts.authorId], references: [users.id] }),
}));
```
⚠️ If you have SQL schema declared in multiple files you can do it like that <CodeTabs items={["index.ts", "schema1.ts", "schema2.ts"]}> ```typescript copy /schema/3 import * as schema1 from './schema1'; import * as schema2 from './schema2'; import { drizzle } from 'drizzle-orm/...';
const db = drizzle({ schema: { ...schema1, ...schema2 } });
const result = await db._query.users.findMany({
with: {
posts: true
},
});
```
```ts
// schema declaration in the first file
```
```ts
// schema declaration in the second file
```
Drizzle relational queries always generate exactly one SQL statement to run on the database and it has certain caveats.
To have best in class support for every database out there we've introduced modes.
Drizzle relational queries use lateral joins of subqueries under the hood and for now PlanetScale does not support them.
When using mysql2 driver with regular MySQL database — you should specify mode: "default"
When using mysql2 driver with PlanetScale — you need to specify mode: "planetscale"
import * as schema from './schema';
import { drizzle } from "drizzle-orm/mysql2";
import mysql from "mysql2/promise";
const connection = await mysql.createConnection({
uri: process.env.PLANETSCALE_DATABASE_URL,
});
const db = drizzle({ client: connection, schema, mode: 'planetscale' });
Relational queries are an extension to Drizzle's original query builder.
You need to provide all tables and relations from your schema file/files upon drizzle()
initialization and then just use the db._query API.
<Callout type="info" emoji="ℹ️">
drizzle import path depends on the database driver you're using.
</Callout>
<CodeTabs items={["index.ts", "schema.ts"]}>
<CodeTab>
import * as schema from './schema';
import { drizzle } from 'drizzle-orm/...';
const db = drizzle({ schema });
await db._query.users.findMany(...);
// if you have schema in multiple files
import * as schema1 from './schema1';
import * as schema2 from './schema2';
import { drizzle } from 'drizzle-orm/...';
const db = drizzle({ schema: { ...schema1, ...schema2 } });
await db._query.users.findMany(...);
import { relations } from 'drizzle-orm';
export const users = pgTable('users', {
id: serial('id').primaryKey(),
name: text('name').notNull(),
verified: boolean('verified').notNull(),
invitedBy: integer('invited_by').references((): AnyPgColumn => users.id),
});
export const usersRelations = relations(users, ({ one, many }) => ({
invitee: one(users, { fields: [users.invitedBy], references: [users.id] }),
usersToGroups: many(usersToGroups),
posts: many(posts),
}));
export const groups = pgTable('groups', {
id: serial('id').primaryKey(),
name: text('name').notNull(),
description: text('description'),
});
export const groupsRelations = relations(groups, ({ many }) => ({
usersToGroups: many(usersToGroups),
}));
export const usersToGroups = pgTable('users_to_groups', {
id: serial('id').primaryKey(),
userId: integer('user_id').notNull().references(() => users.id),
groupId: integer('group_id').notNull().references(() => groups.id),
}, (t) => [
primaryKey({ columns: [t.userId, t.groupId] })
]);
export const usersToGroupsRelations = relations(usersToGroups, ({ one }) => ({
group: one(groups, { fields: [usersToGroups.groupId], references: [groups.id] }),
user: one(users, { fields: [usersToGroups.userId], references: [users.id] }),
}));
export const posts = pgTable('posts', {
id: serial('id').primaryKey(),
content: text('content').notNull(),
authorId: integer('author_id').references(() => users.id),
createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),
});
export const postsRelations = relations(posts, ({ one, many }) => ({
author: one(users, { fields: [posts.authorId], references: [users.id] }),
comments: many(comments),
}));
export const comments = pgTable('comments', {
id: serial('id').primaryKey(),
content: text('content').notNull(),
creator: integer('creator').references(() => users.id),
postId: integer('post_id').references(() => posts.id),
createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),
});
export const commentsRelations = relations(comments, ({ one, many }) => ({
post: one(posts, { fields: [comments.postId], references: [posts.id] }),
author: one(users, { fields: [comments.creator], references: [users.id] }),
likes: many(commentLikes),
}));
export const commentLikes = pgTable('comment_likes', {
id: serial('id').primaryKey(),
creator: integer('creator').references(() => users.id),
commentId: integer('comment_id').references(() => comments.id),
createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),
});
export const commentLikesRelations = relations(commentLikes, ({ one }) => ({
comment: one(comments, { fields: [commentLikes.commentId], references: [comments.id] }),
author: one(users, { fields: [commentLikes.creator], references: [users.id] }),
}));
</CodeTabs>
Drizzle provides `.findMany()` and `.findFirst()` APIs.
### Find many
<Section>
```typescript copy
const users = await db._query.users.findMany();
// result type
const result: {
id: number;
name: string;
verified: boolean;
invitedBy: number | null;
}[];
With operator lets you combine data from multiple related tables and properly aggregate results.
Getting all posts with comments:
const posts = await db._query.posts.findMany({
with: {
comments: true,
},
});
Getting first post with comments:
const post = await db._query.posts.findFirst({
with: {
comments: true,
},
});
You can chain nested with statements as much as necessary.
For any nested with queries Drizzle will infer types using Core Type API.
Get all users with posts. Each post should contain a list of comments:
const users = await db._query.users.findMany({
with: {
posts: {
with: {
comments: true,
},
},
},
});
columns parameter lets you include or omit columns you want to get from the database.
Keep in mind that a single SQL statement is outputted by Drizzle. </Callout>
Get all posts with just id, content and include comments:
const posts = await db._query.posts.findMany({
columns: {
id: true,
content: true,
},
with: {
comments: true,
}
});
Get all posts without content:
const posts = await db._query.posts.findMany({
columns: {
content: false,
},
});
If you include the name field and exclude the id field, id exclusion will be redundant,
all fields apart from name would be excluded anyways.
Exclude and Include fields in the same query:
<Section> ```typescript copy const users = await db._query.users.findMany({ columns: { name: true, id: false //ignored }, }); ``` ```ts // result type const users: { name: string; }; ``` </Section>Only include columns from nested relations:
<Section> ```typescript copy const res = await db._query.users.findMany({ columns: {}, with: { posts: true } }); ``` ```ts // result type const res: { posts: { id: number, text: string } }[]; ``` </Section>Just like with partial select, you can include or exclude columns of nested relations:
const posts = await db._query.posts.findMany({
columns: {
id: true,
content: true,
},
with: {
comments: {
columns: {
authorId: false
}
}
}
});
Just like in our SQL-like query builder,
relational queries API lets you define filters and conditions with the list of our operators.
You can either import them from drizzle-orm or use from the callback syntax:
const users = await db._query.users.findMany({ where: eq(users.id, 1) })
```ts copy
const users = await db._query.users.findMany({
where: (users, { eq }) => eq(users.id, 1),
})
Find post with id=1 and comments that were created before particular date:
await db._query.posts.findMany({
where: (posts, { eq }) => (eq(posts.id, 1)),
with: {
comments: {
where: (comments, { lt }) => lt(comments.createdAt, new Date()),
},
},
});
Drizzle ORM provides limit & offset API for queries and for the nested entities.
Find 5 posts:
await db._query.posts.findMany({
limit: 5,
});
Find posts and get 3 comments at most:
await db._query.posts.findMany({
with: {
comments: {
limit: 3,
},
},
});
Find posts with comments from the 5th to the 10th post:
await db._query.posts.findMany({
limit: 5,
offset: 5,
with: {
comments: true,
},
});
Drizzle provides API for ordering in the relational query builder.
You can use same ordering core API or use
order by operator from the callback with no imports.
await db._query.posts.findMany({ orderBy: [asc(posts.id)], });
```typescript copy
await db._query.posts.findMany({
orderBy: (posts, { asc }) => [asc(posts.id)],
});
Order by asc + desc:
await db._query.posts.findMany({
orderBy: (posts, { asc }) => [asc(posts.id)],
with: {
comments: {
orderBy: (comments, { desc }) => [desc(comments.id)],
},
},
});
Relational query API lets you add custom additional fields.
It's useful when you need to retrieve data and apply additional functions to it.
<Callout type="warning" emoji="⚠️">
As of now aggregations are not supported in extras, please use core queries for that.
</Callout>
await db._query.users.findMany({
extras: {
loweredName: sqllower(${users.name}).as('lowered_name'),
},
})
```typescript copy {3}
await db._query.users.findMany({
extras: {
loweredName: (users, { sql }) => sql`lower(${users.name})`.as('lowered_name'),
},
})
lowerName as a key will be included to all fields in returned object.
To retrieve all users with groups, but with the fullName field included (which is a concatenation of firstName and lastName), you can use the following query with the Drizzle relational query builder.
<Section> ```typescript copy const res = await db._query.users.findMany({ extras: { fullName: sql<string>`concat(${users.name}, " ", ${users.name})`.as('full_name'), }, with: { usersToGroups: { with: { group: true, }, }, }, }); ``` ```ts // result type const res: { id: number; name: string; verified: boolean; invitedBy: number | null; fullName: string; usersToGroups: { group: { id: number; name: string; description: string | null; }; }[]; }[];</Section>
To retrieve all posts with comments and add an additional field to calculate the size of the post content and the size of each comment content:
<Section>
```typescript copy
const res = await db._query.posts.findMany({
extras: (table, { sql }) => ({
contentLength: (sql<number>`length(${table.content})`).as('content_length'),
}),
with: {
comments: {
extras: {
commentSize: sql<number>`length(${comments.content})`.as('comment_size'),
},
},
},
});
// result type
const res: {
id: number;
createdAt: Date;
content: string;
authorId: number | null;
contentLength: number;
comments: {
id: number;
createdAt: Date;
content: string;
creator: number | null;
postId: number | null;
commentSize: number;
}[];
};
Prepared statements are designed to massively improve query performance — see here.
In this section, you can learn how to define placeholders and execute prepared statements using the Drizzle relational query builder.
where<Tabs items={['PostgreSQL', 'MySQL', 'SQLite']}> <Tab>
<Section> ```ts copy const prepared = db._query.users.findMany({ where: ((users, { eq }) => eq(users.id, placeholder('id'))), with: { posts: { where: ((users, { eq }) => eq(users.id, placeholder('pid'))), }, }, }).prepare('query_name');const usersWithPosts = await prepared.execute({ id: 1 });
</Section>
</Tab>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
where: ((users, { eq }) => eq(users.id, placeholder('id'))),
with: {
posts: {
where: ((users, { eq }) => eq(users.id, placeholder('pid'))),
},
},
}).prepare();
const usersWithPosts = await prepared.execute({ id: 1 });
const usersWithPosts = await prepared.execute({ id: 1 });
</Section>
</Tab>
</Tabs>
##### **Placeholder in `limit`**
<Tabs items={['PostgreSQL', 'MySQL', 'SQLite']}>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
with: {
posts: {
limit: placeholder('limit'),
},
},
}).prepare('query_name');
const usersWithPosts = await prepared.execute({ limit: 1 });
const usersWithPosts = await prepared.execute({ limit: 1 });
</Section>
</Tab>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
with: {
posts: {
limit: placeholder('limit'),
},
},
}).prepare();
const usersWithPosts = await prepared.execute({ limit: 1 });
offset<Tabs items={['PostgreSQL', 'MySQL', 'SQLite']}> <Tab>
<Section> ```ts copy const prepared = db._query.users.findMany({ offset: placeholder('offset'), with: { posts: true, }, }).prepare('query_name');const usersWithPosts = await prepared.execute({ offset: 1 });
</Section>
</Tab>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
offset: placeholder('offset'),
with: {
posts: true,
},
}).prepare();
const usersWithPosts = await prepared.execute({ offset: 1 });
const usersWithPosts = await prepared.execute({ offset: 1 });
</Section>
</Tab>
</Tabs>
##### **Multiple placeholders**
<Tabs items={['PostgreSQL', 'MySQL', 'SQLite']}>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
limit: placeholder('uLimit'),
offset: placeholder('uOffset'),
where: ((users, { eq, or }) => or(eq(users.id, placeholder('id')), eq(users.id, 3))),
with: {
posts: {
where: ((users, { eq }) => eq(users.id, placeholder('pid'))),
limit: placeholder('pLimit'),
},
},
}).prepare('query_name');
const usersWithPosts = await prepared.execute({ pLimit: 1, uLimit: 3, uOffset: 1, id: 2, pid: 6 });
const usersWithPosts = await prepared.execute({ pLimit: 1, uLimit: 3, uOffset: 1, id: 2, pid: 6 });
</Section>
</Tab>
<Tab>
<Section>
```ts copy
const prepared = db._query.users.findMany({
limit: placeholder('uLimit'),
offset: placeholder('uOffset'),
where: ((users, { eq, or }) => or(eq(users.id, placeholder('id')), eq(users.id, 3))),
with: {
posts: {
where: ((users, { eq }) => eq(users.id, placeholder('pid'))),
limit: placeholder('pLimit'),
},
},
}).prepare();
const usersWithPosts = await prepared.execute({ pLimit: 1, uLimit: 3, uOffset: 1, id: 2, pid: 6 });