2024-03-01 12:05:27 +01:00
|
|
|
{.push raises: [].}
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
import std/[tables, strutils, os], stew/results, chronicles, chronos
|
2024-03-01 12:05:27 +01:00
|
|
|
import
|
|
|
|
../../../common/databases/common,
|
|
|
|
../../../../migrations/message_store_postgres/pg_migration_manager,
|
|
|
|
../postgres_driver
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
topics = "waku archive migration"
|
|
|
|
|
2024-03-06 20:50:22 +01:00
|
|
|
const SchemaVersion* = 2 # increase this when there is an update in the database schema
|
2024-03-01 12:05:27 +01:00
|
|
|
|
|
|
|
proc breakIntoStatements*(script: string): seq[string] =
|
|
|
|
## Given a full migration script, that can potentially contain a list
|
|
|
|
## of SQL statements, this proc splits it into the contained isolated statements
|
|
|
|
## that should be executed one after the other.
|
|
|
|
var statements = newSeq[string]()
|
|
|
|
|
|
|
|
let lines = script.split('\n')
|
|
|
|
|
|
|
|
var simpleStmt: string
|
|
|
|
var plSqlStatement: string
|
|
|
|
var insidePlSqlScript = false
|
|
|
|
for line in lines:
|
|
|
|
if line.strip().len == 0:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if insidePlSqlScript:
|
|
|
|
if line.contains("END $$"):
|
|
|
|
## End of the Pl/SQL script
|
|
|
|
plSqlStatement &= line
|
|
|
|
statements.add(plSqlStatement)
|
|
|
|
plSqlStatement = ""
|
|
|
|
insidePlSqlScript = false
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
plSqlStatement &= line & "\n"
|
|
|
|
|
|
|
|
if line.contains("DO $$"):
|
|
|
|
## Beginning of the Pl/SQL script
|
|
|
|
insidePlSqlScript = true
|
|
|
|
plSqlStatement &= line & "\n"
|
|
|
|
|
|
|
|
if not insidePlSqlScript:
|
|
|
|
if line.contains(';'):
|
|
|
|
## End of simple statement
|
|
|
|
simpleStmt &= line
|
|
|
|
statements.add(simpleStmt)
|
|
|
|
simpleStmt = ""
|
|
|
|
else:
|
|
|
|
simpleStmt &= line & "\n"
|
|
|
|
|
|
|
|
return statements
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
proc migrate*(
|
|
|
|
driver: PostgresDriver, targetVersion = SchemaVersion
|
|
|
|
): Future[DatabaseResult[void]] {.async.} =
|
2024-03-01 12:05:27 +01:00
|
|
|
debug "starting message store's postgres database migration"
|
|
|
|
|
|
|
|
let currentVersion = (await driver.getCurrentVersion()).valueOr:
|
|
|
|
return err("migrate error could not retrieve current version: " & $error)
|
|
|
|
|
|
|
|
if currentVersion == targetVersion:
|
|
|
|
debug "database schema is up to date",
|
2024-03-16 00:08:47 +01:00
|
|
|
currentVersion = currentVersion, targetVersion = targetVersion
|
2024-03-01 12:05:27 +01:00
|
|
|
return ok()
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
info "database schema is outdated",
|
|
|
|
currentVersion = currentVersion, targetVersion = targetVersion
|
2024-03-01 12:05:27 +01:00
|
|
|
|
|
|
|
# Load migration scripts
|
|
|
|
let scripts = pg_migration_manager.getMigrationScripts(currentVersion, targetVersion)
|
|
|
|
|
|
|
|
# Run the migration scripts
|
|
|
|
for script in scripts:
|
|
|
|
for statement in script.breakIntoStatements():
|
2024-03-16 00:08:47 +01:00
|
|
|
debug "executing migration statement", statement = statement
|
2024-03-01 12:05:27 +01:00
|
|
|
|
|
|
|
(await driver.performWriteQuery(statement)).isOkOr:
|
2024-03-16 00:08:47 +01:00
|
|
|
error "failed to execute migration statement",
|
|
|
|
statement = statement, error = error
|
2024-03-01 12:05:27 +01:00
|
|
|
return err("failed to execute migration statement")
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
debug "migration statement executed succesfully", statement = statement
|
2024-03-01 12:05:27 +01:00
|
|
|
|
|
|
|
debug "finished message store's postgres database migration"
|
|
|
|
|
|
|
|
return ok()
|