我具有以下逻辑,该逻辑可从可完成的数据(使用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()
成功执行)。似乎处理得太早了,所以没有调用回调,为什么?
答案 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.doOnComplete
和Single.doOnSuccess
,更新变量sideEffect
并打印到控制台。