为什么MockMvc请求在测试成功时检索空的responseBody?

问题描述 投票:0回答:2

我正在测试我的Spring Boot rest控制器,以检查如果bean验证失败,请求是否发送适当的错误。

我有一个@RestController:

@RestController
@RequestMapping("/restaurants")
public class RestaurantsApiController {

    private final RestaurantService restaurantService;
    private final ProductRepository productRepository;
    private final ProductMapper productMapper;

    public RestaurantsApiController(RestaurantService restaurantService, ProductRepository productRepository, ProductMapper productMapper) {
        this.restaurantService = restaurantService;
        this.productRepository = productRepository;
        this.productMapper = productMapper;
    }

    @PostMapping("{id}/products")
    public ResponseEntity<ProductDto> addProduct(@PathVariable Long id,
                                                 @Valid @RequestBody ProductDto productDto){
        Product product = this.restaurantService.addProduct(id, productMapper.productDtoToProduct(productDto));
        return new ResponseEntity<>(productMapper.productToProductDto(product), HttpStatus.CREATED);
    }

我有一个带有@ControllerAdvice批注的自定义exceptionHandler:

@ControllerAdvice
public class ExceptionControllerAdvice {
    @ExceptionHandler({MethodArgumentNotValidException.class})
    public ResponseEntity<Object> validationException(MethodArgumentNotValidException ex, WebRequest request) {
        ....
// here i format my custom error message
        return new ResponseEntity<>(apiError, new HttpHeaders(), apiError.getStatus());
    }

这非常有效,如果验证失败,则会向我发送此自定义响应:

{
    "status": "BAD_REQUEST",
    "errors": {
        "price": "doit être supérieur ou égal à 0",
        "name": "ne doit pas être nul",
        "category": "ne doit pas être nul"
    }
}

我正在尝试使用mockMvc来测试此测试类的行为:

@ExtendWith(MockitoExtension.class)
class RestaurantsApiControllerTest {

    @Mock
    private RestaurantService restaurantService;
    @Mock
    private ProductRepository productRepository;
    @Mock
    private ProductMapper productMapper;
    @InjectMocks
    private RestaurantsApiController controller;

    MockMvc mockMvc;

    @BeforeEach
    void setUp() {
        mockMvc = MockMvcBuilders.standaloneSetup(controller).build();
    }

    @Test
    void givenInvalidFrom_whenAddProduct_ThenShouldThrowException() throws Exception {
        // productDto miss name, category and have negative value for price which is forbidden by validations annotations
        ProductDto productDto = ProductDto.builder().id(1L).price(-10.5D).build();

        MvcResult mvcResult = mockMvc.perform(post("/restaurants/1/products")
                .contentType(MediaType.APPLICATION_JSON)
                .content(asJsonString(productDto)))
                .andExpect(status().isBadRequest())
                .andReturn();

        String result = mvcResult.getResponse().getContentAsString();

        then(restaurantService).shouldHaveNoInteractions();
    }

测试通过完美,我可以在日志中看到验证异常正确预期的内容:

14:53:30.345 [main] WARN org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver - Resolved [org.springframework.web.bind.MethodArgumentNotValidException: Validation failed for argument ...
//I removed the rest of the message for readability, but each validation exception appears here properly
14:53:30.348 [main] DEBUG org.springframework.test.web.servlet.TestDispatcherServlet - Completed 400 BAD_REQUEST

但是我找不到一种方法来测试我的错误地图是否包含我期望的字段。当我尝试使用:

检索响应正文时
String result = mvcResult.getResponse().getContentAsString();

字符串为空,我找不到测试响应正文的任何​​方法。

我完全不知道,有些帮助将不胜感激!

非常感谢!

java spring-boot rest junit5 mockmvc
2个回答
0
投票
MvcResult mvcResult = mockMvc.perform(post("/restaurants/1/products")
                .contentType(MediaType.APPLICATION_JSON)
                .content(asJsonString(productDto)))
                .andExpect(status().isBadRequest())
                .andExpect(content().string("Your content"))
                .andReturn();   

或使用自定义匹配器

MvcResult mvcResult = mockMvc.perform(post("/restaurants/1/products")
                .contentType(MediaType.APPLICATION_JSON)
                .content(asJsonString(productDto)))
                .andExpect(status().isBadRequest())
                .andExpect(content().string(new CustomMatcher()))
                .andReturn();

    private static class ContentMatcher extends CustomMatcher<String>{

            public ContentMatcher() {
                super("");
            }

            @Override
            public boolean matches(Object o) {
                final String expected="Some long wide string " +
                        "wich i should check" +
                        "...."
                return o.equals(expected);
            }
        }

0
投票
result.andExpect(status().isBadRequest())
            .andExpect(jsonPath(STATUS), is("BAD_REQUEST"))
            .andExpect(jsonPath("errors.price", is("EXPECTED MESSAGES HERE")));
© www.soinside.com 2019 - 2024. All rights reserved.