Process migration

This commit is contained in:
Adria Navarro 2023-11-29 11:03:33 +01:00
parent f2fcf0f6c2
commit 1d124a59cb
3 changed files with 72 additions and 10 deletions

View File

@ -1,5 +1,5 @@
import { Duration, cache, db, env } from "@budibase/backend-core" import { Duration, cache, context, db, env } from "@budibase/backend-core"
import { Database, App, DocumentType, Document } from "@budibase/types" import { Database, DocumentType, Document } from "@budibase/types"
export interface AppMigrationDoc extends Document { export interface AppMigrationDoc extends Document {
version: string version: string
@ -7,23 +7,25 @@ export interface AppMigrationDoc extends Document {
const EXPIRY_SECONDS = Duration.fromDays(1).toSeconds() const EXPIRY_SECONDS = Duration.fromDays(1).toSeconds()
async function populateFromDB(appId: string) { async function getFromDB(appId: string) {
return db.doWithDB( return db.doWithDB(
appId, appId,
(db: Database) => { (db: Database) => {
return db.get<App>(DocumentType.APP_MIGRATION_METADATA) return db.get<AppMigrationDoc>(DocumentType.APP_MIGRATION_METADATA)
}, },
{ skip_setup: true } { skip_setup: true }
) )
} }
const getCacheKey = (appId: string) => `appmigrations_${env.VERSION}_${appId}`
export async function getAppMigrationMetadata(appId: string): Promise<string> { export async function getAppMigrationMetadata(appId: string): Promise<string> {
const cacheKey = `appmigrations_${env.VERSION}_${appId}` const cacheKey = getCacheKey(appId)
let metadata: AppMigrationDoc | undefined = await cache.get(cacheKey) let metadata: AppMigrationDoc | undefined = await cache.get(cacheKey)
if (!metadata || env.isDev()) { if (!metadata || env.isDev()) {
try { try {
metadata = await populateFromDB(appId) metadata = await getFromDB(appId)
} catch (err: any) { } catch (err: any) {
if (err.status !== 404) { if (err.status !== 404) {
throw err throw err
@ -37,3 +39,19 @@ export async function getAppMigrationMetadata(appId: string): Promise<string> {
return metadata.version return metadata.version
} }
export async function updateAppMigrationMetadata({
appId,
version,
}: {
appId: string
version: string
}): Promise<void> {
const db = context.getAppDB()
const appMigrationDoc = await getFromDB(appId)
await db.put({ ...appMigrationDoc, version })
const cacheKey = getCacheKey(appId)
await cache.destroy(cacheKey)
}

View File

@ -1,4 +1,4 @@
import queue from "./queue" import queue, { PROCESS_MIGRATION_TIMEOUT } from "./queue"
import { getAppMigrationMetadata } from "./appMigrationMetadata" import { getAppMigrationMetadata } from "./appMigrationMetadata"
import { MIGRATIONS } from "./migrations" import { MIGRATIONS } from "./migrations"
@ -13,9 +13,10 @@ export async function checkMissingMigrations(appId: string) {
appId, appId,
}, },
{ {
jobId: appId, jobId: `${appId}_${latestMigration}`,
removeOnComplete: true, removeOnComplete: true,
removeOnFail: true, removeOnFail: true,
timeout: PROCESS_MIGRATION_TIMEOUT,
} }
) )
} }

View File

@ -1,13 +1,56 @@
import { queue } from "@budibase/backend-core" import { context, locks, queue } from "@budibase/backend-core"
import { LockName, LockType } from "@budibase/types"
import { Job } from "bull" import { Job } from "bull"
import { MIGRATIONS } from "./migrations"
import {
getAppMigrationMetadata,
updateAppMigrationMetadata,
} from "./appMigrationMetadata"
const appMigrationQueue = queue.createQueue(queue.JobQueue.APP_MIGRATION) const appMigrationQueue = queue.createQueue(queue.JobQueue.APP_MIGRATION)
appMigrationQueue.process(processMessage) appMigrationQueue.process(processMessage)
export async function runMigration(migrationId: string) {
await MIGRATIONS[migrationId].migration()
}
// TODO
export const PROCESS_MIGRATION_TIMEOUT = 30000
async function processMessage(job: Job) { async function processMessage(job: Job) {
const { appId } = job.data const { appId } = job.data
console.log(`Processing app migration for "${appId}"`)
console.log(appId) await locks.doWithLock(
{
name: LockName.APP_MIGRATION,
type: LockType.DEFAULT,
resource: appId,
ttl: PROCESS_MIGRATION_TIMEOUT,
},
async () => {
await context.doInAppContext(appId, async () => {
const currentVersion = await getAppMigrationMetadata(appId)
const pendingMigrations = Object.keys(MIGRATIONS).filter(
m => m > currentVersion
)
let index = 0
for (const migration of pendingMigrations) {
const counter = `(${++index}/${pendingMigrations.length})`
console.info(`Running migration ${migration}... ${counter}`, {
migration,
appId,
})
await runMigration(migration)
await updateAppMigrationMetadata({ appId, version: migration })
}
})
}
)
console.log(`App migration for "${appId}" processed`)
} }
export default appMigrationQueue export default appMigrationQueue