web-dev-qa-db-ja.com

UnitTestのMutableLiveDataのsetValueとpostValue

アプリケーションでいくつかのメソッドをテストしようとしましたが、mutablelivedata.postValueを呼び出すとエラーが発生します。スニペットとエラーメッセージは次のとおりです。

@Test
public void callStartScreenRepository(){
    Observer<User> userObserver = mock(Observer.class);
    startScreenViewModel.returnUser().observeForever(userObserver);
    maennlich = new User();
    maennlich.setVorname("Christian");
    MutableLiveData<User> userTest = new MutableLiveData<>();
    userTest.postValue(maennlich);
    when(startScreenRepository.userWebService("testUser")).thenReturn(userTest);
    verify(userObserver).onChanged(maennlich);
}

Java.lang.RuntimeException:Android.os.LooperのメソッドgetMainLooperがモックされていません。詳細については、 http://g.co/androidstudio/not-mocked を参照してください。

at Android.os.Looper.getMainLooper(Looper.Java)
at Android.Arch.core.executor.DefaultTaskExecutor.postToMainThread(DefaultTaskExecutor.Java:48)
at Android.Arch.core.executor.AppToolkitTaskExecutor.postToMainThread(AppToolkitTaskExecutor.Java:100)
at Android.Arch.lifecycle.LiveData.postValue(LiveData.Java:277)
at Android.Arch.lifecycle.MutableLiveData.postValue(MutableLiveData.Java:28)
at com.example.cfrindt.foodtracking.ViewModels.StartScreenViewModelTest.callStartScreenRepository(StartScreenViewModelTest.Java:102)
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:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.Java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.Java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.Java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.Java:26)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.Java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.Java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.Java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.Java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.Java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.Java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.Java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.Java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.Java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.Java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.Java:117)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.Java:42)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.Java:262)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.Java:84)
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 com.intellij.rt.execution.application.AppMain.main(AppMain.Java:147)

SetValue()またはpostValue()のどちらを使用していても、このエラーが発生します。どうすればそれを機能させることができますか?

9
ECommerce

まず、InstantTaskExecutorRuleを使用する必要があるということです。つまり、次のことを意味します。

@Rule public InstantTaskExecutorRule instantExecutorRule = new InstantTaskExecutorRule();

これにより、MutableLiveDataを即座に操作できるようになります。アプリのbuild.gradleにインポートを含めることを忘れないでください。

testCompile "Android.Arch.core:core-testing:$rootProject.ext.Arch_version"

次に、RxSchedulersをオーバーライドするルールを使用してJUnitルールを定義するか、@ Before内でRxAndroidPlugins(RxJavaを使用している場合)を使用してスケジューラーをオーバーライドする必要があります。

RxSchedulersOverrideRule:

public class RxSchedulersOverrideRule implements TestRule {
    @Override public Statement apply(final Statement base, Description description) {
    return new Statement() {
        @Override public void evaluate() throws Throwable {
            RxAndroidPlugins.reset();
            RxAndroidPlugins.setMainThreadSchedulerHandler(scheduler -> Schedulers.trampoline());

        RxJavaPlugins.reset();
        RxJavaPlugins.setIoSchedulerHandler(scheduler -> Schedulers.trampoline());
        RxJavaPlugins.setComputationSchedulerHandler(scheduler -> Schedulers.trampoline());
        RxJavaPlugins.setNewThreadSchedulerHandler(scheduler -> Schedulers.trampoline());

        base.evaluate();

        RxAndroidPlugins.reset();
        RxJavaPlugins.reset();
      }
    };
  }
}

そしてそれをテストクラスで呼び出す:

@Rule public RxSchedulersOverrideRule rxSchedulersOverrideRule = new RxSchedulersOverrideRule();

もう1つのオプションは、@ Beforeのsetup()内で呼び出すことです。

RxAndroidPlugins.setInitMainThreadSchedulerHandler(schedulerCallable -> Schedulers.trampoline());

これを使用した後、@ AfterのtearDownClass()でリセットする必要があります。

RxAndroidPlugins.reset(); 
12
Carlos Daniel

オブザーバーがメインスレッドにないことを確認します

Jeroen Mols それぞれの概要 実装 JUnit4とJUnit5の場合Kotlin、JUnit 5、MockK、AssertJライブラリを使用したCoinverse OpenAppのローカルユニットテストにJunit5アプローチを使用しました。

JUnit 4

@Carlos Danielの solutionInstantExecutorRuleを実装することはJUnit4にとって正しいことです。

ExampleUnitTest.kt

class ExampleUnitTest {

    @get:Rule
    val rule = InstantTaskExecutorRule()

    @Test
    fun `my test`() {
        val mutableLiveData = MutableLiveData<String>()
        mutableLiveData.postValue("test")
        assertEquals("test", mutableLiveData.value)
    }
}

build.gradle

dependencies {
    testImplementation 'Android.Arch.core:core-testing:X.X.X'
}

JUnit 5

InstantExecutorExtension.kt

import androidx.Arch.core.executor.ArchTaskExecutor
import androidx.Arch.core.executor.TaskExecutor
import org.junit.jupiter.api.extension.AfterEachCallback
import org.junit.jupiter.api.extension.BeforeEachCallback
import org.junit.jupiter.api.extension.ExtensionContext

class InstantExecutorExtension : BeforeEachCallback, AfterEachCallback {
    override fun beforeEach(context: ExtensionContext?) {
        ArchTaskExecutor.getInstance().setDelegate(object : TaskExecutor() {
            override fun executeOnDiskIO(runnable: Runnable) = runnable.run()
            override fun postToMainThread(runnable: Runnable) = runnable.run()
            override fun isMainThread(): Boolean = true
        })
    }

    override fun afterEach(context: ExtensionContext?) {
        ArchTaskExecutor.getInstance().setDelegate(null)
    }
}

InstantExecutorExtensionは、ローカルユニットテストクラスにアノテーションを付けて実装されます。

ContentViewModelTest.kt

...
import androidx.paging.PagedList
import com.google.firebase.Timestamp
import io.mockk.*
import org.assertj.core.api.Assertions.assertThat
import org.junit.jupiter.api.AfterAll
import org.junit.jupiter.api.BeforeAll
import org.junit.jupiter.api.BeforeEach
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.extension.ExtendWith

@ExtendWith(InstantExecutorExtension::class)
class ContentViewModelTest {
    val timestamp = getTimeframe(DAY)

    @BeforeAll
    fun beforeAll() {
        mockkObject(ContentRepository)
    }

    @BeforeEach
    fun beforeEach() {
        clearAllMocks()
    }

    @AfterAll
    fun afterAll() {
        unmockkAll()
    }

    @Test
    fun `Feed Load`() {
        val content = Content("85", 0.0, Enums.ContentType.NONE, Timestamp.now(), "",
            "", "", "", "", "", "", MAIN,
            0, 0.0, 0.0, 0.0, 0.0,
            0.0, 0.0, 0.0, 0.0)
        every {
            getMainFeedList(any(), any())
        } returns MutableLiveData<Lce<ContentResult.PagedListResult>>().also { lce ->
            lce.value = Lce.Content(
                ContentResult.PagedListResult(
                    pagedList = MutableLiveData<PagedList<Content>>().apply {
                        this.value = listOf(content).asPagedList(
                            PagedList.Config.Builder().setEnablePlaceholders(false)
                                .setPrefetchDistance(24)
                                .setPageSize(12)
                                .build())
                        }, errorMessage = ""))
        }
        val contentViewModel = ContentViewModel(ContentRepository)
        contentViewModel.processEvent(ContentViewEvent.FeedLoad(MAIN, DAY, timestamp, false))
        assertThat(contentViewModel.feedViewState.getOrAwaitValue().contentList.getOrAwaitValue()[0])
            .isEqualTo(content)
        assertThat(contentViewModel.feedViewState.getOrAwaitValue().toolbar).isEqualTo(
            ToolbarState(
                    visibility = GONE,
                    titleRes = app_name,
                    isSupportActionBarEnabled = false))
        verify {
            getMainFeedList(any(), any())
        }
        confirmVerified(ContentRepository)
    }
}
1
Adam Hurwitz

@Carlos Danielの回答は私を大いに助けましたが、RulePowerMockRunnerに適用されなかったという別の問題に直面しました。同じ問題を抱えている人のために、ここに解決策があります:

@RunWith(PowerMockRunner::class)
@PowerMockRunnerDelegate(MockitoJUnitRunner::class) //this line allows you to use the powermock runner and mockito runner
@PrepareForTest(UnderTestClass::class)
class UnderTestClassTest {
@get:Rule
var instantExecutorRule = InstantTaskExecutorRule()
0
Rainmaker