当子句在 kotlin 中不起作用时,Mockito



我正在尝试在 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一样使用它

相关内容

  • 没有找到相关文章

最新更新