Database
Configure your Supabase Database
This guide covers working with Supabase PostgreSQL and Drizzle ORM in Titan projects.
Core Concepts
- PostgreSQL: Supabase’s underlying database engine
- Drizzle ORM: Type-safe database client for TypeScript
- Row Level Security (RLS): PostgreSQL’s feature for row access control
- Migrations: Version-controlled database schema changes
Development vs Production
Environment | Characteristics | Considerations |
---|---|---|
Dev | Safe for experimentation, can be reset | Not for real user data |
Production | Contains real user data | Test all changes in dev first |
Initial Setup
After you’ve spun up a fresh Titan project, you would’ve had a thorough conversation with agent mode in Cursor to come up with all the relevant documentation and all the tables you would need for your MVP.
You can now use this guide by leveraging the database schema that you’ve created to start creating your initial tables in your database, and then subsequently making changes to the database schema as you see fit over time.
- Run the Database Initialization Script
- This will create a new database in Supabase and apply the initial schema to it - i.e. create all the tables defined in
db/schema/index.ts
. - You can optionally seed the database with test data (It will ask you if you want to do this).
- Verify the Setup
- This will open the database studio where you can explore your database to check that the tables have been created correctly.
Data Access Patterns
Titan uses three primary patterns for database access:
Pattern | Use Cases | Benefits |
---|---|---|
Direct Drizzle in Server Actions | Server components, form submissions | Type safety, best performance |
React Query + Server Actions | Client UI with loading states | Caching, loading states |
Supabase Client | Client-side auth, storage | RLS security, simple API |
1. Direct Drizzle (Server-Side)
2. React Query with Server Actions
3. Supabase Client (Client-Side)
Which Pattern to Use?
- Server operations? → Use Direct Drizzle
- Client UI with loading states? → Use React Query + Server Actions
- Client-side auth or RLS needs? → Use Supabase Client
Schema Design and Queries
Schema Definition
Define your database schema with TypeScript:
Best Practices
- Organize tables by domain in separate files
- Use UUID or serial for IDs
- Add created/updated timestamps to all tables
- Create indexes for frequently queried columns
- Define relationships with references
- Use enums for fixed values
- Add appropriate constraints
Common Query Patterns
Row Level Security (RLS)
RLS restricts which rows a user can access based on their identity.
Common RLS Patterns
Authorization in Server Actions
Always add authorization checks in server actions:
Migrations
Titan uses Drizzle for database migrations:
Development Workflow
The typical development workflow for database schema changes:
-
Create or modify schema in
db/schema/
files and update RLS policies indb/rls.sql
-
Generate migrations based on schema changes
-
Apply RLS policies
-
Push migrations to dev database
-
Test your changes thoroughly with the updated schema
-
If you’re not satisfied with the schema, you have two options:
Option A: Reset and start over (for significant changes)
Option B: Create additional migrations (for incremental changes)
-
Repeat steps 1-5 until satisfied with your schema
-
When ready for production, follow the safe deployment process
Safe Dev-to-Production Workflow
Always follow this workflow to safely apply database changes:
-
ALWAYS backup first
-
Apply to production
Quick Reference
Command Cheatsheet
Best Practices
-
For Database Access
- Use Direct Drizzle in Server Actions for performance and type safety
- Use React Query for client-side state management
- Always add authorization checks in server actions
-
For Schema Design
- Organize tables by domain
- Use proper indexes and constraints
- Add timestamps to all tables
-
For Security
- Define comprehensive RLS policies
- Use authorization checks in server actions
- Follow principle of least privilege
-
For Performance
- Use React Query’s caching
- Create proper database indexes
- Use transactions for related operations
Common Scenarios
Adding a New Table with RLS
- Create or edit schema file in
db/schema/
- Export the table from the main schema file
- Generate and apply migrations:
- Add RLS policies to
db/rls.sql
- Apply RLS policies:
Modifying an Existing Table (adding a new column etc.)
- Edit the schema file
- Generate and apply migrations:
Iterative Schema Development
When to reset vs. when to create additional migrations:
Reset approach (Option A) is best when:
- You’re in early development with no valuable data in the dev database
- You’ve made complex structural changes that are difficult to migrate
- You want a clean slate to avoid migration conflicts
- You’re experimenting with significantly different schema designs
Incremental migration approach (Option B) is best when:
- You have test data you want to preserve
- Changes are small and straightforward
- You’re closer to production and want to test actual migration paths
- You want to maintain migration history for documentation
Remember that in production, you’ll almost always use incremental migrations unless you’re doing a complete application redesign.
Troubleshooting Database Issues
- Reset database setup:
bun run db:init
- Check database state:
bun run db:studio
- For deeper issues, you may need to:
- Drop all tables:
bun run db:drop
- Regenerate migrations:
bun run db:generate
- Apply migrations:
bun run db:push
- Apply RLS:
bun run db:rls
- Seed data:
bun run db:seed
- Drop all tables: