当前位置: 首页 > 知识库问答 >
问题:

viewModel和livedata Mockito android的单元测试

吴康平
2023-03-14
@Test
fun getNews() {

    Mockito.`when`(mainRepository.getNews()).thenReturn(Observable.just(getMockResponse()))
    //It is working good

    Mockito.`when`(networkHelper.isNetworkConnected()).thenReturn(true)
    //It is working good

    viewModel.getNews()

    assertEquals(Resource.success(getList()), viewModel.newsResponse.value)
}
 val newsResponse by lazy { SingleLiveEvent<Resource<List<RowsItem>>>() }

        fun getNews() {
        if (!networkHelper.isNetworkConnected()) { //It is true as we macked in Test Clas
            newsResponse.value = Resource.error(AppError.NetworkError)
            return
        }
        newsResponse.value = Resource.loading() // Crashing here when we are assigning value to live data: newsResponse
        val api = mainRepository.getNews().map {
            it.rows
        }

        val disposable = api.subscribeOn(Schedulers.computation())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(
                { result -> handleResults(result) },
                { error -> handleError(error) },
            )

        compositeDisposable.add(disposable)
    }


     private fun handleResults(list: ArrayList<RowsItem>?) {
         if (list != null) {
             newsResponse.value = Resource.success(list)
 
         } else {
             newsResponse.value = Resource.error(AppError.CommonError)
         }
     }

      private fun handleError(t: Throwable?) {
          newsResponse.value = Resource.error(t?.failureAppError() ?: AppError.CommonError)
      }

进程结束,退出代码-1

共有1个答案

史钊
2023-03-14

只需要在测试类中添加规则

@Rule
@JvmField
var testSchedulerRule = RxImmediateSchedulerRule() //For Retrofit

@get:Rule
var instantTaskExecutorRule = InstantTaskExecutorRule()//For LiveData

RximediateSchedulerRule是扩展TestRule的自定义类

class RxImmediateSchedulerRule : TestRule {

private val immediateScheduler = object : Scheduler() {
    @NonNull
    override fun scheduleDirect(@NonNull run: Runnable, delay: Long, @NonNull unit: TimeUnit): Disposable {
        // Hack to prevent stack overflows in unit tests when scheduling with a delay;
        return super.scheduleDirect(run, 0, unit)
    }

    @NonNull
    override fun createWorker(): Worker {
       return ExecutorScheduler.ExecutorWorker(Executor { it.run() })
    }
}

@NonNull
override fun apply(@NonNull base: Statement, @NonNull description: Description): Statement {
    return object : Statement() {
        @Throws(Throwable::class)
        override fun evaluate() {
            RxJavaPlugins.setInitIoSchedulerHandler { immediateScheduler }
            RxJavaPlugins.setInitComputationSchedulerHandler { immediateScheduler }
            RxJavaPlugins.setInitNewThreadSchedulerHandler { immediateScheduler }
            RxJavaPlugins.setInitSingleSchedulerHandler { immediateScheduler }
            RxAndroidPlugins.setInitMainThreadSchedulerHandler { immediateScheduler }

            try {
                base.evaluate()
            } finally {
                RxJavaPlugins.reset()
                RxAndroidPlugins.reset()
            }
        }
    }
}}

现在,在最后,我的测试类看起来像

 @RunWith(MockitoJUnitRunner::class)
 class MainActivityViewModelTest : TestCase() {


private val networkHelper = Mockito.mock(NetworkHelper::class.java)

private val mainRepository = Mockito.mock(MainRepository::class.java)

private val viewModel = MainActivityViewModel(networkHelper, mainRepository)

@Rule
@JvmField
var testSchedulerRule = RxImmediateSchedulerRule() //For Retrofit

@get:Rule
var instantTaskExecutorRule = InstantTaskExecutorRule()//For LiveData


@Before
public override fun setUp() {

}

@Test
fun getNews() {
    Mockito.`when`(mainRepository.getNews()).thenReturn(Observable.just(getMockResponse()))
    //It is working good
    Mockito.`when`(networkHelper.isNetworkConnected()).thenReturn(true)

    //It is working good
    viewModel.getNews()

    assertEquals(Resource.success(getList()), viewModel.newsResponse.value)


}

private fun getMockResponse(): ApiResponse<ArrayList<RowsItem>> {
    return ApiResponse("Response Success", getList())
}

private fun getList(): ArrayList<RowsItem> {
    val rowsItem = RowsItem("ImageHref", "Desc", "Title")
    val list = ArrayList<RowsItem>()
    list.add(rowsItem)
    return list
}}
 类似资料:
  • 我在ViewModel中有一个函数有两个状态,第一个状态总是加载,第二个状态取决于api或db交互的结果。 这是函数 但是我找不到测试状态是否发生的方法,所以我修改了现有的扩展函数,以 若要在LiveData更改时将数据添加到列表中,并在该列表中存储状态,但它从不返回加载状态,因为它发生在observe启动之前。是否有方法测试的多个值?

  • 单元测试 单元测试仅依赖于源代码,是测试代码逻辑是否符合预期的最简单方法。 运行所有的单元测试 make test 仅测试指定的package # 单个package make test WHAT=./pkg/api # 多个packages make test WHAT=./pkg/{api,kubelet} 或者,也可以直接用go test go test -v k8s.io/kubernet

  • 我的WPF-App的主窗口上有许多按钮。这些按钮的命令应该具有相同的实现/功能,但根据按下的按钮,函数访问的文件/路径会发生变化。如何在不使用按钮单击事件处理程序(Windows表单)的情况下检测从ViewModel单击了哪个按钮? 下面是类RelayCommand的实现: 以下是ViewModel中命令的代码:

  • 我开始在我的项目中实践TDD,作为背景,它也包含遗留代码。我们使用Mockito作为一个模拟框架,并遵循Spring MVC方法。 有时,类用许多不同的对象作为属性实现。这些服务中有一些简单的方法,例如。 将使用许多对象来完成其职责 更新并保存事务 推进业务流程 关闭其他挂起的操作 但是,在执行这些操作时,该方法需要调用不同的来获取和更新事务、获取业务流程ID、获取挂起的事务(并保存其更新)。这意

  • 问题内容: 有人与Jenkins CI一起使用过cFix(Visual Assert)吗?如何设置?如何通过Jenkins执行C ++项目测试(Testt Framework cFix {isualAssert)? 任何指针!谢谢 问题答案: 我写了一个小程序将cFix输出转换为XML。我将jenkins识别的最简单的Junit XML格式用于程序输出。

  • 问题内容: 谁能使用WebSockets(Socket.io)为Node.js提供坚如磐石,简单的单元测试? 我使用用于Node.js的socket.io,并查看了socket.io-client以在测试中建立与服务器的客户端连接。但是,我似乎缺少了一些东西。 在下面的示例中,“ worked …”从未被打印出来。 相反,我只是得到: 有什么建议? 问题答案: 经过进一步的摸索和探索,我发现了一些