我正在迁移一个使用Retrofit来使用协程的应用程序。该应用程序有一些UAT正在失败,因为Espresso没有等待协程完成并立即断言。
默认情况下,CoroutineCallAdapterFactory使用OkHttp的Dispatcher来执行异步请求,但Espresso只监控UI Thread和AsyncTaks的Thread池。我想到的一个解决方案是强制OkHttp的Dispatcher使用AsyncTask的ThreadPoolExecutor。
val dispatcher = Dispatcher(AsyncTask.THREAD_POOL_EXECUTOR as ExecutorService)
okHttpClientBuilder.dispatcher(dispatcher)这似乎起作用了,并且测试通过了。
这是个坏主意吗?注册IdlingResource仍然是更好的选择吗?
发布于 2019-05-30 11:26:39
如果没有更多关于设置的详细信息,我不确定这些信息是否会有帮助,但我会提到一些提示,它们可能会对您的测试有所帮助:
转换执行器
您也可以采用另一种方式,使用以下命令从任何执行器创建协程调度程序:
AsyncTask.THREAD_POOL_EXECUTOR.asCoroutineDispatcher()当然,这可以用在任何你有类似Dispatchers.Main的地方,这意味着你可以创建一个作用域,并从这个作用域启动你的协程,并且Espresso应该监控底层执行器池的完成。例如:
...
val espressoScope = CoroutineScope(AsyncTask.THREAD_POOL_EXECUTOR.asCoroutineDispatcher())
...
espressoScope.launch { api.getBooks() }类似地,您可以执行以下操作:
val asyncTaskContext = AsyncTask.THREAD_POOL_EXECUTOR.asCoroutineDispatcher()
withContext(asyncTaskContext) {
api.getBooks()
}
// OR:
@Test
fun someAndroidTest() = runBlocking(asyncTaskContext) {
// espresso logic
}加入作业(推荐)
最后但并非最不重要的一点是,您可以加入在测试中创建的任何作业,测试将等到作业完成后才会退出。这听起来像是在你的情况下最有帮助的方法,因为你真的只想等待协程完成:
@Test
fun `first book has a title`() = runBlocking {
launch {
// run a function that suspends and takes a while
val firstBook = api.getAllBooks().first()
assertNotNull(firstBook.title)
}.join()
}https://stackoverflow.com/questions/56333608
复制相似问题