使用 Koin 的 KotlinTest:InvocationTargetException

KotlinTest with Koin: InvocationTargetException

我无法将 Koin 2.0.1 与 Kotlin-test 3.4.2 一起使用。我得到这样的 InvocationTargetException:

Running koinexample.KoinSampleTests
Tests run: 1, Failures: 0, Errors: 1, Skipped: 0, Time elapsed: 0.009 sec <<< FAILURE! - in koinexample.KoinSampleTests
koinexample.KoinSampleTests  Time elapsed: 0.009 sec  <<< ERROR!
java.lang.reflect.InvocationTargetException
        at koinexample.KoinSampleTests.getKoin(KoinSampleTests.kt:26)
        at koinexample.KoinSampleTests.<init>(KoinSampleTests.kt:61)

我在 GitHub 上创建了一个重现此错误的小示例: https://github.com/elifarley/kotlin-tests-with-koin-examples

只需执行这些命令来克隆 repo 和 运行 测试:

git clone https://github.com/elifarley/kotlin-tests-with-koin-examples.git
cd kotlin-tests-with-koin-examples
mvn

这是主要的 Kotlin 文件:

package koinexample

import io.kotlintest.koin.KoinListener
import io.kotlintest.shouldBe
import io.kotlintest.specs.FreeSpec
import org.koin.core.inject
import org.koin.dsl.module
import org.koin.test.KoinTest

data class Stats(var ok: Long = 0, var error: Long = 0)

interface StatsServer {
    fun newError(): Long
}

class StatsServerSimple(private val stats: Stats) : StatsServer {
    override fun newError() = stats.error++
}

val appModule = module {
    single { Stats() }
    single { StatsServerSimple(get()) as StatsServer }
}

class KoinSampleTests : FreeSpec(), KoinTest {

    private val modules = listOf(
        appModule
    )

    override fun listeners() = listOf(KoinListener(modules))

    val statsServer: StatsServer by inject()

    init {

        "Happy path" {
            statsServer.newError() shouldBe 1
        }
    }
}

您似乎从未启动过 Koin 应用程序。你需要

startKoin {
            modules(appModule)
        }

在您的测试方法中或在规范的 beforeSpec/beforeTest 函数调用中。

或使用类似此处的内容:

    override fun listeners() = listOf(KoinListener(appModule))

kotlintest/kotest 的文档中对此进行了描述:https://github.com/Kotest/Kotest/blob/master/doc/extensions.md#koin

您的问题似乎是简单的导入混淆。

请注意,您使用的是 import org.koin.core.inject,即此函数:

inline fun <reified T> KoinComponent.inject(
        qualifier: Qualifier? = null,
        noinline parameters: ParametersDefinition? = null
): Lazy<T> =
        getKoin().inject(qualifier, parameters)

它需要 getKoin 才能工作,因此您无法初始化测试(测试 class 在 Koin 有机会从侦听器开始之前就已初始化)。

正确的导入是import org.koin.test.inject,转换为:

inline fun <reified T> KoinTest.inject(
    qualifier: Qualifier? = null,
    noinline parameters: ParametersDefinition? = null
): Lazy<T> = lazy { get<T>(qualifier, parameters) }

请注意,这确实是惰性的,因此 Kotest 将有机会在您的测试开始之前初始化 Koin。

修复导入应该可以解决这个问题