自定义 JUnit 规则允许所有测试通过

Ada*_*331 5 junit android mockito powermock kotlin

我有一个引用一些静态方法的文件:

class MyViewModel {

    fun test() { }

    companion object {
        private val MY_STRING = ResourceGrabber.grabString(R.string.blah_blah)
    }
}
Run Code Online (Sandbox Code Playgroud)

在我对此文件的 JUnit 测试中,我编写了一些代码来模拟设置中的资源抓取器。这将编译并运行,并且以下测试失败,正如我所期望的那样:

@PrepareForTest(ResourceGrabber::class)
@RunWith(PowerMockRunner::class)
class MyViewModelTest {
    private lateinit var viewModel: MyViewModel

    @Before
    fun setup() {
        PowerMockito.mockStatic(ResourceGrabber::class.java)
        val mockResourceGrabber = Mockito.mock(ResourceGrabber::class.java)

        whenever(mockResourceGrabber.grabString(Mockito.anyInt())).thenAnswer { invocation ->
            val res: Int? = invocation?.arguments?.get(0) as? Int
            TestResourceGrabber.grabString(res)
        }

        viewModel = MyViewModel()
    }

    @Test
    fun someTest() {
        // Fails, as expected.
        assertEquals(2, 3)
    }
}
Run Code Online (Sandbox Code Playgroud)

这就是事情变得奇怪的地方。我最近了解了自定义 JUnit 规则,您可以使用这些规则来避免跨测试出现一些重复的代码。在这种情况下,我不想将我的资源抓取器工作复制并粘贴到每个使用它的测试套件中,所以我制定了一个自定义规则:

class ResourceGrabberRule : TestRule {
    override fun apply(base: Statement?, description: Description?): Statement {
        return object : Statement() {
            override fun evaluate() {
                PowerMockito.mockStatic(ResourceGrabber::class.java)
                val mockResourceGrabber = Mockito.mock(ResourceGrabber::class.java)

                whenever(mockResourceGrabber.grabString(Mockito.anyInt())).thenAnswer { invocation ->
                    val res: Int? = invocation?.arguments?.get(0) as? Int
                    TestResourceGrabber.grabString(res)
                }
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

下面是它的实现。疯狂的是,现在每个测试都通过了,无论如何:

@PrepareForTest(ResourceGrabber::class)
@RunWith(PowerMockRunner::class)
class MyViewModelTest {
    private lateinit var viewModel: MyViewModel

    @Rule
    @JvmField
    val resourceGrabber = ResourceGrabberRule()

    @Before
    fun setup() {
        viewModel = MyViewModel()
    }

    @Test
    fun someTest() {
        // PASSES!!!?!?!?!?!
        assertEquals(2, 3)
    }
}
Run Code Online (Sandbox Code Playgroud)

我不确定问题出在哪里。我已经尝试从 Android Studio 和命令行构建和运行测试。我不知道我是否错误地实施了我的规则,或者是与 Powermock 相关的 JUnit 规则的问题,还是 Kotlin 注释处理的问题。测试编译并运行,但不管测试本身内部是什么都通过。

我愿意接受关于这里架构的评论(我相信社区有很多),但我真的在寻找关于为什么我编写的规则通过每个测试的解释。

prf*_*low 2

在您的自定义 TestRule 中,您需要调用base.evaluate()以继续规则链https://github.com/junit-team/junit4/wiki/rules#custom-rules