knex-migration-with-schema-by-riyad v1.0.1
Simplifies the execution of database migrations across different schemas with Knex
While knex.js supports the execution of queries across different schemas with the use of the withSchema() method, there's no simple way to run migration across different schemas programmatically, sending the schema name as a parameter to the knex.migrate() command.
This library was created to address this issue.
How to use
This library offers two functions, one to create new schemas, and one to execute migrations on a schema
npm i knex-migration-with-schemaCreating new schemas
Import the createSchema method:
import { createSchema } from './creator'Create a schema by providing a knex connection and a schemaName. If the schema already exists, it will be skipped (no exceptions will be thrown)
await createSchema({ knex, schemaName: 'users' })Executing migrations on a schema
Import the executeSchemaMigration method:
import { executeSchemaMigration } from 'knex-migration-with-schema'Execute the migrations passing as parameters an open knex database connection (knex), the name of the schema in which you want to (schemaName) and the migrations you want to execute, as a dictionary (migrations)
const schemaName = 'users'
const userMigrations = {
  createUsersTable: (schemaName) => ({
    async up(knex) {
      return knex.schema.withSchema(schemaName).createTable('users', (table) => {
        table.increments('id').primary()
        table.text('email').notNullable().unique
      })
    },
    async down(knex) {
      return knex.schema.withSchema(schemaName).dropTableIfExists('users')
    },
  }),
}
await executeSchemaMigration({ knex, schemaName, migrations: userMigrations })Note that the migrations dictionary has the name of the migration as key (the one that will be stored in the knex_migrations table), and the up and down functions as values.
To add new migrations you can simply expand this object; as expected, only the non-executed migrations will be performed:
const userMigrations = {
  createUsersTable: (schemaName) => ({
    async up(knex) {
      return knex.schema.withSchema(schemaName).createTable('users', (table) => {
        table.increments('id').primary()
        table.text('email').notNullable().unique
      })
    },
    async down(knex) {
      return knex.schema.withSchema(schemaName).dropTableIfExists('users')
    },
  }),
  addUserNameColumn: (schemaName) => ({
    async up(knex) {
      return knex.schema.withSchema(schemaName).table('users', (table) => {
        table.text('user_name')
      })
    },
    async down(knex) {
      return knex.schema.withSchema(schemaName).table('users', (table) => {
        table.dropColumn('user_name')
      })
    },
  }),
)Executing migrations on a schema based on a directory
await executeSchemaMigrationFromDir({
  knex,
  schemaName,
  directory: `migration_files`,
})Using the executeSchemaMigrationFromDir function, you have access to the schema name in your migration files.
// migration_files/0001_create_customers_table.ts
import { Knex } from 'knex'
export default (schemaName: string): Knex.Migration => ({
  async up(knex: Knex) {
    return knex.schema.withSchema(schemaName).createTable('customers', (table) => {
      table.increments('id').primary()
      table.text('name').notNullable()
    })
  },
  async down(knex: Knex) {
    return knex.schema.withSchema(schemaName).dropTableIfExists('customers')
  },
})