【问题标题】:What is causing this mockito exception?是什么导致了这个 mockito 异常?
【发布时间】:2019-03-05 22:16:46
【问题描述】:

我正在为一个项目创建测试用例。项目是如何设计的,很难继续进行简单的单元测试,并且似乎缺少一个依赖项,以便使用 Mockito 来模拟函数和对象。这是我所拥有的:

@RunWith(SpringRunner.class)
public class RequestHandlerTest {
    private byte[] buffer;
    private Connection conn;
    private Server server;

    @Mock ResponseHandler response;
    @Mock Services serviceImpl;
    @InjectMocks RequestHandler request;

    @Before
    // This just creates the buffer in order to simulate sending and 
    // receiving a response across a server and client
    public void createRequests(); 

    @After
    public void cleanUpBuffer(); // Cleans up buffer after test

    @Test
    public void TestReceiveMessage() {
        doNothing().when(response.updataCache(conn, reqID));
        ... // Do nothing for service function
        handler.receiveMsg(conn, buffer);
        // Verify request
    }
}

现在每当我运行测试时,我都会收到以下异常:

 org.mockito.exceptions.base.MockitoException: At the moment you cannot provide own implementations of MockHandler.
Please see the javadocs for the MockMaker interface.
    at org.springframework.boot.test.mock.mockito.MockitoTestExecutionListener.initMocks(MockitoTestExecutionListener.java:68)
    at org.springframework.boot.test.mock.mockito.MockitoTestExecutionListener.prepareTestInstance(MockitoTestExecutionListener.java:53)
    at org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:246)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:227)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner$1.runReflectiveCall(SpringJUnit4ClassRunner.java:289)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:291)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:246)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:97)
    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.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
        at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
        at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
        at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:190)
        at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:89)
        at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:41)
        at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:541)
        at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:763)
        at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:463)
        at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:209)

我不确定是什么原因造成的,以及需要做些什么来解决这个问题。我尝试设置初始化 Mockito,使用 @Autowired 将 @Mock 更改为 @MockBean,还尝试使用包装类设置我的值。有什么我可能会丢失的吗?我以前从未见过这种情况。

【问题讨论】:

    标签: java spring-boot junit mockito


    【解决方案1】:

    首先,这里不需要使用SpringRunner。将@RunWith(SpringRunner.class) 替换为@RunWith(MockitoJUnitRunner.class)

    其次,如果还是出现这个问题,尝试使用next(假设RequestHandlerImpl是RequestHandler的实现):

    @InjectMocks RequestHandler request = new RequestHandlerImpl();

    甚至@InjectMocks RequestHandlerImpl request

    【讨论】:

    • 问题仍然存在,但堆栈跟踪现在要短得多。这确实适用于其他测试,由于某种原因,一旦我尝试模拟一个对象,它就会失败。我可能需要查看设计,看看是否有更好的方法来实现现有的功能以使其更易于测试和维护。
    • 也许你也可以发布RequestHandler、ResponseHandler、Services的源码?
    • 不幸的是,由于政策原因,我无法这样做:/我知道一旦通过调试器调用 mock 就会出现问题。当我自己模拟课程时,我会做我想做的事,但这会更耗时。我将继续调查它,看看是什么原因造成的。
    • 您提供的部分解决方案确实有帮助,我只是缺少一些依赖项的配置。
    【解决方案2】:

    好的,所以在查看了这个之后,我看到有很多 bean 正在被 Autowired。所以我需要为我所依赖的类添加上下文配置。然后我可以使用 Mock 和 Inject Mock 的注释。所以我的灯具变成了这样:

    @RunWith(SpringRunner.class)
    @ContextConfiguration(classes = {RequestHandler.class, ResponseHandler.class,
          DataCache.class, Service.class})
    public class RequestHandlerTest {
       @Mock private ResponseHandler response;
       @Mock private Service serv;
    
       @InjectMocks
       @Resource
       private RequestHandler request;
    
       public void setUp() {
          MockitoAnnotations.initMocks(this);
    
          // Add mock data logic using doAnswer or doNothing for void methods if needed 
          // and adding when().then() for mocking return values for other functions
    
       }
    
       @Test
       public void makeRequest() {
          assertTrue(request.sendRequest(Object message));
       }
    }
    

    Mocking Void Methods with Mockito 和这个问题的答案here 中提取信息。

    【讨论】:

    • @amseager 的回答仍然正确。您可以创建一个带有模拟的单元测试并由 aMockitoJUnitRunner 运行(它将初始化和注入模拟),或者一个由 SpringRunner 运行的集成测试(这将创建一个注入适当 bean 的 spring 上下文)。跨度>
    • 是的,在创建测试时它是正确的。但是如果ContextConfiguration 不存在,它甚至无法创建模拟。这就是为什么我得到了例外。一旦注释被删除,它就会失败。设计是紧密耦合的,因此我们在创建模拟时遇到了问题:(
    猜你喜欢
    • 2014-11-07
    • 2010-10-29
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-08-10
    • 2019-07-01
    相关资源
    最近更新 更多