Mockito:如何部分使用 @InjectMocks-Annotation 进行 MockMvc 测试?

Mockito: How to partially use @InjectMocks-Annotation for MockMvc testing?

我需要测试一个控制器,对于大多数部分来说,它仍然足以使用 MockMvc。但是仍然有一些分支很难生成测试用例。所以我使用 Mockito 作为存根方法并为这个特殊的测试用例抛出异常。

我的控制器实现了一项服务,设置如下所示:

MyController.class:

@RestController
@RequestMapping("/myTest")
public class MyController {

    @Autowired
    private MyService myService;

    @RequestMapping(value = "/", method = { RequestMethod.POST }, produces = "application/json")
    public final int controllerMethod() {
        return myService.serviceMethod();
    }
}

MyService.class:

@Service
public class MyService {
    @PreAuthorize("hasRole('ROLE_ADMIN')")
    public int serviceMethod() {
        return 5;
    }
}

所以我有多个测试方法,在某些方法中我模拟 MyService.class 以抛出异常。 但在其他方法中,我想测试正常的功能。所以这里出现了问题。 如果测试 运行 的顺序不正确,正常的功能测试将无法工作,因为 MyService.class 仍然是模拟的,我猜。 所以在这个例子中,serviceMethod returns null 而不是 5.

有没有办法部分 InjectMocks - 所以我的意思是只针对特殊方法而不是 class?

示例代码如下:

MyControllerTest.class:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = Config.class)
@WebAppConfiguration
public class MyControllerTest {
    @Autowired
    private WebApplicationContext webApplicationContext;

    @Autowired
    MyService myService;

    @InjectMocks
    @Resource
    MyController myController;

    @Resource
    private FilterChainProxy springSecurityFilterChain;

    private MockMvc mockMvc;

    @Before
    public void setup() throws SQLException {
        this.mockMvc = MockMvcBuilders.webAppContextSetup(webApplicationContext).apply(springSecurity()).build();
    }

    @Test
    @WithMockUser(username = "validUser", roles = { "ADMIN" })
    public void testMyController() throws Exception {
        mockMvc.perform(post("/myTest/").with(csrf())).andDo(print()).andExpect(status().is2xxSuccessful()).andExpect(jsonPath("$", is(5)));
    }

    @Test
    @WithMockUser(username = "validUser", roles = { "ADMIN" })
    public void testMyControllerException() throws Exception {
        myService = Mockito.spy(MyService.class);

        MockitoAnnotations.initMocks(this);

        Mockito.doThrow(new IndexOutOfBoundsException()).when(myService).serviceMethod();

        mockMvc.perform(post("/myTest/").with(csrf())).andDo(print()).andExpect(status().is4xxClientError());

        Mockito.verify(myService, Mockito.times(1)).serviceMethod();
    }
}

所以如果我 运行 特别是这些测试中的一个,两者都运行良好。 但是如果我 运行 一起执行并且 testMyControllerException() 将首先执行,我猜 testMyController() 会因为模拟的 myService-Instance 而失败。

有什么方法可以解决这个问题,还是我必须将这个测试分成两个不同的测试classes?

我试图找到一种方法来取消 myController.class@After-注释的 Mockito.reset(myController); 的模拟。但对我没用。

对于 jUnit4,您可以使用 @FixMethodOrder(MethodSorters.NAME_ASCENDING) https://github.com/junit-team/junit4/wiki/Test-execution-order

无论如何,我个人更喜欢使用不同的 类。

要独立于执行顺序且不严格执行顺序,最好的方法是使用Class-Annotation @DirtiesContext(classMode = ClassMode.AFTER_EACH_TEST_METHOD)。对于每个测试,它都会生成一个新的 ApplicationContext。