--- name: Backend Migration Standards description: Create and manage database migrations with reversible changes, proper naming conventions, and zero-downtime deployment strategies. Use this skill when creating database migration files, modifying schema, adding or removing tables/columns, managing indexes, or handling data migrations. Apply when working with migration files (e.g., db/migrate/, migrations/, alembic/, sequelize migrations), schema changes, database versioning, rollback implementations, or when you need to ensure backwards compatibility during deployments. Use for any task involving database structure changes, index creation, constraint modifications, or data transformation scripts. --- # Database Migration Standards Apply these rules when creating or modifying database migrations. Migrations are permanent records of schema evolution and must be treated with extreme care. ## When to use this skill - When creating new database migration files (db/migrate/, migrations/, alembic/, etc.) - When modifying database schema such as adding, removing, or altering tables and columns - When implementing rollback/down methods for reversible migrations - When creating indexes on database tables, especially large tables requiring concurrent indexing - When writing data migrations to transform or populate data - When planning zero-downtime deployments that require backwards-compatible schema changes - When establishing naming conventions for migration files - When separating schema changes from data migrations - When reviewing or modifying existing migrations for safety and clarity This Skill provides Claude Code with specific guidance on how to adhere to coding standards as they relate to how it should handle backend migrations. ## Core Principles **Reversibility is Mandatory**: Every migration MUST have a working rollback method. Test the down migration immediately after writing the up migration. If a change cannot be reversed safely (e.g., dropping a column with data), document why in comments and consider a multi-step approach. **One Logical Change Per Migration**: Each migration should do exactly one thing - add a table, add a column, create an index, etc. This makes debugging easier, rollbacks safer, and code review clearer. If you need to make multiple related changes, create multiple migrations. **Never Modify Deployed Migrations**: Once a migration runs in any shared environment (staging, production), it becomes immutable. Create a new migration to fix issues. Modifying deployed migrations breaks version control and causes deployment failures. ## Migration Structure **Naming Convention**: Use timestamps and descriptive names that indicate the change: - `20241118120000_add_email_to_users.py` - `20241118120100_create_orders_table.rb` - `20241118120200_add_index_on_users_email.js` The name should answer "what does this migration do?" without reading the code. **File Organization**: - Schema changes: `migrations/schema/` - Data migrations: `migrations/data/` - Keep them separate for rollback safety and clarity ## Schema Changes **Adding Columns**: Always specify default values for NOT NULL columns on existing tables to avoid locking issues: ```python # BAD - locks table during backfill op.add_column('users', sa.Column('status', sa.String(), nullable=False)) # GOOD - uses default, no lock op.add_column('users', sa.Column('status', sa.String(), nullable=False, server_default='active')) ``` **Removing Columns**: Use multi-step approach for zero-downtime: 1. Deploy code that stops using the column 2. Deploy migration that removes the column 3. Never combine these steps **Renaming Columns**: Treat as add + remove for zero-downtime: 1. Add new column 2. Deploy code that writes to both columns 3. Backfill data 4. Deploy code that reads from new column 5. Remove old column ## Index Management **Creating Indexes**: Use concurrent index creation on large tables to avoid blocking writes: ```python # PostgreSQL op.create_index('idx_users_email', 'users', ['email'], postgresql_using='btree', postgresql_concurrently=True) # MySQL op.create_index('idx_users_email', 'users', ['email'], mysql_algorithm='INPLACE', mysql_lock='NONE') ``` **Index Naming**: Use pattern `idx_