Merge pull request #34112 from overleaf/dn0-migration-check-predeploy-hook
Add predeploy migration gate for Mongo-bundling services GitOrigin-RevId: d9eb192ea32b5328fb24fd453ddb1370f373858e
This commit is contained in:
@@ -0,0 +1,111 @@
|
||||
// Predeploy migration gate.
|
||||
//
|
||||
// Run as a Cloud Deploy predeploy hook (a K8s Job using the service's released image).
|
||||
// Blocks the rollout when there are pending migrations the image expects to have been applied.
|
||||
//
|
||||
// Exit codes:
|
||||
// 0 nothing pending, or all pending migrations are non-blocking
|
||||
// 1 one or more blocking pending migrations - rollout must not proceed
|
||||
// 2 the check itself failed (e.g. Mongo unreachable) - fail closed
|
||||
//
|
||||
// A migration is non-blocking when either:
|
||||
// - it is tagged "nonblocking" (preferred; safe to set on an already-applied migration), or
|
||||
// - its filename ends with "_nonBlocking" before the extension,
|
||||
// e.g. 20260101000000_backfill_thing_nonBlocking.mjs.
|
||||
|
||||
import path from 'node:path'
|
||||
import { fileURLToPath } from 'node:url'
|
||||
import east from 'east'
|
||||
|
||||
const { MigrationManager } = east
|
||||
|
||||
const NON_BLOCKING_SUFFIX = '_nonBlocking'
|
||||
const NON_BLOCKING_TAG = 'nonblocking'
|
||||
const TAG = process.env.MIGRATION_TAG || 'saas'
|
||||
|
||||
const scriptDir = path.dirname(fileURLToPath(import.meta.url))
|
||||
const migrationsDir = path.resolve(scriptDir, '..')
|
||||
|
||||
function skipRequested() {
|
||||
const value = process.env.SKIP_MIGRATION_CHECK
|
||||
return Boolean(value) && value !== '0' && value !== 'false'
|
||||
}
|
||||
|
||||
async function findBlockingMigrations() {
|
||||
// east resolves .eastrc, the migrations dir and the adapter relative to the
|
||||
// working directory (same as `npm run migrations`). chdir so the check works
|
||||
// regardless of how the container invokes it.
|
||||
process.chdir(migrationsDir)
|
||||
// The adapter aborts unless a tag is passed on argv; we pass the tag to east
|
||||
// directly, so disable that CLI-only guard.
|
||||
process.env.SKIP_TAG_CHECK = '1'
|
||||
|
||||
const manager = new MigrationManager()
|
||||
await manager.configure({ esModules: true })
|
||||
await manager.connect()
|
||||
|
||||
try {
|
||||
const pending = await manager.getMigrationNames({ status: 'new' })
|
||||
const candidates = pending.filter(
|
||||
name => !name.endsWith(NON_BLOCKING_SUFFIX)
|
||||
)
|
||||
const nonBlocking = pending.filter(name =>
|
||||
name.endsWith(NON_BLOCKING_SUFFIX)
|
||||
)
|
||||
const blocking = candidates.length
|
||||
? await manager.getMigrationNames({
|
||||
migrations: candidates,
|
||||
tag: `${TAG} & !${NON_BLOCKING_TAG}`,
|
||||
})
|
||||
: []
|
||||
|
||||
return { pending, nonBlocking, blocking }
|
||||
} finally {
|
||||
await manager.disconnect()
|
||||
}
|
||||
}
|
||||
|
||||
async function main() {
|
||||
if (skipRequested()) {
|
||||
console.warn(
|
||||
`SKIP_MIGRATION_CHECK is set ("${process.env.SKIP_MIGRATION_CHECK}") - ` +
|
||||
'bypassing the pending-migration gate. Only use this for incident recovery.'
|
||||
)
|
||||
return 0
|
||||
}
|
||||
|
||||
const { pending, nonBlocking, blocking } = await findBlockingMigrations()
|
||||
|
||||
if (nonBlocking.length) {
|
||||
console.log(
|
||||
`Ignoring ${nonBlocking.length} non-blocking pending migration(s):\n` +
|
||||
nonBlocking.map(name => ` - ${name}`).join('\n')
|
||||
)
|
||||
}
|
||||
|
||||
if (blocking.length === 0) {
|
||||
console.log(
|
||||
pending.length === 0
|
||||
? `No pending "${TAG}" migrations. Safe to deploy.`
|
||||
: `No blocking pending "${TAG}" migrations. Safe to deploy.`
|
||||
)
|
||||
return 0
|
||||
}
|
||||
|
||||
console.error(
|
||||
`${blocking.length} blocking pending "${TAG}" migration(s) must be applied before deploying:\n` +
|
||||
blocking.map(name => ` - ${name}`).join('\n') +
|
||||
`\n\nApply them, then retry the rollout. To intentionally ship a non-blocking ` +
|
||||
`migration, add the "${NON_BLOCKING_TAG}" tag (preferred — safe on already-applied ` +
|
||||
`migrations) or rename the file with the "${NON_BLOCKING_SUFFIX}" suffix.`
|
||||
)
|
||||
return 1
|
||||
}
|
||||
|
||||
main()
|
||||
.then(code => process.exit(code))
|
||||
.catch(err => {
|
||||
console.error('Migration check failed; blocking rollout (fail closed):')
|
||||
console.error(err)
|
||||
process.exit(2)
|
||||
})
|
||||
Reference in New Issue
Block a user