Тестирование сопрограмм и LiveData: результат сопрограммы не отражается в LiveData - PullRequest
1 голос
/ 24 октября 2019

Я новичок в тестировании, и я хотел научиться тестировать сопрограммы с шаблоном MVVM. Я просто следил за проектом https://github.com/android/architecture-samples и сделал несколько изменений (удалил удаленный источник). Но при тестировании ViewModel на выборку данных из хранилища он продолжает терпеть неудачу с этой ошибкой.

value of    : iterable.size()
expected    : 3
but was     : 0
iterable was: []
Expected :3
Actual   :0

Ниже приведен мой тестовый класс для ViewModel, не знаю, что я пропускаю. Также при издевательстве над хранилищем я могу получить ожидаемые результаты от него при печати taskRepository.getTasks(), он просто не отражается на LiveData при вызове loadTasks()

ViewModelTest

@ExperimentalCoroutinesApi
@RunWith(MockitoJUnitRunner::class)
class TasksViewModelTest {

    private lateinit var tasksViewModel: TasksViewModel

    val tasksRepository = mock(TasksRepository::class.java)

    @ExperimentalCoroutinesApi
    @get:Rule
    var mainCoroutineRule = TestMainCoroutineRule()

    // Executes each task synchronously using Architecture Components.
    @get:Rule
    var instantExecutorRule = InstantTaskExecutorRule()

    @Before
    fun setupViewModel() {
        tasksViewModel = TasksViewModel(tasksRepository)
    }

    @Test
    fun whenLoading_hasListOfTasks() = runBlockingTest {
        val task1 = Task("title1", "description1")
        val task2 = Task("title2", "description2")
        val task3 = Task("title3", "description3")
        `when`(tasksRepository.getTasks()).thenReturn(Result.Success(listOf(
           task1,
           task2,
           task3
        )))

        tasksViewModel.loadTasks()

        val tasks = LiveDataTestUtil.getValue(tasksViewModel.tasks)
        assertThat(tasks).hasSize(3)
    }
}

TasksViewModel

class TasksViewModel @Inject constructor(
  private val repository: TasksRepository
) : ViewModel() {

  private val _tasks = MutableLiveData<List<Task>>().apply { value = emptyList() }
  val tasks: LiveData<List<Task>> = _tasks

  fun loadTasks() {
    viewModelScope.launch {
      val tasksResult = repository.getTasks()

      if (tasksResult is Success) {
        val tasks = tasksResult.data
        _tasks.value = ArrayList(tasks)
      }
    }
  }
}

Вспомогательные классы перечислены ниже, я просто скопировал те же классы из примера проекта.

LiveDataTestUtil

object LiveDataTestUtil {

    /**
     * Get the value from a LiveData object. We're waiting for LiveData to emit, for 2 seconds.
     * Once we got a notification via onChanged, we stop observing.
     */
    fun <T> getValue(liveData: LiveData<T>): T {
        val data = arrayOfNulls<Any>(1)
        val latch = CountDownLatch(1)
        val observer = object : Observer<T> {
            override fun onChanged(o: T?) {
                data[0] = o
                latch.countDown()
                liveData.removeObserver(this)
            }
        }
        liveData.observeForever(observer)
        latch.await(2, TimeUnit.SECONDS)

        @Suppress("UNCHECKED_CAST")
        return data[0] as T
    }
}

MainCoroutineRule

@ExperimentalCoroutinesApi
class TestMainCoroutineRule : TestWatcher(), TestCoroutineScope by TestCoroutineScope() {

    override fun starting(description: Description?) {
        super.starting(description)
        Dispatchers.setMain(this.coroutineContext[ContinuationInterceptor] as CoroutineDispatcher)
    }

    override fun finished(description: Description?) {
        super.finished(description)
        Dispatchers.resetMain()
    }
}

1 Ответ

0 голосов
/ 24 октября 2019

Оказывается, это была проблема с mockito, у меня есть более старая версия, и я обнаружил, что есть библиотека под названием mockito-kotlin для упрощения тестирования сопрограмм, как указано здесь . Затем я подправил свой код к этому, и он работает хорошо.

tasksRepository.stub {
    onBlocking { getTasks() }.doReturn(Result.Success(listOf(task1, task2, task3)))
}
...