java - when() 需要一个必须为 'a method call on a mock' 的参数

标签 java spring unit-testing

我在 Spring 中为我的 Controller 类编写 JUnit 测试用例。因为我收到一个 MissingMethodInvocationException 说 when() 需要一个参数,该参数必须是“模拟方法调用”,同时在 when 中调用 WebTarget 的模拟对象().then 方法。

这是代码。

@Controller 
public class JobController {   

    WebTarget target = null;   
    ClientConfig config = new ClientConfig();  
    Client client = ClientBuilder.newClient(config);`
    List<ScheduleJob> scheduleJobsList = new ArrayList<ScheduleJob>();

    @RequestMapping(value = "displayScheduledJobs")
    public ModelAndView displayScheduledJobs(@ModelAttribute(value = "Server") Server server) throws JAXBException {

        List<Server> serverList = serverService.getJobServerList(user.getAccountId());
        target = client.target(getBaseURI(server));
        String xml = target.path("rest").path("getScheduledJobs").request().accept(MediaType.APPLICATION_XML).get(String.class);
        if (xml.contains("<scheduleJob>") && xml.contains("</scheduleJob>")){
            //some code
       }
        model.addObject("scheduleJobsList", scheduleJobsList);
        return model;

     }

    private static URI getBaseURI(Server server) {
        if(server.getSecureFlag().equalsIgnoreCase("N")){
            return UriBuilder.fromUri("http://"+server.getServerIp()+":"+server.getServerPort()+"/jobserver").build();
        } else {
            return UriBuilder.fromUri("https://"+server.getServerIp()+":"+server.getServerPort()+"/jobserver").build();
        }
    }
}

这是我的测试代码。

@RunWith(MockitoJUnitRunner.class)
public class JobControllerTest {

    @Mock
    private WebTarget target;

    @Mock
    private ClientConfig config;

    @Mock
    private Client client;

    @Test
    public void testdisplayScheduledJobs() throws JAXBException {

        List<Server> serverList = new ArrayList<Server>();
        Server server = new Server();
        server.setSecureFlag("N");
        serverList.add(server);
        config = new ClientConfig();
        client = ClientBuilder.newClient(config);
        String xml = "<scheduledJobs>" + "</scheduledJobs>";

        when(serverService.getJobServerList(user.getAccountId())).thenReturn(serverList);
        when(client.target(getBaseURI(server))).thenReturn(target);


        when(target.path("rest")).thenReturn(target); //here the error is getting generated
        when(target.path("getScheduledJobs")).thenReturn(target);
        when(target.request()).thenReturn(builder);
        when(builder.accept(MediaType.APPLICATION_XML)).thenReturn(builder);
        when(builder.get(String.class)).thenReturn(xml);

        model = jobController.displayScheduledJobs(server, request);

        assertEquals("displayScheduledJobs", model.getViewName());

    }
}

请提出解决办法

org.mockito.exceptions.misusing.MissingMethodInvocationException: 
when() requires an argument which has to be 'a method call on a mock'.
For example:
  when(mock.getArticles()).thenReturn(articles);

Also, this error might show up because:
1. you stub either of: final/private/equals()/hashCode() methods.
   Those methods *cannot* be stubbed/verified.
2. inside when() you don't call method on mock but on some other object.

at com.techm.job.administration.console.controller.JobControllerTest.testdisplayScheduledJobs(JobControllerTest.java:309)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:76)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
at org.mockito.internal.runners.JUnit45AndHigherRunnerImpl.run(JUnit45AndHigherRunnerImpl.java:37)
at org.mockito.runners.MockitoJUnitRunner.run(MockitoJUnitRunner.java:62)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

最佳答案

您收到错误是因为您的代码中有以下行:

client = ClientBuilder.newClient(config);

当您编写此语句时,它会将您在上面创建的模拟客户端对象替换为新的客户端,因此 client 将不再被 Mockito 识别为模拟对象。

此外,我希望您正在创建 builder 的模拟,就像您正在执行 when(builder.accept(MediaType.APPLICATION_XML)).thenReturn(builder); 一样。

此外,我看不到任何@InjectMocks 语句。您必须创建一个您正在测试的类的对象,并使用 @InjectMocks 注释对其进行注释。

此外,您还必须通过在 testdisplayScheduledJobs() 方法中执行 MockitoAnnotations.initMocks(this); 来初始化模拟。

下面是一个示例代码片段,它将为您提供更多见解:

public class Foo {

    @Mock
    private JerseyWebTarget target;

    @Mock
    private Builder requestBuilder;

    @Mock
    private Response serviceResponse;

    @InjectMocks
    private Foo foo = new Foo();

    @Rule
    public ExpectedException thrown = ExpectedException.none();

    @Before
    public void init() {
        MockitoAnnotations.initMocks(this);

        Mockito.when(target.queryParam(Matchers.anyString(), Matchers.anyString()))
                .thenReturn(target);

        Mockito.when(target.queryParam(Matchers.anyString(), Matchers.anyString()))
                .thenReturn(target);

        Mockito.when(target.queryParam(Matchers.anyString(), Matchers.anyString()))
                .thenReturn(target);

        Mockito.when(target.queryParam(Matchers.anyString(), Matchers.anyString()))
                .thenReturn(target);

        Mockito.when(target.request(MediaType.APPLICATION_JSON_TYPE)).thenReturn(
                requestBuilder);
    }

    @Test
    public void testGetData() {
        List<String> responseList = new ArrayList<>();
        responseList.add("foobar");

        Mockito.when(requestBuilder.get(Matchers.eq(Response.class))).thenReturn(serviceResponse);

        Mockito.when(serviceResponse.getStatus()).thenReturn(200);

        Mockito.when(serviceResponse.readEntity(Matchers.any(GenericType.class))).thenReturn(
                responseList);

        List<String> resultList = foo.getData("foo", "bar");

        Mockito.verify(requestBuilder, Mockito.times(1)).get(Matchers.eq(Response.class));
        Mockito.verify(serviceResponse, Mockito.times(1)).getStatus();
        Mockito.verify(serviceResponse, Mockito.times(1)).readEntity(
                Matchers.any(GenericType.class));

        assertNotNull(resultList);
        assertEquals(1, resultList.size());
        assertEquals("true", resultList.get(0).getConfigSetValue());
    }
}

关于java - when() 需要一个必须为 'a method call on a mock' 的参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35840428/

相关文章:

java - RichFaces 丰富 :datatable how to align left/right/center columns

java - 为什么使用 JUnit 编写测试类时不需要扩展 TestCase?

java - 使用内存数据库和模拟测试 DAO 和服务层

java - 将一条线分成两个不同的对象

java - 为什么 "short thirty = 3 * 10"是合法分配?

java.time.LocalDateTime 在带有 hibernate 的 JPA 中被持久化为tinyblob而不是 TemporalType

c# - 如何将此 EF Mock 设置代码编写为可重用的通用样板?

java - 如何知道两个条件何时为真

java - 依赖 Spring 引导项目 application.properties 不注入(inject)默认值

java - Spring 网络流程 : setting request parameter for JUnit test