我正在尝试在 kotlin 中使用 mockito。我创建了以下测试:
class HeroesDataSourceTest {
@Mock
lateinit var heroesRepository: HeroesRepository
@Mock
lateinit var params: PageKeyedDataSource.LoadInitialParams<Int>
@Mock
lateinit var callback: PageKeyedDataSource.LoadInitialCallback<Int, Heroes.Hero>
val hero = Heroes.Hero(1, "superman", "holasuperman", 1, null, null)
val results = Arrays.asList(hero)
val data = Heroes.Data(results)
val dataResult = Heroes.DataResult(data)
val compositeDisposable = CompositeDisposable()
lateinit var heroesDataSource: HeroesDataSource
private val heroesPublishSubject = PublishSubject.create<Heroes.DataResult>()
@Before
fun initTest(){
MockitoAnnotations.initMocks(this)
}
@Test
fun testLoadInitialSuccess(){
`when`(heroesRepository.getHeroes(ArgumentMatchers.anyInt())).thenReturn(heroesPublishSubject.singleOrError())
heroesDataSource = HeroesDataSource(heroesRepository, compositeDisposable)
val testObserver = TestObserver<Heroes.DataResult>()
heroesDataSource.loadInitial(params, callback)
heroesPublishSubject.onNext(dataResult)
testObserver.assertComplete()
}
}
但是当我在行when(heroesRepository.getHeroes(ArgumentMatchers.anyInt())).thenReturn(heroesPublishSubject.singleOrError())
中执行它时,它只是进入 getHeroes 方法而不是嘲笑它(可以肯定的是,因为 heroesRepository 没有初始化,因为是 mocket 该方法失败)。我在 java 中使用了很多次,它从来没有给我带来任何问题。在 kotlin 中我必须做什么才能正确模拟它?
编辑
在这里我把英雄存储库类
open class HeroesRepository {
val privateKey = "5009bb73066f50f127907511e70f691cd3f2bb2c"
val publicKey = "51ef4d355f513641b490a80d32503852"
val apiDataSource = DataModule.create()
val pageSize = 20
fun getHeroes(page: Int): Single<Heroes.DataResult> {
val now = Date().time.toString()
val hash = generateHash(now + privateKey + publicKey)
val offset: Int = page * pageSize
return apiDataSource.getHeroes(now, publicKey, hash, offset, pageSize)
}
fun generateHash(variable: String): String {
val md = MessageDigest.getInstance("MD5")
val digested = md.digest(variable.toByteArray())
return digested.joinToString("") {
String.format("%02x", it)
}
}
}
在不添加其他依赖项的情况下,您可以在某处使用 @Mock
注释替换为帮助程序函数:
inline fun <reified T> mock(): T =
Mockito.mock(T::class.java)
// To avoid having to use backticks for "when"
fun <T> whenever(methodCall: T): OngoingStubbing<T> =
Mockito.`when`(methodCall)
然后在测试中:
val heroesRepository: HeroesRepository = mock()
@Test
fun yourTest() {
whenever(heroesRepository.getHeroes(ArgumentMatchers.anyInt()))
.thenReturn(heroesPublishSubject.singleOrError())
}
就像你以前一样。这应该有效,因为您不希望 Mockito 处理@Mock lateinit var
,它似乎正在努力解决,而是自己实例化模拟。
FWIW,when
语句只是调用实际函数的原因是因为函数本身(getHeroes
)没有标记为open
。
因此,即使类是非最终的,该方法也是最终的,而不是模拟的。
Mockito 与 kotlin 不太兼容,你可以改用 mockito-kotlin 资源。查看此参考:https://github.com/nhaarman/mockito-kotlin
您可以轻松地模拟这样的对象:
val heroesRepository = mock<HeroesRepository>()
然后你可以像使用mockito一样使用它