dev.monosoul.jooq.migration.ReflectiveMigrationRunner.kt Maven / Gradle / Ivy
Show all versions of jooq-gradle-plugin Show documentation
package dev.monosoul.jooq.migration
import dev.monosoul.jooq.settings.DatabaseCredentials
import org.flywaydb.core.Flyway
import org.flywaydb.core.api.configuration.FluentConfiguration
import org.flywaydb.core.api.output.MigrateResult
import kotlin.reflect.KFunction1
import kotlin.reflect.KFunction2
import kotlin.reflect.KFunction4
import kotlin.reflect.jvm.jvmName
internal class ReflectiveMigrationRunner(codegenAwareClassLoader: ClassLoader) : MigrationRunner {
private val flyway = ReflectiveFlywayConfiguration(codegenAwareClassLoader)
override fun migrateDb(
schemas: Array,
migrationLocations: Array,
flywayProperties: Map,
credentials: DatabaseCredentials,
defaultFlywaySchema: String,
flywayTable: String,
) = flyway
.dataSource(credentials)
.schemas(*schemas)
.locations(*migrationLocations)
.defaultSchema(defaultFlywaySchema)
.table(flywayTable)
.configuration(flywayProperties)
.load()
.migrate()
/**
* Wrapper for Flyway configuration object obtained via reflection.
* @see FluentConfiguration
*/
private class ReflectiveFlywayConfiguration(private val codegenAwareClassLoader: ClassLoader) {
private val flywayClass = codegenAwareClassLoader.loadClass(Flyway::class.jvmName)
private val configurationClass = codegenAwareClassLoader.loadClass(FluentConfiguration::class.jvmName)
private val configureMethod =
flywayClass.getMethod(oneArgFunctionName(Flyway::configure), ClassLoader::class.java)
private val configuration = configureMethod.invoke(null, codegenAwareClassLoader)
/**
* FluentConfiguration.dataSource(String, String, String)
* @see FluentConfiguration.dataSource
*/
fun dataSource(credentials: DatabaseCredentials) =
also {
val dataSourceMethod =
configurationClass.getMethod(
fourArgFunctionName(FluentConfiguration::dataSource),
String::class.java,
String::class.java,
String::class.java,
)
dataSourceMethod.invoke(configuration, credentials.jdbcUrl, credentials.username, credentials.password)
}
/**
* FluentConfiguration.schemas(String...)
* @see FluentConfiguration.schemas
*/
fun schemas(vararg schemas: String) =
also {
val schemasMethod =
configurationClass.getMethod(
twoArgFunctionName(FluentConfiguration::schemas),
Array::class.java,
)
schemasMethod.invoke(configuration, schemas)
}
/**
* FluentConfiguration.locations(String...)
* @see FluentConfiguration.locations
*/
fun locations(vararg locations: String) =
also {
val locationsMethod =
configurationClass.getMethod(
twoArgFunctionName, FluentConfiguration>(
FluentConfiguration::locations,
),
Array::class.java,
)
locationsMethod.invoke(configuration, locations)
}
/**
* FluentConfiguration.defaultSchema(String)
* @see FluentConfiguration.defaultSchema
*/
fun defaultSchema(schema: String) =
also {
val defaultSchemaMethod =
configurationClass.getMethod(
twoArgFunctionName(FluentConfiguration::defaultSchema),
String::class.java,
)
defaultSchemaMethod.invoke(configuration, schema)
}
/**
* FluentConfiguration.table(String)
* @see FluentConfiguration.table
*/
fun table(table: String) =
also {
val tableMethod =
configurationClass.getMethod(
twoArgFunctionName(FluentConfiguration::table),
String::class.java,
)
tableMethod.invoke(configuration, table)
}
/**
*
* FluentConfiguration.configuration(Map)
* @see FluentConfiguration.configuration
*/
fun configuration(props: Map) =
also {
val configurationMethod =
configurationClass.getMethod(
twoArgFunctionName, FluentConfiguration>(
FluentConfiguration::configuration,
),
Map::class.java,
)
configurationMethod.invoke(configuration, props)
}
/**
* FluentConfiguration.load()
* @see FluentConfiguration.load
*/
fun load(): ReflectiveFlyway {
val loadMethod = configurationClass.getMethod(FluentConfiguration::load.name)
return ReflectiveFlyway(
flywayClass = flywayClass,
migrateResultClass = codegenAwareClassLoader.loadClass(MigrateResult::class.jvmName),
flywayInstance = loadMethod.invoke(configuration),
)
}
private companion object {
// those functions are required to provide a hint for overloaded functions
fun oneArgFunctionName(ref: KFunction1
) = ref.name
fun twoArgFunctionName(ref: KFunction2) = ref.name
fun fourArgFunctionName(ref: KFunction4) = ref.name
}
}
/**
* Wrapper for Flyway object obtained via reflection.
* @see Flyway
*/
private class ReflectiveFlyway(
private val flywayClass: Class<*>,
private val migrateResultClass: Class<*>,
private val flywayInstance: Any,
) {
/**
* Flyway.migrate()
* @see Flyway.migrate
* @see MigrateResult.targetSchemaVersion
*/
fun migrate(): SchemaVersion {
val migrateMethod = flywayClass.getMethod(Flyway::migrate.name)
val migrateResult = migrateMethod.invoke(flywayInstance)
val targetSchemaVersionProperty = migrateResultClass.getField(MigrateResult::targetSchemaVersion.name)
val targetSchemaVersion = targetSchemaVersionProperty.get(migrateResult) as String?
return SchemaVersion(targetSchemaVersion.toString())
}
}
}