gpt4 book ai didi

android - 如何对返回 livedata 的函数进行单元测试

转载 作者:行者123 更新时间:2023-12-03 23:47:01 26 4
gpt4 key购买 nike

在我的 viewModel 中,我有一个返回 liveData 的函数。该函数在 fragment 中直接调用,因此可以直接在那里观察到。我无法了解如何测试此函数,因为在测试的情况下未观察到该函数发出的 liveData,因此它不会返回该值。

这是我的功能,我想为以下内容编写测试:

    fun saveRating(rating: Float, eventName: String): LiveData<Response<SaveRatingData?>?> {
val request = RatingRequest(rating.toDouble(), eventName, false)

return liveData(viewModelScope.coroutineContext + Dispatchers.IO) {
emit(repository.saveRatings(request))
}

}

这就是我在 fragment 中调用它的方式:
   viewModel.saveRating(rating, npsEventData?.eventName ?: "").observe(this, Observer {
// on getting data
})

提前致谢!

最佳答案

您需要有一个 testCoroutineDispatcher 或 testCoroutineScope 才能将 viewModel 的范围设置为测试范围。

class TestCoroutineRule : TestRule {

private val testCoroutineDispatcher = TestCoroutineDispatcher()

val testCoroutineScope = TestCoroutineScope(testCoroutineDispatcher)

override fun apply(base: Statement, description: Description?) = object : Statement() {

@Throws(Throwable::class)
override fun evaluate() {

Dispatchers.setMain(testCoroutineDispatcher)

base.evaluate()

Dispatchers.resetMain()
try {
testCoroutineScope.cleanupTestCoroutines()
} catch (exception: Exception) {
exception.printStackTrace()
}
}
}

fun runBlockingTest(block: suspend TestCoroutineScope.() -> Unit) =
testCoroutineScope.runBlockingTest { block() }

}

在任何官方 kotlin 或 Android 文档中都没有提到 Try-catch block ,但是测试异常会导致异常,而不是像我在此 here 中所要求的那样通过测试。 .

我使用 testCoroutineDispatcher 作为调度程序的另一件事是不足以让某些测试通过,您需要将 coroutineScope 而不是调度程序注入(inject) viewModel。

例如
fun throwExceptionInAScope(coroutineContext: CoroutineContext) {


viewModelScope.launch(coroutineContext) {

delay(2000)
throw RuntimeException("Exception Occurred")
}
}

你有一个像这样抛出异常的函数,你将 testCoroutineContext 传递给这个失败的测试。
@Test(expected = RuntimeException::class)
fun `Test function that throws exception`() =
testCoroutineDispatcher.runBlockingTest {

// 🔥 Using testCoroutineDispatcher causes this test to FAIL
viewModel.throwExceptionInAScope(testCoroutineDispatcher.coroutineContext)

// 🔥 This one passes since we use context of current coroutineScope
viewModel.throwExceptionInAScope(this.coroutineContext)

}

如果你使用类 MyViewModel(private val coroutineScope: CoroutineScope),它会通过

现在,让我们来看看如何使用异步任务测试 liveData。我用这个类,谷歌的 LiveDataTestUtil类,用于同步 liveData
@get:Rule
var instantTaskExecutorRule = InstantTaskExecutorRule()

作为规则
fun <T> LiveData<T>.getOrAwaitValue(
time: Long = 2,
timeUnit: TimeUnit = TimeUnit.SECONDS,
afterObserve: () -> Unit = {}
): T {
var data: T? = null
val latch = CountDownLatch(1)
val observer = object : Observer<T> {
override fun onChanged(o: T?) {
data = o
latch.countDown()
this@getOrAwaitValue.removeObserver(this)
}
}
this.observeForever(observer)

afterObserve.invoke()

// Don't wait indefinitely if the LiveData is not set.
if (!latch.await(time, timeUnit)) {
this.removeObserver(observer)
throw TimeoutException("LiveData value was never set.")
}

@Suppress("UNCHECKED_CAST")
return data as T
}

/**
* Observes a [LiveData] until the `block` is done executing.
*/
fun <T> LiveData<T>.observeForTesting(block: () -> Unit) {
val observer = Observer<T> { }
try {
observeForever(observer)
block()
} finally {
removeObserver(observer)
}
}

现在,您可以像测试同步代码一样测试它
@Test
fun `Given repo saves response, it should return the correct one` = testCoroutineScope.runBlockingTest {

// GIVEN
val repository = mockk<<Repository>()
val actual = Response(...)
coEvery { repository.saveRatings } returns actual

// WHEN
val expected = viewModel.saveResponse()

// THEN
Truth.assertThat(actual).isEqualTo(expected)

}

我使用了 mockK,它适用于暂停模拟。

你也不需要使用 Dispatchers.IO如果您有改造或房间功能调用,如果您不执行改造或房间操作以外的其他任务,它们会使用自己的带有挂起修饰符的线程。

关于android - 如何对返回 livedata 的函数进行单元测试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62113645/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com