Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add allowUnorderedMigrations option to Migrator #723

Merged
merged 10 commits into from
Jan 5, 2024
16 changes: 15 additions & 1 deletion site/docs/migrations.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,21 @@ Migrations can use the `Kysely.schema` module to modify the schema. Migrations c

## Execution order

Execution order of the migrations is the alpabetical order of their names. An excellent way to name your migrations is to prefix them with an ISO 8601 date string. A date prefix works well in large teams where multiple team members may add migrations at the same time in parallel commits without knowing about the other migrations.
Migrations will be run in the alpha-numeric order of your migration names. An excellent way to name your migrations is to prefix them with an ISO 8601 date string.

By default, Kysely will also ensure this order matches the execution order of any previously executed migrations in your database. If the orders do not match (for example, a new migration was added alphabetically before a previously executed one), an error will be returned. This adds safety by always executing your migrations in the correct, alphanumeric order.

There is also an `allowUnorderedMigrations` option. This option will allow new migrations to be run even if they are added alphabetically before ones that have already executed. Allowing unordered migrations works well in large teams where multiple team members may add migrations at the same time in parallel commits without knowing about the other migrations. Pending (unexecuted) migrations will be run in alpha-numeric order when migrating up. When migrating down, migrations will be undone in the opposite order in which they were executed (reverse sorted by execution timestamp).

To allow unordered migrations, pass the `allowUnorderedMigrations` option to Migrator:

```ts
const migrator = new Migrator({
db,
provider: new FileMigrationProvider(...),
allowUnorderedMigrations: true
})
```

## Single file vs multiple file migrations

Expand Down
202 changes: 141 additions & 61 deletions src/migration/migrator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { freeze, getLast } from '../util/object-utils.js'

export const DEFAULT_MIGRATION_TABLE = 'kysely_migration'
export const DEFAULT_MIGRATION_LOCK_TABLE = 'kysely_migration_lock'
export const DEFAULT_ALLOW_UNORDERED_MIGRATIONS = false
export const MIGRATION_LOCK_ID = 'migration_lock'
export const NO_MIGRATIONS: NoMigrations = freeze({ __noMigrations__: true })

Expand Down Expand Up @@ -96,7 +97,7 @@ export class Migrator {
* This method goes through all possible migrations provided by the provider and runs the
* ones whose names come alphabetically after the last migration that has been run. If the
* list of executed migrations doesn't match the beginning of the list of possible migrations
* an error is thrown.
* an error is returned.
*
* ### Examples
*
Expand Down Expand Up @@ -133,7 +134,7 @@ export class Migrator {
* ```
*/
async migrateToLatest(): Promise<MigrationResultSet> {
return this.#migrate(({ migrations }) => migrations.length - 1)
return this.#migrate(() => ({ direction: 'Up', step: Infinity }))
}

/**
Expand Down Expand Up @@ -162,21 +163,43 @@ export class Migrator {
async migrateTo(
targetMigrationName: string | NoMigrations
): Promise<MigrationResultSet> {
return this.#migrate(({ migrations }) => {
if (targetMigrationName === NO_MIGRATIONS) {
return -1
}
return this.#migrate(
({
migrations,
executedMigrations,
pendingMigrations,
}: MigrationState) => {
if (targetMigrationName === NO_MIGRATIONS) {
return { direction: 'Down', step: Infinity }
}

const index = migrations.findIndex(
(it) => it.name === targetMigrationName
)
if (
!migrations.find((m) => m.name === (targetMigrationName as string))
) {
throw new Error(`migration "${targetMigrationName}" doesn't exist`)
}

if (index === -1) {
throw new Error(`migration "${targetMigrationName}" doesn't exist`)
}
const executedIndex = executedMigrations.indexOf(
targetMigrationName as string
)
const pendingIndex = pendingMigrations.findIndex(
(m) => m.name === (targetMigrationName as string)
)

return index
})
if (executedIndex !== -1) {
return {
direction: 'Down',
step: executedMigrations.length - executedIndex - 1,
}
} else if (pendingIndex !== -1) {
return { direction: 'Up', step: pendingIndex + 1 }
} else {
throw new Error(
`migration "${targetMigrationName}" isn't executed or pending`
)
}
}
)
}

/**
Expand All @@ -194,9 +217,7 @@ export class Migrator {
* ```
*/
async migrateUp(): Promise<MigrationResultSet> {
return this.#migrate(({ currentIndex, migrations }) =>
Math.min(currentIndex + 1, migrations.length - 1)
)
return this.#migrate(() => ({ direction: 'Up', step: 1 }))
}

/**
Expand All @@ -214,15 +235,18 @@ export class Migrator {
* ```
*/
async migrateDown(): Promise<MigrationResultSet> {
return this.#migrate(({ currentIndex }) => Math.max(currentIndex - 1, -1))
return this.#migrate(() => ({ direction: 'Down', step: 1 }))
}

async #migrate(
getTargetMigrationIndex: (state: MigrationState) => number | undefined
getMigrationDirectionAndStep: (state: MigrationState) => {
direction: MigrationDirection
step: number
}
): Promise<MigrationResultSet> {
try {
await this.#ensureMigrationTablesExists()
return await this.#runMigrations(getTargetMigrationIndex)
return await this.#runMigrations(getMigrationDirectionAndStep)
} catch (error) {
if (error instanceof MigrationResultSetError) {
return error.resultSet
Expand All @@ -244,6 +268,12 @@ export class Migrator {
return this.#props.migrationLockTableName ?? DEFAULT_MIGRATION_LOCK_TABLE
}

get #allowUnorderedMigrations(): boolean {
return (
this.#props.allowUnorderedMigrations ?? DEFAULT_ALLOW_UNORDERED_MIGRATIONS
)
}

get #schemaPlugin(): KyselyPlugin {
if (this.#migrationTableSchema) {
return new WithSchemaPlugin(this.#migrationTableSchema)
Expand Down Expand Up @@ -383,7 +413,10 @@ export class Migrator {
}

async #runMigrations(
getTargetMigrationIndex: (state: MigrationState) => number | undefined
getMigrationDirectionAndStep: (state: MigrationState) => {
direction: MigrationDirection
step: number
}
): Promise<MigrationResultSet> {
const adapter = this.#props.db.getExecutor().adapter

Expand All @@ -397,23 +430,22 @@ export class Migrator {
const run = async (db: Kysely<any>): Promise<MigrationResultSet> => {
try {
await adapter.acquireMigrationLock(db, lockOptions)

const state = await this.#getState(db)

if (state.migrations.length === 0) {
return { results: [] }
}

const targetIndex = getTargetMigrationIndex(state)
const { direction, step } = getMigrationDirectionAndStep(state)

if (targetIndex === undefined) {
if (step <= 0) {
return { results: [] }
}

if (targetIndex < state.currentIndex) {
return await this.#migrateDown(db, state, targetIndex)
} else if (targetIndex > state.currentIndex) {
return await this.#migrateUp(db, state, targetIndex)
if (direction === 'Down') {
return await this.#migrateDown(db, state, step)
} else if (direction === 'Up') {
return await this.#migrateUp(db, state, step)
}

return { results: [] }
Expand All @@ -433,13 +465,30 @@ export class Migrator {
const migrations = await this.#resolveMigrations()
const executedMigrations = await this.#getExecutedMigrations(db)

this.#ensureMigrationsNotCorrupted(migrations, executedMigrations)
this.#ensureNoMissingMigrations(migrations, executedMigrations)
if (!this.#allowUnorderedMigrations) {
this.#ensureMigrationsInOrder(migrations, executedMigrations)
}

const pendingMigrations = this.#getPendingMigrations(
migrations,
executedMigrations
)

return freeze({
migrations,
currentIndex: migrations.findIndex(
(it) => it.name === getLast(executedMigrations)
),
executedMigrations,
lastMigration: getLast(executedMigrations),
pendingMigrations,
})
}

#getPendingMigrations(
migrations: ReadonlyArray<NamedMigration>,
executedMigrations: ReadonlyArray<string>
): ReadonlyArray<NamedMigration> {
return migrations.filter((migration) => {
return !executedMigrations.includes(migration.name)
})
}

Expand All @@ -461,27 +510,31 @@ export class Migrator {
.withPlugin(this.#schemaPlugin)
.selectFrom(this.#migrationTable)
.select('name')
.orderBy('name')
.orderBy(['timestamp', 'name'])
.execute()

return executedMigrations.map((it) => it.name)
}

#ensureMigrationsNotCorrupted(
#ensureNoMissingMigrations(
migrations: ReadonlyArray<NamedMigration>,
executedMigrations: ReadonlyArray<string>
) {
// Ensure all executed migrations exist in the `migrations` list.
for (const executed of executedMigrations) {
if (!migrations.some((it) => it.name === executed)) {
throw new Error(
`corrupted migrations: previously executed migration ${executed} is missing`
)
}
}
}

// Now we know all executed migrations exist in the `migrations` list.
// Next we need to make sure that the executed migratiosns are the first
// ones in the migration list.
#ensureMigrationsInOrder(
migrations: ReadonlyArray<NamedMigration>,
executedMigrations: ReadonlyArray<string>
) {
// Ensure the executed migrations are the first ones in the migration list.
for (let i = 0; i < executedMigrations.length; ++i) {
if (migrations[i].name !== executedMigrations[i]) {
throw new Error(
Expand All @@ -494,22 +547,27 @@ export class Migrator {
async #migrateDown(
db: Kysely<any>,
state: MigrationState,
targetIndex: number
step: number
): Promise<MigrationResultSet> {
const results: MigrationResult[] = []
const migrationsToRollback: ReadonlyArray<NamedMigration> =
state.executedMigrations
.slice()
.reverse()
.slice(0, step)
.map((name) => {
return state.migrations.find((it) => it.name === name)!
})

for (let i = state.currentIndex; i > targetIndex; --i) {
results.push({
migrationName: state.migrations[i].name,
const results: MigrationResult[] = migrationsToRollback.map((migration) => {
return {
migrationName: migration.name,
direction: 'Down',
status: 'NotExecuted',
})
}
}
})

for (let i = 0; i < results.length; ++i) {
const migration = state.migrations.find(
(it) => it.name === results[i].migrationName
)!
const migration = migrationsToRollback[i]

try {
if (migration.down) {
Expand Down Expand Up @@ -546,22 +604,21 @@ export class Migrator {
async #migrateUp(
db: Kysely<any>,
state: MigrationState,
targetIndex: number
step: number
): Promise<MigrationResultSet> {
const results: MigrationResult[] = []
const migrationsToRun: ReadonlyArray<NamedMigration> =
state.pendingMigrations.slice(0, step)

for (let i = state.currentIndex + 1; i <= targetIndex; ++i) {
results.push({
migrationName: state.migrations[i].name,
const results: MigrationResult[] = migrationsToRun.map((migration) => {
return {
migrationName: migration.name,
direction: 'Up',
status: 'NotExecuted',
})
}
}
})

for (let i = 0; i < results.length; ++i) {
const migration = state.migrations.find(
(it) => it.name === results[i].migrationName
)!
for (let i = 0; i < results.length; i++) {
const migration = state.pendingMigrations[i]

try {
await migration.up(db)
Expand Down Expand Up @@ -657,6 +714,21 @@ export interface MigratorProps {
* This only works on postgres.
*/
readonly migrationTableSchema?: string

/**
* Enforces whether or not migrations must be run in alpha-numeric order.
*
* When false, migrations must be run in their exact alpha-numeric order.
* This is checked against the migrations already run in the database
* (`migrationTableName'). This ensures your migrations are always run in
* the same order and is the safest option.
*
* When true, migrations are still run in alpha-numeric order, but
* the order is not checked against already-run migrations in the database.
* Kysely will simply run all migrations that haven't run yet, in alpha-numeric
* order.
*/
readonly allowUnorderedMigrations?: boolean
}

/**
Expand Down Expand Up @@ -694,13 +766,15 @@ export interface MigrationResultSet {
readonly results?: MigrationResult[]
}

type MigrationDirection = 'Up' | 'Down'

export interface MigrationResult {
readonly migrationName: string

/**
* The direction in which this migration was executed.
*/
readonly direction: 'Up' | 'Down'
readonly direction: MigrationDirection

/**
* The execution status.
Expand Down Expand Up @@ -772,8 +846,14 @@ interface MigrationState {
// All migrations sorted by name.
readonly migrations: ReadonlyArray<NamedMigration>

// Index of the last executed migration.
readonly currentIndex: number
// Names of executed migrations sorted by execution timestamp
readonly executedMigrations: ReadonlyArray<string>

// Name of the last executed migration.
readonly lastMigration?: string

// Migrations that have not yet ran
readonly pendingMigrations: ReadonlyArray<NamedMigration>
}

class MigrationResultSetError extends Error {
Expand Down
Loading
Loading