在非最终非静态类上,mock-maker-inline 使测试失败,并显示“传递给 Mockito.mockingDetails() 的 NotAMockException 参数应该是一个模拟”

dom*_*aru 5 java junit mockito

我有一个 JUnit 测试,其中有 5 个模拟,其中一个模拟类是最终的,我启用了 mock-maker-inline 来克服这个问题,现在测试通过了,但在测试执行结束时,我还得到了 NotAMockException:

org.mockito.exceptions.misusing.NotAMockException: Argument passed to Mockito.mockingDetails() should be a mock, but is an instance of class ...Locals!
at org.mockito.internal.runners.DefaultInternalRunner$1$2.testFinished(DefaultInternalRunner.java:63)
    at org.junit.runner.notification.SynchronizedRunListener.testFinished(SynchronizedRunListener.java:56)
    at org.junit.runner.notification.RunNotifier$7.notifyListener(RunNotifier.java:190)
    at org.junit.runner.notification.RunNotifier$SafeNotifier.run(RunNotifier.java:72)
    at org.junit.runner.notification.RunNotifier.fireTestFinished(RunNotifier.java:187)
    at org.junit.internal.runners.model.EachTestNotifier.fireTestFinished(EachTestNotifier.java:38)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:331)
    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.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:74)
    at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:80)
    at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
    at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
...
Run Code Online (Sandbox Code Playgroud)

嘲笑非常简单:

    @Mock
    private Locals locals; //class

    @Mock
    private Globals globals; // class

    @Mock
    private Cache cache; // interface

    @Mock
    private Reminds reminds; // final class

    @Mock
    private Employeereminds employeereminds; // class
Run Code Online (Sandbox Code Playgroud)

...然后是几个when().thenReturn()设置以使其协同工作

在测试过程中,模拟看起来像这样:

locals = {Locals@3233} 
globals = {Globals@3234} 
cache = {Cache$MockitoMock$665734991@3237} "cache"
reminds = {Reminds@3235} 
employeereminds = {Employeereminds@3236} 
Run Code Online (Sandbox Code Playgroud)

我正在使用 Mockito 2.25.0 也尝试过 2.28.2 但没有区别。

我可以重构我的测试来测试期末课程的内容,但我更喜欢孤立的测试。我也可以使用 powermock,但也看到过 powermock 报告的类似问题。

有没有人偶然发现这样的问题并找到一个不错的解决方法?

编辑:我做了一些更多的调查,我删除了Locals模拟并在Globals模拟上遇到了同样的错误。然后我删除了Globals模拟并得到了(!?)以下异常(@ v2.28.2):

org.mockito.exceptions.misusing.NotAMockException: Argument passed to Mockito.mockingDetails() should be a mock, but is an instance of class Cache$MockitoMock$149288076!

    at org.mockito.internal.runners.DefaultInternalRunner$1$2.testFinished(DefaultInternalRunner.java:63)
    at org.junit.runner.notification.SynchronizedRunListener.testFinished(SynchronizedRunListener.java:56)
    at org.junit.runner.notification.RunNotifier$7.notifyListener(RunNotifier.java:190)
    at org.junit.runner.notification.RunNotifier$SafeNotifier.run(RunNotifier.java:72)
    at org.junit.runner.notification.RunNotifier.fireTestFinished(RunNotifier.java:187)
    at org.junit.internal.runners.model.EachTestNotifier.fireTestFinished(EachTestNotifier.java:38)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:331)
    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.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:74)
    at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:80)
    at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
    at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    ...
Run Code Online (Sandbox Code Playgroud)

dom*_*aru 2

好的,我发现了问题:我正在调用

Mockito.framework().clearInlineMocks();
Run Code Online (Sandbox Code Playgroud)

在注释调用的代码中@After,但正确的方法是在注释调用的代码中执行此操作@AfterClass,否则在清除模拟后会运行 Mockito 自己的验证。