Provides database migration best practices across PostgreSQL, MySQL, and ORMs (Prisma, Drizzle, Django, TypeORM, etc.) covering schema changes, data migrations, rollbacks, and zero-downtime deployments.
How this skill is triggered — by the user, by Claude, or both
Slash command
/everything-claude-code:database-migrationsThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
为生产系统提供安全、可逆的数据库模式变更。
为生产系统提供安全、可逆的数据库模式变更。
应用任何迁移之前:
-- 好:可为空的列,无锁
ALTER TABLE users ADD COLUMN avatar_url TEXT;
-- 好:带默认值的列(Postgres 11+ 即时完成,无需重写)
ALTER TABLE users ADD COLUMN is_active BOOLEAN NOT NULL DEFAULT true;
-- 差:在现有表上没有默认值的 NOT NULL(需要完全重写)
ALTER TABLE users ADD COLUMN role TEXT NOT NULL;
-- 这会锁定表并重写每一行
-- 差:在大表上阻塞写入
CREATE INDEX idx_users_email ON users (email);
-- 好:非阻塞,允许并发写入
CREATE INDEX CONCURRENTLY idx_users_email ON users (email);
-- 注意:CONCURRENTLY 不能在事务块内运行
-- 大多数迁移工具需要对此做特殊处理
绝不在生产中直接重命名。使用扩展-收缩模式:
-- 步骤 1:添加新列(迁移 001)
ALTER TABLE users ADD COLUMN display_name TEXT;
-- 步骤 2:回填数据(迁移 002,数据迁移)
UPDATE users SET display_name = username WHERE display_name IS NULL;
-- 步骤 3:更新应用代码同时读写两列
-- 部署应用变更
-- 步骤 4:停止写入旧列,删除它(迁移 003)
ALTER TABLE users DROP COLUMN username;
-- 步骤 1:移除所有对该列的应用引用
-- 步骤 2:部署不含该列引用的应用
-- 步骤 3:在下一次迁移中删除列
ALTER TABLE orders DROP COLUMN legacy_status;
-- 对于 Django:使用 SeparateDatabaseAndState 从模型中移除
-- 而不生成 DROP COLUMN(然后在下次迁移中删除)
-- 差:在一个事务中更新所有行(锁定表)
UPDATE users SET normalized_email = LOWER(email);
-- 好:带进度的批量更新
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 '已更新 % 行', rows_updated;
EXIT WHEN rows_updated = 0;
COMMIT;
END LOOP;
END $$;
# 从模式变更创建迁移
npx prisma migrate dev --name add_user_avatar
# 在生产中应用待处理的迁移
npx prisma migrate deploy
# 重置数据库(仅开发)
npx prisma migrate reset
# 模式变更后生成客户端
npx prisma generate
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])
}
对于 Prisma 无法表达的操作(并发索引、数据回填):
# 创建空迁移,然后手动编辑 SQL
npx prisma migrate dev --create-only --name add_email_index
-- migrations/20240115_add_email_index/migration.sql
-- Prisma 无法生成 CONCURRENTLY,所以手动编写
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_users_email ON users (email);
# 从模式变更生成迁移
npx drizzle-kit generate
# 应用迁移
npx drizzle-kit migrate
# 直接推送模式(仅开发,无迁移文件)
npx drizzle-kit push
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(),
});
# 初始化配置文件(kysely.config.ts)
kysely init
# 创建新迁移文件
kysely migrate make add_user_avatar
# 应用所有待处理迁移
kysely migrate latest
# 回滚最后一次迁移
kysely migrate down
# 显示迁移状态
kysely migrate list
// migrations/2024_01_15_001_create_user_profile.ts
import { type Kysely, sql } from 'kysely'
// 重要:始终使用 Kysely<any>,不要使用你类型化的 DB 接口。
// 迁移在时间上被冻结,不能依赖当前模式类型。
export async function up(db: Kysely<any>): Promise<void> {
await db.schema
.createTable('user_profile')
.addColumn('id', 'serial', (col) => col.primaryKey())
.addColumn('email', 'varchar(255)', (col) => col.notNull().unique())
.addColumn('avatar_url', 'text')
.addColumn('created_at', 'timestamp', (col) =>
col.defaultTo(sql`now()`).notNull()
)
.execute()
await db.schema
.createIndex('idx_user_profile_avatar')
.on('user_profile')
.column('avatar_url')
.execute()
}
export async function down(db: Kysely<any>): Promise<void> {
await db.schema.dropTable('user_profile').execute()
}
import { Migrator, FileMigrationProvider } from 'kysely'
import { promises as fs } from 'fs'
import * as path from 'path'
// 仅 ESM — CJS 可直接使用 __dirname
import { fileURLToPath } from 'url'
const migrationFolder = path.join(
path.dirname(fileURLToPath(import.meta.url)),
'./migrations',
)
// `db` 是你的 Kysely<any> 数据库实例
const migrator = new Migrator({
db,
provider: new FileMigrationProvider({
fs,
path,
migrationFolder,
}),
// 警告:仅在开发中启用。禁用时间戳排序验证,
// 可能导致环境间的模式漂移。
// allowUnorderedMigrations: true,
})
const { error, results } = await migrator.migrateToLatest()
results?.forEach((it) => {
if (it.status === 'Success') {
console.log(`迁移 "${it.migrationName}" 执行成功`)
} else if (it.status === 'Error') {
console.error(`迁移 "${it.migrationName}" 执行失败`)
}
})
if (error) {
console.error('迁移失败', error)
process.exit(1)
}
# 从模型变更生成迁移
python manage.py makemigrations
# 应用迁移
python manage.py migrate
# 显示迁移状态
python manage.py showmigrations
# 为自定义 SQL 生成空迁移
python manage.py makemigrations --empty app_name -n description
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 # 数据迁移,无需反向操作
class Migration(migrations.Migration):
dependencies = [("accounts", "0015_add_display_name")]
operations = [
migrations.RunPython(backfill_display_names, reverse_backfill),
]
从 Django 模型中移除列而不立即从数据库中删除:
class Migration(migrations.Migration):
operations = [
migrations.SeparateDatabaseAndState(
state_operations=[
migrations.RemoveField(model_name="user", name="legacy_field"),
],
database_operations=[], # 暂不触碰数据库
),
]
# 创建迁移对
migrate create -ext sql -dir migrations -seq add_user_avatar
# 应用所有待处理迁移
migrate -path migrations -database "$DATABASE_URL" up
# 回滚最后一次迁移
migrate -path migrations -database "$DATABASE_URL" down 1
# 强制版本(修复脏状态)
migrate -path migrations -database "$DATABASE_URL" force VERSION
-- 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;
对于关键生产变更,遵循扩展-收缩模式:
阶段 1:扩展
- 添加新列/表(可为空或有默认值)
- 部署:应用同时写入新旧两处
- 回填现有数据
阶段 2:迁移
- 部署:应用从新处读取,写入新旧两处
- 验证数据一致性
阶段 3:收缩
- 部署:应用仅使用新的
- 在单独的迁移中删除旧列/表
第 1 天:迁移添加 new_status 列(可为空)
第 1 天:部署应用 v2 — 同时写入 status 和 new_status
第 2 天:运行回填迁移处理现有行
第 3 天:部署应用 v3 — 仅从 new_status 读取
第 7 天:迁移删除旧 status 列
| 反模式 | 为什么会失败 | 更好的方法 |
|---|---|---|
| 在生产中手动执行 SQL | 没有审计跟踪,不可重复 | 始终使用迁移文件 |
| 编辑已部署的迁移 | 导致环境间漂移 | 创建新迁移替代 |
| 没有默认值的 NOT NULL | 锁定表,重写所有行 | 先添加可为空的列,回填,然后添加约束 |
| 在大表上内联索引 | 构建期间阻塞写入 | CREATE INDEX CONCURRENTLY |
| 一个迁移中的模式 + 数据 | 难以回滚,长事务 | 分离迁移 |
| 在移除代码之前删除列 | 应用因缺少列出错 | 先移除代码,下次部署再删列 |
npx claudepluginhub aaione/everything-claude-code-zhGuides safe schema changes, data migrations, rollbacks, and zero-downtime deployments for PostgreSQL, MySQL, and ORMs including Prisma, Drizzle, Django, TypeORM, golang-migrate. Use for table modifications, column additions, indexes, backfills.
Provides database migration best practices for schema changes, data migrations, rollbacks, and zero-downtime deployments across PostgreSQL, MySQL, and ORMs like Prisma, Drizzle, and Django.
Creates, validates, executes, and rolls back schema migrations for PostgreSQL, MySQL, MongoDB using Flyway, Alembic, Prisma, Knex.