Skills › Data & Databases › Migrations
database-migrations
Database migration best practices for schema changes, data migrations, rollbacks, and zero-downtime deployments across PostgreSQL, MySQL, and common ORMs (Prisma, Drizzle, Django, TypeORM, golang-migrate). Use when planning or implementing database schema changes.
The full skill
—
name: database-migrations
description: >
Database migration best practices for schema changes, data migrations, rollbacks,
and zero-downtime deployments across PostgreSQL, MySQL, and common ORMs (Prisma,
Drizzle, Django, TypeORM, golang-migrate). Use when planning or implementing
database schema changes.
metadata:
origin: ECC
—
# Database Migration Patterns
Safe, reversible database schema changes for production systems.
## When to Activate
– Creating or altering database tables
– Adding/removing columns or indexes
– Running data migrations (backfill, transform)
– Planning zero-downtime schema changes
– Setting up migration tooling for a new project
## Core Principles
1. **Every change is a migration** — never alter production databases manually
2. **Migrations are forward-only in production** — rollbacks use new forward migrations
3. **Schema and data migrations are separate** — never mix DDL and DML in one migration
4. **Test migrations against production-sized data** — a migration that works on 100 rows may lock on 10M
5. **Migrations are immutable once deployed** — never edit a migration that has run in production
## Migration Safety Checklist
Before applying any migration:
– [ ] Migration has both UP and DOWN (or is explicitly marked irreversible)
– [ ] No full table locks on large tables (use concurrent operations)
– [ ] New columns have defaults or are nullable (never add NOT NULL without default)
– [ ] Indexes created concurrently (not inline with CREATE TABLE for existing tables)
– [ ] Data backfill is a separate migration from schema change
– [ ] Tested against a copy of production data
– [ ] Rollback plan documented
## PostgreSQL Patterns
### Adding a Column Safely
“`sql
— GOOD: Nullable column, no lock
ALTER TABLE users ADD COLUMN avatar_url TEXT;
— GOOD: Column with default (Postgres 11+ is instant, no rewrite)
ALTER TABLE users ADD COLUMN is_active BOOLEAN NOT NULL DEFAULT true;
— BAD: NOT NULL without default on existing table (requires full rewrite)
ALTER TABLE users ADD COLUMN role TEXT NOT NULL;
— This locks the table and rewrites every row
“`
### Adding an Index Without Downtime
“`sql
— BAD: Blocks writes on large tables
CREATE INDEX idx_users_email ON users (email);
— GOOD: Non-blocking, allows concurrent writes
CREATE INDEX CONCURRENTLY idx_users_email ON users (email);
— Note: CONCURRENTLY cannot run inside a transaction block
— Most migration tools need special handling for this
“`
### Renaming a Column (Zero-Downtime)
Never rename directly in production. Use the expand-contract pattern:
“`sql
— Step 1: Add new column (migration 001)
ALTER TABLE users ADD COLUMN display_name TEXT;
— Step 2: Backfill data (migration 002, data migration)
UPDATE users SET display_name = username WHERE display_name IS NULL;
— Step 3: Update application code to read/write both columns
— Deploy application changes
— Step 4: Stop writing to old column, drop it (migration 003)
ALTER TABLE users DROP COLUMN username;
“`
### Removing a Column Safely
“`sql
— Step 1: Remove all application references to the column
— Step 2: Deploy application without the column reference
— Step 3: Drop column in next migration
ALTER TABLE orders DROP COLUMN legacy_status;
— For Django: use SeparateDatabaseAndState to remove from model
— without generating DROP COLUMN (then drop in next migration)
“`
### Large Data Migrations
“`sql
— BAD: Updates all rows in one transaction (locks table)
UPDATE users SET normalized_email = LOWER(email);
— GOOD: Batch update with progress
DO $$
DECLARE
batch_size INT := 10000;
rows_updated INT;
BEGIN
LOOP
UPDATE users
SET normalized_email = LOWER(email)
WHERE id IN (
SELECT id FROM users
WHERE normalized_email IS NULL
LIMIT batch_size
FOR UPDATE SKIP LOCKED
);
GET DIAGNOSTICS rows_updated = ROW_COUNT;
RAISE NOTICE 'Updated % rows', rows_updated;
EXIT WHEN rows_updated = 0;
COMMIT;
END LOOP;
END $$;
“`
## Prisma (TypeScript/Node.js)
### Workflow
“`bash
# Create migration from schema changes
npx prisma migrate dev –name add_user_avatar
# Apply pending migrations in production
npx prisma migrate deploy
# Reset database (dev only)
npx prisma migrate reset
# Generate client after schema changes
npx prisma generate
“`
### Schema Example
“`prisma
model User {
id String @id @default(cuid())
email String @unique
name String?
avatarUrl String? @map("avatar_url")
createdAt DateTime @default(now()) @map("created_at")
updatedAt DateTime @updatedAt @map("updated_at")
orders Order[]
@@map("users")
@@index([email])
}
“`
### Custom SQL Migration
For operations Prisma cannot express (concurrent indexes, data backfills):
“`bash
# Create empty migration, then edit the SQL manually
npx prisma migrate dev –create-only –name add_email_index
“`
“`sql
— migrations/20240115_add_email_index/migration.sql
— Prisma cannot generate CONCURRENTLY, so we write it manually
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_users_email ON users (email);
“`
## Drizzle (TypeScript/Node.js)
### Workflow
“`bash
# Generate migration from schema changes
npx drizzle-kit generate
# Apply migrations
npx drizzle-kit migrate
# Push schema directly (dev only, no migration file)
npx drizzle-kit push
“`
### Schema Example
“`typescript
import { pgTable, text, timestamp, uuid, boolean } from "drizzle-orm/pg-core";
export const users = pgTable("users", {
id: uuid("id").primaryKey().defaultRandom(),
email: text("email").notNull().unique(),
name: text("name"),
isActive: boolean("is_active").notNull().default(true),
createdAt: timestamp("created_at").notNull().defaultNow(),
updatedAt: timestamp("updated_at").notNull().defaultNow(),
});
“`
## Django (Python)
### Workflow
“`bash
# Generate migration from model changes
python manage.py makemigrations
# Apply migrations
python manage.py migrate
# Show migration status
python manage.py showmigrations
# Generate empty migration for custom SQL
python manage.py makemigrations –empty app_name -n description
“`
### Data Migration
“`python
from django.db import migrations
def backfill_display_names(apps, schema_editor):
User = apps.get_model("accounts", "User")
batch_size = 5000
users = User.objects.filter(display_name="")
while users.exists():
batch = list(users[:batch_size])
for user in batch:
user.display_name = user.username
User.objects.bulk_update(batch, ["display_name"], batch_size=batch_size)
def reverse_backfill(apps, schema_editor):
pass # Data migration, no reverse needed
class Migration(migrations.Migration):
dependencies = [("accounts", "0015_add_display_name")]
operations = [
migrations.RunPython(backfill_display_names, reverse_backfill),
]
“`
### SeparateDatabaseAndState
Remove a column from the Django model without dropping it from the database immediately:
“`python
class Migration(migrations.Migration):
operations = [
migrations.SeparateDatabaseAndState(
state_operations=[
migrations.RemoveField(model_name="user", name="legacy_field"),
],
database_operations=[], # Don't touch the DB yet
),
]
“`
## golang-migrate (Go)
### Workflow
“`bash
# Create migration pair
migrate create -ext sql -dir migrations -seq add_user_avatar
# Apply all pending migrations
migrate -path migrations -database "$DATABASE_URL" up
# Rollback last migration
migrate -path migrations -database "$DATABASE_URL" down 1
# Force version (fix dirty state)
migrate -path migrations -database "$DATABASE_URL" force VERSION
“`
### Migration Files
“`sql
— migrations/000003_add_user_avatar.up.sql
ALTER TABLE users ADD COLUMN avatar_url TEXT;
CREATE INDEX CONCURRENTLY idx_users_avatar ON users (avatar_url) WHERE avatar_url IS NOT NULL;
— migrations/000003_add_user_avatar.down.sql
DROP INDEX IF EXISTS idx_users_avatar;
ALTER TABLE users DROP COLUMN IF EXISTS avatar_url;
“`
## Zero-Downtime Migration Strategy
For critical production changes, follow the expand-contract pattern:
“`
Phase 1: EXPAND
– Add new column/table (nullable or with default)
– Deploy: app writes to BOTH old and new
– Backfill existing data
Phase 2: MIGRATE
– Deploy: app reads from NEW, writes to BOTH
– Verify data consistency
Phase 3: CONTRACT
– Deploy: app only uses NEW
– Drop old column/table in separate migration
“`
### Timeline Example
“`
Day 1: Migration adds new_status column (nullable)
Day 1: Deploy app v2 — writes to both status and new_status
Day 2: Run backfill migration for existing rows
Day 3: Deploy app v3 — reads from new_status only
Day 7: Migration drops old status column
“`
## Anti-Patterns
| Anti-Pattern | Why It Fails | Better Approach |
|————-|————-|—————–|
| Manual SQL in production | No audit trail, unrepeatable | Always use migration files |
| Editing deployed migrations | Causes drift between environments | Create new migration instead |
| NOT NULL without default | Locks table, rewrites all rows | Add nullable, backfill, then add constraint |
| Inline index on large table | Blocks writes during build | CREATE INDEX CONCURRENTLY |
| Schema + data in one migration | Hard to rollback, long transactions | Separate migrations |
| Dropping column before removing code | Application errors on missing column | Remove code first, drop column next deploy |
## When to Use This Skill
– Planning database schema changes
– Implementing zero-downtime migrations
– Setting up migration tooling
– Troubleshooting migration issues
– Reviewing migration pull requests