#kotlin #generics #mockk
#котлин #дженерики #mockk #kotlin #универсальные
Вопрос:
Я пытаюсь вернуть разные значения для разных параметров типа в моей универсальной функции, но по какой-то причине учитывается только один из макетов (последний, как кажется). Я попробовал четыре разных подхода, все они перечислены ниже.
Если вы знаете, что такое поведение может быть достигнуто с помощью Mockito или любого другого фреймворка Mockito, пожалуйста, дайте мне тоже знать
import io.mockk.MockKAnnotations
import io.mockk.every
import io.mockk.impl.annotations.MockK
import io.reactivex.Single
import org.junit.Before
import org.junit.Test
import retrofit2.Response
class MyTest {
class A
class B
class MyClass() {
fun <T> myMethod(callback: () -> Single<Response<T>>): Single<T> {
error("")
}
}
val a = A()
val b = B()
@MockK
lateinit var myClass: MyClass
@Before
fun setUp() {
MockKAnnotations.init(this)
}
@Test
fun tmp() {
every { myClass.myMethod<A>(any()) } returns (Single.just(a))
every { myClass.myMethod<B>(any()) } returns (Single.just(b))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp2() {
every { myClass.myMethod(any<() -> Single<Response<A>>>()) } returns (Single.just(a))
every { myClass.myMethod(any<() -> Single<Response<B>>>()) } returns (Single.just(b))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp3() {
every { myClass.myMethod(any<() -> Single<Response<A>>>()).hint(A::class) } returns (Single.just(a))
every { myClass.myMethod(any<() -> Single<Response<B>>>()).hint(B::class) } returns (Single.just(b))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp4() {
every { myClass.myMethod<A>(any()).hint(A::class) } returns (Single.just(a))
every { myClass.myMethod<B>(any()).hint(B::class) } returns (Single.just(b))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
}
Ошибка, которую показывает каждый из методов:
java.lang.AssertionError: Values at position 0 differ; expected: my.packagge.MyTest$A@130a0f66 (class: A) but was: my.packagge.MyTest$B@4c432866 (class: B) (latch = 0, values = 1, errors = 0, completions = 1)
Expected :my.packagge.MyTest$A@130a0f66 (class: A)
Actual :my.packagge.MyTest$B@4c432866 (class: B) (latch = 0, values = 1, errors = 0, completions = 1)
<Click to see difference>
at io.reactivex.observers.BaseTestConsumer.fail(BaseTestConsumer.java:189)
at io.reactivex.observers.BaseTestConsumer.assertValues(BaseTestConsumer.java:545)
at io.reactivex.observers.BaseTestConsumer.assertResult(BaseTestConsumer.java:796)
at my.packagge.MyTest.tmp(MyTest.kt:35)
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: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)
Заранее спасибо!
Комментарии:
1. Есть ли у вас какие-либо решения для этого? У меня тоже похожая проблема.
2. @AlexandraFilyakova Я не помню, как нашел решение, поэтому я придумал не очень приятный обходной путь с некоторыми вспомогательными функциями (я не помню, как именно, потому что это было давно, и теперь этот проект для меня недоступен). Извините, я не могу больше помочь: (
Ответ №1:
Попробуйте еще раз. Я не уверен, есть ли лучший способ сделать это.
class MyTest {
class A
class B
class MyClass() {
fun <T> myMethod(callback: () -> Single<Response<T>>): Single<T> {
error("")
}
}
val a = A()
val b = B()
@RelaxedMockK
lateinit var myClass: MyClass
@Before
fun setUp() {
MockKAnnotations.init(this)
}
@Test
fun tmp() {
every { myClass.myMethod<A>(any()) } returns (Single.just(a))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
every { myClass.myMethod<B>(any()) } returns (Single.just(b))
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp2() {
every { myClass.myMethod(any<() -> Single<Response<A>>>()) } returns (Single.just(a))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
every { myClass.myMethod(any<() -> Single<Response<B>>>()) } returns (Single.just(b))
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp3() {
every { myClass.myMethod(any<() -> Single<Response<A>>>()).hint(A::class) } returns (Single.just(a))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
every { myClass.myMethod(any<() -> Single<Response<B>>>()).hint(B::class) } returns (Single.just(b))
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
@Test
fun tmp4() {
every { myClass.myMethod<A>(any()).hint(A::class) } returns (Single.just(a))
myClass.myMethod<A> { Single.just(Response.success(a)) }.test().assertResult(a)
every { myClass.myMethod<B>(any()).hint(B::class) } returns (Single.just(b))
myClass.myMethod<B> { Single.just(Response.success(b)) }.test().assertResult(b)
}
}
Комментарии:
1. Спасибо! Это работает, но это не тот подход, который я мог бы использовать в своем реальном тестировании. Переключение поддельного ответа каждый раз непосредственно перед вызовом невозможно для теста в моем проекте, поэтому мне было интересно, как я могу возвращать разные макеты на основе параметра типа функции