使用Completable.timer对RxJava进行单元测试

时间:2019-11-11 15:28:47

标签: android kotlin rx-java2

Android Studio 3.5.2
RxJava2
Kotlin 1.3.50

我有下面的类像一个计时器。

class DelayTimerImp(private val scheduler: IScheduler)
    : DelayTimer {

    override fun createTimeout(delay: Long, timeUnites: TimeUnit): Completable {
        return Completable.timer(delay, TimeUnit.SECONDS, scheduler.main())
    }
}

我正在这样的方法中使用它:

private fun startTimeoutRequest(delay: Long) {
    compositeSubscription.add(delayTimer.createTimeout(delay, TimeUnit.SECONDS)
            .subscribeOn(schedulers.io())
            .subscribe(::timeoutHasExceeded, ::timeoutError))
}

在我的测试类中,它使用TestScheduler返回io线程。我在嘲笑delayTimer。即使我thenReturn(Completable.complete()) timeOutHasExceeded中的方法也永远不会进入它们。我在想,如果Completable已完成,请使用该方法。

@Test
fun `should reload and show loading state`() {
    // Arrange
    whenever(delayTimer.createTimeout(10, TimeUnit.SECONDS))
            .thenReturn(Completable.complete())

    // Act
    vipInformationPresenterImp.tryAgainTapped()

    // Assert methods in the timeOutHasExceeded
}

另外,如果我通过将延迟更改为1来更改以下内容:

whenever(delayTimer.createTimeout(1, TimeUnit.SECONDS))
                .thenReturn(Completable.complete())

我将在下一行得到crash

.subscribeOn(schedulers.io())

对此进行单元测试的最佳方法是什么?

非常感谢

1 个答案:

答案 0 :(得分:7)

根据this tutorial,您应该使用TestScheduler覆盖它们:

class TestSchedulerProvider(private val scheduler: TestScheduler) : BaseSchedulerProvider {
    override fun computation() = scheduler
    override fun ui() = scheduler
    override fun io() = scheduler
}

随后,本教程将按以下方式使用TestScheduler:

@Test
fun delayTestExample() {
    //given
    val presenter = DemoPresenter(testSchedulerProvider, view, service)
    given(service.getSomeRemoteData()).willReturn(Single.just(5))
    val delayInMillis = 1000L

    //when
    presenter.getSomeDataWithDelay(delayInMillis)

    //then
    then(view).should(never()).showData(anyInt())
    // HERE:
    testScheduler.advanceTimeBy(delayInMillis, TimeUnit.MILLISECONDS) 
    then(view).should().showData(5)
}

在代码中的注释advanceTimeBy下描述了//HERE:的使用。是否可以使用相同的方法来代替您:

whenever(delayTimer.createTimeout(1, TimeUnit.SECONDS))
            .thenReturn(Completable.complete())

您仅举个例子:

testScheduler.advanceTimeBy(1500, TimeUnit.MILLISECONDS)
在创建超时之后的那一行

中,我选择了1500,以便在AdvanceTimeBy继续之前在计时器中真正滴答1秒。尽管每个值> 1000都应该很好用,我只是想确定一下;)