MockK模拟ViewModel的savedStateHandle.getLiveData()

Thr*_*ian 8 android android-livedata android-viewmodel mockk

我正在尝试使用 saveStateHandle 测试此 ViewModel

class PostListViewModel @ViewModelInject constructor(
    private val coroutineScope: CoroutineScope,
    private val getPostsUseCase: GetPostListUseCaseFlow,
    @Assisted savedStateHandle: SavedStateHandle
) : AbstractPostListVM() {

    private val _goToDetailScreen =
        savedStateHandle.getLiveData<Event<Post>>(POST_DETAIL)
//        MutableLiveData<Event<Post>>()

    override val goToDetailScreen: LiveData<Event<Post>>
        get() = _goToDetailScreen

    private val _postViewState =
        savedStateHandle.getLiveData<ViewState<List<Post>>>(POST_LIST)
//        MutableLiveData<ViewState<List<Post>>>()

    override val postViewState: LiveData<ViewState<List<Post>>>
        get() = _postViewState

    override fun getPosts() {

        getPostsUseCase.getPostFlowOfflineFirst()
            .convertToFlowViewState()
            .onStart {
                _postViewState.value = ViewState(status = Status.LOADING)
            }
            .onEach {
                _postViewState.value = it
            }
            .launchIn(coroutineScope)
    }

    override fun refreshPosts() {
        getPostsUseCase.getPostFlowOfflineLast()
            .convertToFlowViewState()
            .onStart {
                _postViewState.value = ViewState(status = Status.LOADING)
            }
            .onEach {
                _postViewState.value = it
            }
            .launchIn(coroutineScope)
    }

    override fun onClick(post: Post) {
        _goToDetailScreen.value = Event(post)
    }
}
Run Code Online (Sandbox Code Playgroud)

测试是

@Test
fun `given exception returned from useCase, should have ViewState ERROR offlineFirst`() =
    testCoroutineRule.runBlockingTest {

        // GIVEN
        every {
            savedStateHandle.getLiveData<ViewState<List<Post>>>(AbstractPostListVM.POST_LIST)
        } returns MutableLiveData()

        every {
            savedStateHandle.getLiveData<Event<Post>>(AbstractPostListVM.POST_DETAIL)
        } returns MutableLiveData()

        every {
            useCase.getPostFlowOfflineFirst()
        } returns flow<List<Post>> {
            emit(throw Exception("Network Exception"))
        }

        val testObserver = viewModel.postViewState.test()

        // WHEN
        viewModel.getPosts()

        // THEN
        testObserver
            .assertValue { states ->
                (states[0].status == Status.LOADING &&
                        states[1].status == Status.ERROR)
            }
            .dispose()

        val finalState = testObserver.values()[1]
        Truth.assertThat(finalState.error?.message).isEqualTo("Network Exception")
        Truth.assertThat(finalState.error).isInstanceOf(Exception::class.java)
        verify(atMost = 1) { useCase.getPostFlowOfflineFirst() }
    }
Run Code Online (Sandbox Code Playgroud)

MutableLiveData当我使用此测试而不是SavedStateHandle.getLiveData()与此测试和其他测试一起使用时,此测试有效。

saveStateHandle 被嘲笑为

private val savedStateHandle: SavedStateHandle = mockk()

回报

io.mockk.MockKException: no answer found for: SavedStateHandle(#2).getLiveData(POST_LIST)
Run Code Online (Sandbox Code Playgroud)

我将模拟的 saveStateHandle 更改为 withrelaxed = true

private val savedStateHandle: SavedStateHandle = mockk(relaxed = true)
Run Code Online (Sandbox Code Playgroud)

基于这个问题

现在 postViewState 的值不会改变,测试失败

java.lang.IndexOutOfBoundsException: Index: 0, Size: 0
Run Code Online (Sandbox Code Playgroud)

模拟和获取 LiveData 的正确方法是什么SavedStateHandle

jep*_*bio 11

也许是因为我使用junit5,所选的答案对我不起作用。

如果它对某人有帮助,我所做的是使用SavedStateHandle我需要的信息创建一个实例,而不是在测试的第一部分(在给定/安排部分)中模拟它,然后初始化 ViewModel:

@Test
fun `whatever`() =
    coroutineRule.testDispatcher.runBlockingTest {
        val savedStateHandle = SavedStateHandle().apply {
            set(MyViewModel.KEY, "something")
        }
        val viewModel = PostListViewModel(coroutineScope, getPostsUseCase, savedStateHandle)

        ...
}
Run Code Online (Sandbox Code Playgroud)

  • 这是理想的情况不需要模拟“SavedStateHandle”,您可以直接在实例本身上构​​造它并操作其数据(并验证其状态)。 (3认同)

Sid*_*mit 10

我已经找到了MockK Library 的解决方案,它工作正常,

我已经通过以下方式完成了

val savedStateHandle = mockk<SavedStateHandle>(relaxed = true)
Run Code Online (Sandbox Code Playgroud)

通过上面的行,我创建了savedStateHandle,我在ViewModel中使用了它

@Test
fun `test something simple`() {
    val savedStateHandle = mockk<SavedStateHandle>(relaxed = true)
    val viewModel = MyViewModel(savedStateHandle)
    verify { savedStateHandle.set(MyViewModel.KEY, "Something") }
}
Run Code Online (Sandbox Code Playgroud)

您还可以set用索引运算符替换该函数:

    verify { savedStateHandle[MyViewModel.KEY] = "Something" }
Run Code Online (Sandbox Code Playgroud)