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。
我需要测试一个控制器,对于大多数部分来说,它仍然足以使用 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。