All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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