在 kotlin 中,如何模拟包装回调的挂起函数?

Mir*_*tor 10 mocking callback mockito kotlin kotlinx.coroutines

假设有一个带有回调的接口:

interface SomeInterface {
    fun doSomething(arg: String, callback: (Exception?, Long) -> Unit)
}
Run Code Online (Sandbox Code Playgroud)

我将其扩展为这样的挂起函数:

suspend fun SomeInterface.doSomething(arg: String): Long = suspendCoroutine { cont ->
    this.doSomething(arg) { err, result ->
        if (err == null) {
            cont.resume(result)
        } else {
            cont.resumeWithException(err)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我想在测试中模拟这个,但失败了。理想情况下,我想使用这样的东西:

@Test
fun checkService() {
    runBlocking {
        val myService = mock<SomeInterface>()
        whenever(myService.doSomething(anyString())).thenReturn(1234L)
        val result = myService.doSomething("")
        assertEquals(result, 1234L)
    }
}
Run Code Online (Sandbox Code Playgroud)

上述语法因模拟异常而失败,因为它期望回调的匹配器。

org.mockito.exceptions.misusing.InvalidUseOfMatchersException: 
Invalid use of argument matchers!
2 matchers expected, 1 recorded:
Run Code Online (Sandbox Code Playgroud)

我怎样才能模拟这样的挂起功能?如果无法使用类似的语法,我如何才能使用所需的参数进行模拟回调,以便在整个代码中使用的挂起变体在测试期间返回所需的结果?

更新:当它是一个扩展功能时,它似乎是不可能的。根据Marko Topolnik的评论,我认为这是因为扩展只是一个静态函数,它超出了 mockito 的能力。

当 suspend 函数是一个成员函数时,它会按我原来的语法按预期工作。

这是一个带有一些演示代码的要点:https : //gist.github.com/mirceanis/716bf019a47826564fa57a77065f2335

Vah*_*iri 22

当你需要时

上..一定要回来..

并且方法被挂起,使用mockito你可以使用这个:

  • 使用这个库mockito-kotlin
  • 模拟您的对象,将其命名为 myObject(myObject 有名为 isFoo 的挂起方法)

然后:

 myObject.stub {
    onBlocking { isFoo() }.doReturn(true)
}
Run Code Online (Sandbox Code Playgroud)


Adi*_*rzi 11

我建议使用MockK进行测试,它对协程更友好。

要模拟协程,您可以使用coEveryreturns如下所示:

val interf = mockk<SomeInterface>()
coEvery { a.doSomething(any()) } returns Outcome.OK
Run Code Online (Sandbox Code Playgroud)