我正在尝试测试我的控制器端点和我的请求者注释@Valid
注释。我的Testclass看起来如下:
@RunWith(SpringRunner.class)
@WebMvcTest(value = BalanceInquiryController.class, secure = false)
public class BalanceInquiryControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private BalanceInquiryController balanceInquiryController;
@Test
public void testGetBalanceInquiry() throws Exception {
RequestBuilder requestBuilder = MockMvcRequestBuilders
.post("/com/balanceInquiry")
.accept(MediaType.APPLICATION_JSON)
.content("{\"comGiftCard\":{\"cardNumber\":\"1234567890\",\"pinNumber\":\"0123\"},\"comMerchant\":\"MERCHANT1\"}")
.contentType(MediaType.APPLICATION_JSON);
MvcResult mvcResult = mockMvc.perform(requestBuilder).andReturn();
MockHttpServletResponse response = mvcResult.getResponse();
assertEquals(HttpStatus.OK.value(), response.getStatus());
}
}
我的控制器 - @PostMapping
看起来像这样:
@PostMapping(value = "/com/balanceInquiry")
public ResponseEntity<?> getBalanceInquiry(@Valid @RequestBody BalanceInquiryModel balanceInquiry, Errors errors) {
if (errors.hasErrors()) {
return new ResponseEntity<String>("Validation error", HttpStatus.BAD_REQUEST);
}
//do any stuff...
return new ResponseEntity<BalanceInquiryResponse>(balanceInquiryResponse, HttpStatus.OK);
}
我的BalanceInquiryModel
使用@Valid
进行了注释,并且后面有一些休眠和自定义验证。这些验证都没问题,已经过单元测试。
我喜欢测试的是我的端点,我发送一个有效的json请求主体,期望200响应,也是一个无效的json请求主体,期望通过set @Valid
实现验证400响应。
例如,无效调用是不发送pinNumber或长度&lt; 4。
我已经读过一些线程,有些使用MockMvcBuilders.standaloneSetup()
来模拟完整的控制器。但我不会进行完整的集成测试。
不太确定如何继续这种情况,如果我继续下去。
P.S。:目前,无论验证是否应该出错,我都会得到200响应。
Here a gist获取更多代码和验证类/模型。
答案 0 :(得分:1)
这是我在我的项目中工作的一个例子 希望它可以帮到你:
我有一个全局异常处理程序来处理我的MethodArgumentNotValidException并抛出它
@RequestMapping(value = "/add", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
public ResponseEntity<User> createUser(@Valid @RequestBody User user) {
User savedUser = userService.save(user);
return new ResponseEntity<User>(savedUser, HttpStatus.CREATED);
}
public void testAdduser() throws Exception{
final User request = new User();
request.setFirstName("Test");
request.setLastName("some description");
mockMvc.perform(post(END_POINT+"/add")
.contentType(MediaType.APPLICATION_JSON)
.content(stringify(request))
).andDo(print()).andExpect(status().isUnprocessableEntity())
;
}
private String stringify(Object object) throws JsonProcessingException {
return new ObjectMapper().writeValueAsString(object);
}
更新
我认为您的主要问题是您使用 @WebMvcTest 而不是 @SpringBootTest 。
其中两个之间的差异是:
@SpringBootTest 注释将加载完整的应用程序并注入所有可能很慢的bean。
@WebMvcTest - 用于测试控制器层。它不会在@RestController
旁边注入其他bean所以,如果您只是测试纯控制器以查看您可以到达endpont,那么您可以使用@WebMvcTest,这将使您的测试运行得更快。
但是在你的情况下,你希望它运行弹簧验证,你需要使用@SpringBootTest