RxJava可观察完成,但在单元测试中未触发doOnComplete回调

时间:2019-01-02 13:17:25

标签: unit-testing junit kotlin mockito rx-java2

我具有以下逻辑,该逻辑可从可完成的数据(使用doOnComplete回调)创建可观察的数据,并已对其进行了单元测试:

class ObservableSrc(val completableSrc: CompletableSrc) {

    fun getObservable(client: Client): Observable<State> {
        return client.getResults()
            .concatMap { processResult(client, it.values) }
    }

    private fun processResult(Client: Client, values: Values): Observable<State> =
        completableSrc.getCompletable(client.type, values)
            .doOnComplete { client.doSomething(values)}
            .toSingleDefault(…)
            .map { … }
            .toObservable()
            .startWith(State.InProgress)
}

@Test
fun test() {
    whenever(client.type).doReturn(Type.SOME_TYPE)
    whenever(client.getResults()).doReturn(Observable.just<Result>(Result(mock())))
    whenever(completableSrc.getCompletable(any(), any())).doReturn(Completable.complete())
    doNothing().whenever(client).doSomething(any())

    val observer = tested.getObservable(client).test()

    observer.assertComplete()
    verify(completableSrc, times(1)).getCompletable(any(), any())
    verify(client, times(1)).doSomething(any())
}

问题在于,验证doSomething可以与模拟client进行2次交互,但是没有调用所需的方法。我发现,如果我改变链条有点像这样:

completableSrc.getCompletable(client.type, values)
                .toSingleDefault(…)
                .map { … }
                .doOnSuccess { client.doSomething(values)}
                .toObservable()
                .startWith(State.InProgress)

然后它起作用。我只是不知道为什么带有doOnComplete的先前版本会失败(observer.assertComplete()成功执行)。似乎处理得太早了,所以没有调用回调,为什么?

1 个答案:

答案 0 :(得分:2)

更新

使用单元测试和模拟来测试代码。我将 kolin.test mockk 用于个人喜好。

interface CompletableSrc {
    fun getCompletable(): Completable
}

interface Client {
    fun doSomething()
    fun doSomethingElse()
}

class CompletableTest {

    @Test
    fun `functions doOnComplete and doOnSuccess should work as expected`() {
        val completableSrc: CompletableSrc = mockk {
            every { getCompletable() } returns Completable.complete()
        }

        val client: Client = mockk {
            every { doSomething() } returns Unit
            every { doSomethingElse() } returns Unit
        }

        val observable = completableSrc.getCompletable()
            .doOnComplete { client.doSomething() }
            .toSingleDefault(0)
            .map { it + 1 }
            .doOnSuccess { client.doSomethingElse() }
            .toObservable()
            .startWith(-1)

        val test = observable.test()
        test.assertComplete()
        test.assertValues(-1, 1)

        verify(exactly = 1) {
            completableSrc.getCompletable()
            client.doSomething()
            client.doSomethingElse()
        }
    }
}

如您所见,断言和模拟验证均成功。

上一个

您的代码应该可以正常工作,对我来说真的很难分辨出什么问题是因为我无法运行它,但是我创建了一个片段,它的功能几乎相同,并且可以按预期工作。

示例

fun main(args: Array<String>) {
    var sideEffect = 0

    val observable = Completable.complete()
        .doOnComplete { sideEffect += 1 }
        .toSingleDefault(sideEffect)
        .map { sideEffect + 1 }
        .doOnSuccess(::println)
        .toObservable()
        .startWith(-1)

    val test = observable.test()
    test.assertComplete()
    test.assertValues(-1, 2)
}

输出

2

执行两个副作用函数Completable.doOnCompleteSingle.doOnSuccess,更新变量sideEffect并打印到控制台。