Тестирование ViewModel с помощью Rx и GIPHY

Я полный новичок в написании модульных тестов и пытаюсь понять, как протестировать ViewModel, использующий Giphy API.

Это моя ViewModel:


import androidx.hilt.lifecycle.ViewModelInject
import androidx.lifecycle.ViewModel
import com.example.android.myproject.api.Gif
import com.example.android.myproject.api.Result
import com.example.android.myproject.repository.GifRepository
import com.giphy.sdk.core.models.Media
import com.giphy.sdk.core.models.enums.MediaType
import com.giphy.sdk.ui.pagination.GPHContent
import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers
import io.reactivex.rxjava3.core.Observable
import io.reactivex.rxjava3.schedulers.Schedulers
import io.reactivex.rxjava3.subjects.BehaviorSubject
import java.util.concurrent.TimeUnit

class GifViewModel @ViewModelInject constructor(
    private val gifRepository: GifRepository
) : ViewModel() {

    private val isSearchingBehaviorSubject = BehaviorSubject.createDefault(false)
    val isSearchingObservable: Observable<Boolean> =
        isSearchingBehaviorSubject.observeOn(AndroidSchedulers.mainThread())

    private val searchResultsBehaviorSubject = BehaviorSubject.create<GPHContent>()
    val searchResultsObservable: Observable<GPHContent> =
        searchResultsBehaviorSubject
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())

    private lateinit var selectedGif: Media

    val randomGifObservable: Observable<Result<Gif, String>> =
        Observable.interval(0, 10, TimeUnit.SECONDS, Schedulers.io())
            .flatMap<Result<Gif, String>> {
                gifRepository.getRandomGif()
                    .toObservable()
                    .map { Result.Success(it) }
            }
            .onErrorReturn { Result.Failure("Error getting a random GIF.") }
            .observeOn(AndroidSchedulers.mainThread())

    fun searchStateChanged(isSearching: Boolean) {
        isSearchingBehaviorSubject.onNext(isSearching)

        // this is needed to "reset" the results from a previous search
        if (isSearching)
            searchResultsBehaviorSubject.onNext(GPHContent.trendingGifs)
    }

    fun gifSearchQueryChanged(searchQuery: String) {
        val results = if (searchQuery.length >= 2)
            GPHContent.searchQuery(search = searchQuery, mediaType = MediaType.gif)
        else
            GPHContent.trendingGifs

        searchResultsBehaviorSubject.onNext(results)
    }

    fun gifSelected(media: Media) {
        selectedGif = media
    }

    fun selectedGif(): Media {
        return selectedGif
    }
}

Это мой тест:


import androidx.arch.core.executor.testing.InstantTaskExecutorRule
import com.example.android.myproject.RxImmediateSchedulerRule
import com.example.android.myproject.repository.GifRepository
import com.example.android.myproject.ui.main.GifViewModel
import org.junit.Assert.assertFalse
import org.junit.Assert.assertTrue
import org.junit.Before
import org.junit.Rule
import org.junit.Test
import org.junit.runner.RunWith
import org.mockito.Mock
import org.mockito.Mockito.spy
import org.mockito.MockitoAnnotations
import org.mockito.junit.MockitoJUnitRunner

@RunWith(MockitoJUnitRunner::class)
class GifViewModelTest {

    @Rule
    @JvmField
    var testSchedulerRule = RxImmediateSchedulerRule()

    @Rule
    @JvmField
    val instantTaskExecutorRule = InstantTaskExecutorRule()

    private lateinit var viewModel: GifViewModel

    @Mock
    private lateinit var gifRepository: GifRepository

    @Before
    fun setUp() {
        MockitoAnnotations.initMocks(this)

        viewModel = spy(GifViewModel(gifRepository))
    }

    @Test
    fun `initial isSearching value is false`() {
        val isSearching = viewModel.isSearchingObservable.blockingFirst()
        assertFalse(isSearching)
    }

    @Test
    fun `changing search state to true makes isSearching true`() {
        viewModel.searchStateChanged(isSearching = true)

        val isSearching = viewModel.isSearchingObservable.blockingSingle()

        assertTrue(isSearching)
    }
}

Это RxImmediateSchedulerRule:

import io.reactivex.rxjava3.android.plugins.RxAndroidPlugins
import io.reactivex.rxjava3.plugins.RxJavaPlugins
import io.reactivex.rxjava3.schedulers.Schedulers
import org.junit.rules.TestRule
import org.junit.runner.Description
import org.junit.runners.model.Statement

class RxImmediateSchedulerRule : TestRule {
    override fun apply(base: Statement, description: Description?): Statement {
        return object : Statement() {
            @Throws(Throwable::class)
            override fun evaluate() {
                RxJavaPlugins.setIoSchedulerHandler { Schedulers.trampoline() }
                RxJavaPlugins.setComputationSchedulerHandler { Schedulers.trampoline() }
                RxJavaPlugins.setNewThreadSchedulerHandler { Schedulers.trampoline() }
                RxJavaPlugins.setSingleSchedulerHandler { Schedulers.trampoline() }
                RxAndroidPlugins.setInitMainThreadSchedulerHandler { Schedulers.trampoline() }

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

Первый тест пройден. Когда я запускаю второй тест, я получаю следующее:

java.lang.ExceptionInInitializerError
    at com.example.android.myproject.ui.main.GifViewModel.searchStateChanged(GifViewModel.kt:44)
    at com.example.android.myproject.viewmodel.GifViewModelTest.changing search state to true makes isSearching true(GifViewModelTest.kt:49)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:59)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:56)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
    at org.mockito.internal.runners.DefaultInternalRunner$1$1.evaluate(DefaultInternalRunner.java:46)
    at org.junit.rules.TestWatcher$1.evaluate(TestWatcher.java:61)
    at com.example.android.myproject.RxImmediateSchedulerRule$apply$1.evaluate(RxImmediateSchedulerRule.kt:22)
    at org.junit.runners.ParentRunner$3.evaluate(ParentRunner.java:306)
    at org.junit.runners.BlockJUnit4ClassRunner$1.evaluate(BlockJUnit4ClassRunner.java:100)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:366)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:103)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:63)
    at org.junit.runners.ParentRunner$4.run(ParentRunner.java:331)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:79)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:329)
    at org.junit.runners.ParentRunner.access$100(ParentRunner.java:66)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:293)
    at org.junit.runners.ParentRunner$3.evaluate(ParentRunner.java:306)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:413)
    at org.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:77)
    at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:83)
    at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
    at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:33)
    at com.intellij.rt.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:230)
    at com.intellij.rt.junit.JUnitStarter.main(JUnitStarter.java:58)
Caused by: kotlin.UninitializedPropertyAccessException: lateinit property apiClient has not been initialized
    at com.giphy.sdk.core.a.b()
    at com.giphy.sdk.ui.pagination.GPHContent.<init>()
    at com.giphy.sdk.ui.pagination.GPHContent.<clinit>()
    at com.example.android.myproject.ui.main.GifViewModel.searchStateChanged(GifViewModel.kt:48)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.mockito.internal.creation.bytebuddy.MockMethodAdvice.tryInvoke(MockMethodAdvice.java:213)
    at org.mockito.internal.creation.bytebuddy.MockMethodAdvice.access$400(MockMethodAdvice.java:35)
    at org.mockito.internal.creation.bytebuddy.MockMethodAdvice$RealMethodCall.invoke(MockMethodAdvice.java:165)
    at org.mockito.internal.invocation.InterceptedInvocation.callRealMethod(InterceptedInvocation.java:152)
    at org.mockito.internal.stubbing.answers.CallsRealMethods.answer(CallsRealMethods.java:44)
    at org.mockito.Answers.answer(Answers.java:100)
    at org.mockito.internal.handler.MockHandlerImpl.handle(MockHandlerImpl.java:103)
    at org.mockito.internal.handler.NullResultGuardian.handle(NullResultGuardian.java:29)
    at org.mockito.internal.handler.InvocationNotifierHandler.handle(InvocationNotifierHandler.java:35)
    at org.mockito.internal.creation.bytebuddy.MockMethodInterceptor.doIntercept(MockMethodInterceptor.java:61)
    at org.mockito.internal.creation.bytebuddy.MockMethodAdvice.handle(MockMethodAdvice.java:106)
    ... 35 more

apiClient - это свойство в GPHContent, которое, по-видимому, не инициализировано во время запуска теста, я не уверен, как решить эту проблему. Разве нельзя протестировать мою модель просмотра в том виде, в каком она есть сейчас? Что мне нужно изменить, чтобы можно было запускать на нем тесты? Буду признателен за любое руководство или совет.

0 ответов

Другие вопросы по тегам