Database
Welcome to the Database Setup guide for Zap.ts! This page helps you set up your database easily. We’ll talk about the DATABASE_URL
in your .env.local
file, the choice between Drizzle and Prisma, and how to use them in Zap.ts. We’ll also share a core schema that works with any ORM (Object-Relational Mapping tool).
WARNING
You can use a different ORM, but most other Zap.ts plugins won't be compatible.
DATABASE_URL variable
To connect Zap.ts to your database, you need to set the DATABASE_URL
in your .env.local
file. This is a string that tells Zap.ts where your database lives and how to log in.
How to Set It Up
- Open your
.env.local
file in the root of your project. - Find the line that says:
DATABASE_URL=your_database_url_here
- Replace
your_database_url_here
with your actual database URL.
Example
If you’re using Neon (our default database), your DATABASE_URL
might look like this:
DATABASE_URL=postgres://your_username:your_password@your-neon-hostname:5432/your_database_name
your_username
: Your Neon username.your_password
: Your Neon password.your-neon-hostname
: The hostname from Neon (e.g.,ep-silent-forest-123456.us-east-2.aws.neon.tech
).your_database_name
: The name of your database (e.g.,mydb
).
If you’re using another database like Supabase or a local PostgreSQL, the format is similar but the hostname and credentials will be different.
TIP
Make sure your database is running and you can connect to it. You can test it with a tool like psql
or a database GUI like pgAdmin.
Choosing Between Drizzle and Prisma
Zap.ts lets you pick an ORM to talk to your database. An ORM helps you write database code in JavaScript or TypeScript instead of raw SQL. You must choose one of these ORMs:
- Drizzle: Available and recommended. It’s lightweight and easy to use.
- Prisma: Not available yet, but we’re working on it!
When you run bunx create-zap-app@latest
, the script will ask you to pick an ORM. Let’s look at each one.
Drizzle
Drizzle is the default ORM in Zap.ts. It’s a modern, lightweight ORM that works well with PostgreSQL (and other databases). We recommend reading the Drizzle documentation first to understand it better. But don’t worry—we’ll summarize the key ideas here to get you started!
Key Concepts to Understand
Codebase-First vs. Database-First
Drizzle supports two ways to work with your database:
Codebase-First: You write your database schema (structure) in TypeScript files, and Drizzle creates the database tables for you. This is great for new projects because you control everything in your code.
- Example: You write a
user
table insrc/db/schema/auth.ts
, then runnpm run db:push
to create the table in your database.
- Example: You write a
Database-First: You already have a database with tables, and Drizzle generates TypeScript code to match it. This is useful if you’re working with an existing database.
- Example: You have a database with a
user
table, and you runnpm run db:pull
to create a schema file insrc/db/schema/
.
- Example: You have a database with a
In Zap.ts, we use the codebase-first approach by default because it’s easier to manage for new projects.
How to Write a Schema
A schema in Drizzle is a TypeScript file where you define your database tables. You use Drizzle’s functions like pgTable
to create tables and columns.
Here’s a simple example of a schema for a user
table:
// src/db/schema/auth.ts
import { pgTable, text, timestamp } from "drizzle-orm/pg-core";
export const user = pgTable("user", {
id: text("id").primaryKey(),
name: text("name").notNull(),
email: text("email").notNull().unique(),
createdAt: timestamp("created_at").notNull(),
});
pgTable("user", {...})
: Creates a table nameduser
.id: text("id").primaryKey()
: A column namedid
that’s a text string and the primary key.name: text("name").notNull()
: A column namedname
that’s a text string and cannot be empty.email: text("email").notNull().unique()
: A column namedemail
that’s a text string, cannot be empty, and must be unique.createdAt: timestamp("created_at").notNull()
: A column namedcreated_at
that’s a timestamp and cannot be empty.
After writing your schema, run npm run db:push
to create the tables in your database.
Drizzle Structure in Zap.ts
Drizzle in Zap.ts is set up in two main places: drizzle.config.ts
and the src/db/
folder.
drizzle.config.ts
The drizzle.config.ts
file in the root of your project tells Drizzle how to connect to your database and where your schema files are. Here’s what it looks like:
// drizzle.config.ts
import { defineConfig } from "drizzle-kit";
export default defineConfig({
schema: "./src/db/schema", // All schema files are in src/db/schema/
out: "./drizzle", // Where migration files go (you don’t need to change this)
dialect: "postgresql", // We use PostgreSQL by default
dbCredentials: {
url: process.env.DATABASE_URL!, // Uses the DATABASE_URL from .env.local
},
});
schema: "./src/db/schema"
: Points to the folder where your schema files live (src/db/schema/
).dialect: "postgresql"
: Tells Drizzle we’re using PostgreSQL. Check the Drizzle docs for other options like MySQL or SQLite.dbCredentials.url
: Uses theDATABASE_URL
from your.env.local
to connect to the database.
src/db Folder
The src/db/
folder has the code to connect to your database and work with it. Here’s what’s inside:
src/db/index.ts
: This file sets up the Drizzle client and connects to your database.typescript// src/db/index.ts import { neon } from "@neondatabase/serverless"; import { drizzle } from "drizzle-orm/neon-http"; import * as schema from "./schema"; const sql = neon(process.env.DATABASE_URL!); export const db = drizzle({ client: sql, schema });
neon
: We use Neon as the default database provider. It’s a serverless PostgreSQL database.process.env.DATABASE_URL
: Gets the database URL from your.env.local
.drizzle({ client: sql, schema })
: Creates a Drizzle client with the Neon connection and your schema for type safety.schema
: Imports all schema files fromsrc/db/schema/
so Drizzle knows your tables.
If you want to use a different database (like Supabase or a local PostgreSQL), check the Drizzle docs to change the client. For example, to use a local PostgreSQL, you’d use
pg
instead ofneon
.src/db/schema/
: This folder has all your schema files, likeauth.ts
(for user authentication) andnotifications.ts
(for PWA notifications). We’ll talk more about the schema later in the Core Schema section.
Bonus: Using Supabase with RLS
Supabase is a great alternative to Neon for your database. It’s a PostgreSQL database with extra features like Row-Level Security (RLS), which lets you control who can see or change data based on user roles.
Here’s how to set up Supabase with RLS in Zap.ts:
Get Your Supabase URL:
- Sign up for Supabase and create a project.
- Go to your project settings in the Supabase dashboard and find the database connection string (it looks like
postgres://...
). - Copy the connection string and add it to your
.env.local
as theDATABASE_URL
.
Enable RLS in Supabase:
- In the Supabase dashboard, go to the “Authentication” section and set up your user roles.
- Go to the “Database” section, select a table (e.g.,
user
), and enable RLS. - Add a policy, like “Allow users to read their own data”:sqlThis policy means users can only see their own user data.
CREATE POLICY user_read_own_data ON user FOR SELECT USING (auth.uid() = id);
Update Drizzle to Work with Supabase:
- You need to switch to Supabase’s client directly. Check the Drizzle docs for Supabase for more details.
Test Your Setup:
- Test your app to make sure RLS is working—users should only see data they’re allowed to see.
TIP
Drizzle allows for RLS and policy support directly within the codebase-first approach, eliminating the need for step 2.
Prisma (Not Available Yet)
Prisma is another ORM you can use with Zap.ts, but it’s not available yet. We’re working on adding support for it! When it’s ready, you’ll be able to choose Prisma when you run bunx create-zap-app@latest
. Prisma is great if you like a more visual way to manage your database schema, and it has a nice query builder.
For now, we recommend using Drizzle. Check back later for updates on Prisma support!
Core Schema (ORM-Agnostic)
Zap.ts comes with a core schema that works with any ORM. This schema is used for user authentication and is generated by the better-auth
package, which is a core part of Zap.ts. The schema includes tables for users, sessions, accounts, and more.
Here’s what the core schema looks like in Drizzle (but the idea is the same for any ORM):
User Table
Stores information about users, like their name, email, and role.
// src/db/schema/auth.ts
export const user = pgTable("user", {
id: text("id").primaryKey(),
name: text("name").notNull(),
email: text("email").notNull().unique(),
emailVerified: boolean("email_verified").notNull(),
image: text("image"),
createdAt: timestamp("created_at").notNull(),
updatedAt: timestamp("updated_at").notNull(),
twoFactorEnabled: boolean("two_factor_enabled"),
username: text("username").unique(),
displayUsername: text("display_username"),
isAnonymous: boolean("is_anonymous"),
role: text("role"),
banned: boolean("banned"),
banReason: text("ban_reason"),
banExpires: timestamp("ban_expires"),
});
id
: A unique ID for each user.email
: The user’s email, which must be unique.role
: The user’s role (e.g., "admin" or "user").banned
: Whether the user is banned.
Session Table
Keeps track of user sessions (when they log in).
export const session = pgTable("session", {
id: text("id").primaryKey(),
expiresAt: timestamp("expires_at").notNull(),
token: text("token").notNull().unique(),
createdAt: timestamp("created_at").notNull(),
updatedAt: timestamp("updated_at").notNull(),
ipAddress: text("ip_address"),
userAgent: text("user_agent"),
userId: text("user_id")
.notNull()
.references(() => user.id, { onDelete: "cascade" }),
impersonatedBy: text("impersonated_by"),
activeOrganizationId: text("active_organization_id"),
});
token
: A unique token for the session.userId
: Links to theuser
table (if the user is deleted, their sessions are deleted too).expiresAt
: When the session expires.
Account Table
Stores info about external accounts (e.g., if a user logs in with Google).
export const account = pgTable("account", {
id: text("id").primaryKey(),
accountId: text("account_id").notNull(),
providerId: text("provider_id").notNull(),
userId: text("user_id")
.notNull()
.references(() => user.id, { onDelete: "cascade" }),
accessToken: text("access_token"),
refreshToken: text("refresh_token"),
idToken: text("id_token"),
accessTokenExpiresAt: timestamp("access_token_expires_at"),
refreshTokenExpiresAt: timestamp("refresh_token_expires_at"),
scope: text("scope"),
password: text("password"),
createdAt: timestamp("created_at").notNull(),
updatedAt: timestamp("updated_at").notNull(),
});
providerId
: The provider (e.g., "google").accessToken
: The token for accessing the provider’s API.
Other Tables
The schema also includes tables for:
- Verification: For email verification codes.
- Two-Factor: For two-factor authentication secrets.
- Passkey: For passkey authentication.
- Organization: For managing organizations (if you use teams).
- Member: For organization members.
- Invitation: For inviting users to organizations.
You can find all these tables in src/db/schema/auth.ts
or by reading Better Auth documentation.
PWA Notifications Table
If you enable the pwa
plugin, there’s an extra table for push notifications:
// src/db/schema/notifications.ts
export const pushNotifications = pgTable("push_notifications", {
id: text("uuid").primaryKey().default("gen_random_uuid()"),
subscription: text("jsonb").$type<webpush.PushSubscription>().notNull(),
userId: text("uuid").references(() => user.id, { onDelete: "cascade" }),
});
subscription
: Stores the push notification subscription data.userId
: Links to theuser
table.
Next Steps
Now that your database is set up, you can:
- Run
npm run db:push
to create your tables in the database. - Start building your app with Zap.ts!
- If you need help, check the Drizzle docs or ask on X.
Happy coding! ⚡