commonMain.kr.jadekim.jext.koin.BaseKoinApplication.kt Maven / Gradle / Ivy
package kr.jadekim.jext.koin
import kr.jadekim.common.enumeration.Environment
import kr.jadekim.common.enumeration.IEnvironment
import kr.jadekim.common.util.parseArgument
import kr.jadekim.jext.koin.extension.Properties
import kr.jadekim.jext.koin.util.loadPropertiesFromArguments
import kr.jadekim.jext.koin.util.shutdownHook
import kr.jadekim.logger.JLog
import kr.jadekim.logger.JLogger
import kr.jadekim.logger.context.GlobalLogContext
import kr.jadekim.logger.integration.koin.KoinLogger
import kr.jadekim.logger.pipeline.LoggerNameShorter
import kr.jadekim.logger.pipeline.StdOutPrinter
import org.koin.core.Koin
import org.koin.core.context.KoinContext
import org.koin.core.module.Module
import org.koin.core.module.dsl.binds
import org.koin.core.module.dsl.withOptions
import org.koin.core.qualifier.StringQualifier
import org.koin.dsl.module
import org.koin.mp.KoinPlatformTools
val APP_PROPERTIES = StringQualifier("app-properties")
abstract class BaseKoinApplication(
private vararg val args: String,
applicationName: String? = null,
val environment: IEnvironment? = null,
val version: String? = null,
logger: JLogger? = null,
properties: Properties? = null,
val koinContext: KoinContext = KoinPlatformTools.defaultContext(),
) {
open class EnvironmentOption {
var enableLoggerNameShorter: Boolean = true
}
abstract val modules: List
val applicationName: String = applicationName ?: this::class.simpleName ?: "KoinApplication"
var isInitialized = false
private set
var properties: Properties = properties ?: Properties()
protected set
lateinit var environmentOption: EnvironmentOption
protected set
val isGlobal: Boolean = koinContext == KoinPlatformTools.defaultContext()
lateinit var koin: Koin
private set
protected val logger = logger ?: JLog.get(this.applicationName)
private var isStart = false
protected abstract fun startApplication()
protected abstract fun stopApplication()
fun init(arguments: Map> = parseArgument(*args)) {
logger.info {
meta = arguments
"Init $applicationName with ${arguments.size} arguments"
}
environmentOption = getEnvironmentOption(environment)
val propertiesMap = properties +
System.getenv() +
loadPropertiesFromArguments(arguments) +
System.getProperties().mapKeys { it.toString() }.mapValues { it.toString() }
properties = Properties(propertiesMap)
if (isGlobal) {
GlobalLogContext["applicationName"] = this.applicationName
GlobalLogContext["environment"] = environment?.name
GlobalLogContext["version"] = version
configureJLogger()
}
onInit()
koin = initContainer().koin
onInitializedContainer()
isInitialized = true
}
fun start() {
if (!isInitialized) {
init()
}
logger.info { "Start $applicationName" }
shutdownHook {
if (isStart) {
stop()
}
}
isStart = true
startApplication()
}
fun stop() {
isStart = false
stopApplication()
koinContext.stopKoin()
}
protected open fun configureJLogger() {
JLog.pipeline = mutableListOf()
if (environmentOption.enableLoggerNameShorter) {
JLog.installPipe(LoggerNameShorter())
}
JLog.installPipe(StdOutPrinter())
}
protected open fun onInit() {
//do nothing
}
protected open fun initContainer() = koinContext.startKoin {
logger(KoinLogger())
properties(properties)
modules(module {
single { environment } withOptions {
binds(listOf(Environment::class))
}
single(APP_PROPERTIES) { properties }
single { properties }
})
modules([email protected])
}
protected open fun onInitializedContainer() {
//do nothing
}
protected open fun getEnvironmentOption(environment: IEnvironment?): EnvironmentOption {
val env = environment as? Environment ?: return EnvironmentOption()
return when (env) {
Environment.LOCAL -> EnvironmentOption().apply {
enableLoggerNameShorter = false
}
else -> EnvironmentOption().apply {
enableLoggerNameShorter = true
}
}
}
private operator fun List?.plus(data: List?): List {
val result = mutableListOf()
this?.also { result.addAll(it) }
data?.also { result.addAll(it) }
return result
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy