Jacob Paris
← Back to all content

Add Drizzle ORM to Remix Vite with Cloudflare D1

Drizzle is a typescript ORM for interacting with relational databases. It gives you type-safe reads and writes, plus automatic migration handling.

Drizzle works with many different databases, and can be configured to work with Cloudflare's distributed SQLite database, D1.

In this guide, we'll implement this example of Drizzle and D1 to a Remix Vite app running on Cloudflare.

Start with the remix cloudflare vite template:

bash
npx create-remix@latest --template remix-run/remix/templates/cloudflare

In addition to Drizzle, we will also need to install Drizzle Kit, which is a CLI migration tool, and any SQLite driver, such as better-sqlite3.

bash
npm i drizzle-orm
npm i --save-dev drizzle-kit

You may also want to check out my guide on deploying to Cloudflare Pages if you have any issues along the way.

Write your schema

The schema is where you define your tables and columns. Drizzle uses typescript to infer types from your schema, so you get type-safe reads and writes.

Create a new file app/drizzle/schema.server.ts and write a basic schema to get started.

ts
import {
sqliteTable,
text,
integer,
} from "drizzle-orm/sqlite-core"
export const resources = sqliteTable("resources", {
id: integer("id").primaryKey(),
title: text("title").notNull(),
href: text("href").notNull(),
})

To power typescript's type inference and autocomplete, we need to generate types based on our schema. That's what the drizzle-kit CLI tool was for.

Go to your package.json, add a db:migrations script to run the migrations:

json
{
"scripts": {
"drizzle:update": "drizzle-kit generate:sqlite --out ./app/drizzle/migrations --schema ./app/drizzle/schema.server.ts"
}
}

Run npm run drizzle:update and you should see a migrations directory in your app/drizzle folder. This will be committed to your repo.

Set up a D1 database

If you haven't already, you'll need to set up a D1 databases.

Create a new database named "db""

sh
npx wrangler d1 create db

Paste the command output in wrangler.toml:

toml
[[d1_databases]]
binding = "DB" # i.e. available in your Worker on env.DB
database_name = "db"
database_id = "ccccccvktlfkrjvvrngbckuvtrrkugtu"
# Also add this field
migrations_dir="./app/drizzle/migrations"

Adding the migration field tells Cloudflare to use the Drizzle migrations when setting up the database.

We need to tell typescript that this database exists, so run the following command to generate an interface in worker.configuration.d.ts

sh
npx wrangler types

which will look something like this:

ts
interface Env {
DB: D1Database
}

The remix-cloudflare template currently comes with a load-context.ts file that has an empty Env interface. You'll need to delete this to get the generated one to work.

diff
- interface Env {}

You may need to restart the Typescript server to pick up the changes. In VS Code, open the command pallette with CMD Shift P and run Typescript: Restart TS Server.

Read and write documents

Drizzle should be fully set up now. Let's try it out!

In one of your routes

  • add an action that inserts a new item into the database
  • add a loader that reads all the items from the database and returns them as JSON
  • add a form that submits to the action

You can access the handle to the database from context.cloudflare.env.DB in your action and loader functions.

tsx
// http://localhost:5173/
import {
json,
type MetaFunction,
} from "@remix-run/cloudflare"
import type {
LoaderFunctionArgs,
ActionFunctionArgs,
} from "@remix-run/node"
import { Form, useLoaderData } from "@remix-run/react"
import { drizzle } from "drizzle-orm/d1"
import { resources } from "#app/drizzle/schema.server"
export async function action({
request,
context,
}: ActionFunctionArgs) {
const formData = await request.formData()
const title = formData.get("title") as string
const href = formData.get("href") as string
const db = drizzle(context.cloudflare.env.DB)
await db
.insert(resources)
.values({ title, href })
.execute()
return json(
{ message: "Resource added" },
{ status: 201 },
)
}
export async function loader({
context,
}: LoaderFunctionArgs) {
const db = drizzle(context.cloudflare.env.DB)
const resourceList = await db
.select({
id: resources.id,
title: resources.title,
href: resources.href,
})
.from(resources)
.orderBy(resources.id)
return json({
resourceList,
})
}
export default function Index() {
const { resourceList } = useLoaderData<typeof loader>()
return (
<div>
<h1>
Welcome to Remix (with Drizzle, Vite and Cloudflare
D1)
</h1>
<ul>
{resourceList.map((resource) => (
<li key={resource.id}>
<a
target="_blank"
href={resource.href}
rel="noreferrer"
>
{resource.title}
</a>
</li>
))}
</ul>
<Form method="POST">
<div>
<label>
Title:{" "}
<input type="text" name="title" required />
</label>
</div>
<div>
<label>
URL: <input type="url" name="href" required />
</label>
</div>
<button type="submit">Add Resource</button>
</Form>
</div>
)
}
Professional headshot
Moulton
Moulton

Hey there! I'm a developer, designer, and digital nomad building cool things with Remix, and I'm also writing Moulton, the Remix Community Newsletter

About once per month, I send an email with:

  • New guides and tutorials
  • Upcoming talks, meetups, and events
  • Cool new libraries and packages
  • What's new in the latest versions of Remix

Stay up to date with everything in the Remix community by entering your email below.

Unsubscribe at any time.