java .lang.IllegalStateException: отмена звонка не может произойти без возможного запуска - PullRequest
3 голосов
/ 27 апреля 2020

Я пытаюсь протестировать простое ViewModel с Robolectric.

Вот мой ViewModel

GreetingsViewModel.kt

@FlowPreview
@ExperimentalCoroutinesApi
class GreetingsViewModel : ViewModel() {


    private val greetingsChannel = ConflatedBroadcastChannel<String>()

    /**
     * Whenever greetingsChannel offers a value, prepend `Hello ` with it and return.
     */
    val greetingsResponse = greetingsChannel.asFlow()
        .flatMapLatest { name ->
            flowOf("Helo $name")
        }.asLiveData(viewModelScope.coroutineContext)

    /**
     * To get new greetings
     */
    fun askGreetings(name: String) {
        greetingsChannel.offer(name)
    }
}

и вот мой файл модульного теста

GreetingsViewModelTest.kt

@FlowPreview
@ExperimentalCoroutinesApi
@RunWith(AndroidJUnit4::class)
class GreetingsViewModelTest2 {

    @Test
    fun greeting_askGreetings_getWithName() {
        val testViewModel = GreetingsViewModel()
        testViewModel.askGreetings("john")
        testViewModel.greetingsResponse.getOrAwaitValue().should.equal("Hello john")
    }
}

Но при запуске теста появляется ошибка ниже.

java.lang.IllegalStateException: Cancel call cannot happen without a maybeRun

    at androidx.lifecycle.BlockRunner.cancel(CoroutineLiveData.kt:184)
    at androidx.lifecycle.CoroutineLiveData.onInactive(CoroutineLiveData.kt:245)
    at androidx.lifecycle.LiveData$ObserverWrapper.activeStateChanged(LiveData.java:440)
    at androidx.lifecycle.LiveData.observeForever(LiveData.java:232)
    at com.theapache64.twinkill.test.GetOrAwaitValueKt.getOrAwaitValue(getOrAwaitValue.kt:26)
    at com.theapache64.twinkill.test.GetOrAwaitValueKt.getOrAwaitValue$default(getOrAwaitValue.kt:15)
    at com.theapache64.tracktor.ui.activities.test.GreetingsViewModelTest.greeting_askGreetings_getWithName(GreetingsViewModelTest.kt:23)
    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.runners.ParentRunner$3.evaluate(ParentRunner.java:306)
    at org.robolectric.RobolectricTestRunner$HelperTestRunner$1.evaluate(RobolectricTestRunner.java:546)
    at org.robolectric.internal.SandboxTestRunner$2.lambda$evaluate$0(SandboxTestRunner.java:252)
    at org.robolectric.internal.bytecode.Sandbox.lambda$runOnMainThread$0(Sandbox.java:89)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)

Есть идеи почему?

ПРИМЕЧАНИЕ: getOrAwaitValue () - это функция расширения, которая получает значение LiveData или ожидает его получения с тайм-аутом.

Ответы [ 2 ]

1 голос
/ 09 мая 2020

Я столкнулся с той же проблемой. Когда @Rahul Singhal указывал, причиной ошибки была эта строка

    this@getOrAwaitValue.removeObserver(this)

Поэтому я решил преобразовать LiveData обратно в поток и затем получить значение. Это расширение, которое я использую для получения значения.

    suspend fun <T> LiveData<T>.getFlowAsLiveDataValue(timeMillis: Long = 2_000): T? =
        withTimeoutOrNull(timeMillis) {
        this@getFlowAsLiveDataValue.asFlow().first()
    }

Ваш тест должен выглядеть следующим образом:

    testViewModel.greetingsResponse.getFlowAsLiveDataValue().should.equal("Hello john")
0 голосов
/ 01 мая 2020

Попробуйте код ниже

class GreetingsViewModelTest {

@get:Rule
var instantExecutorRule = InstantTaskExecutorRule()


@Test
fun greeting_askGreetings_getWithName() {
    kotlinx.coroutines.Dispatchers.setMain(TestCoroutineDispatcher())
    val testViewModel = GreetingsViewModel()
    testViewModel.askGreetings("john")
    var name: String? = null
    testViewModel.greetingsResponse.observeForever {
        name = it
    }
    assertEquals("Helo john", name)
}

}

Или удалите this@getOrAwaitValue.removeObserver(this)

в обоих направлениях

`

...