在 Spring Boot 中进行请求验证时,响应不符合预期

Posted

技术标签:

【中文标题】在 Spring Boot 中进行请求验证时,响应不符合预期【英文标题】:While Doing Request Validation in Springboot, Response is not as expected 【发布时间】:2021-12-07 23:24:39 【问题描述】:

我正在尝试使用 hibernate-validator 验证 json-resquest,它按预期工作,但邮递员中没有响应。

Customer.java

import java.time.LocalDate;
import java.util.List;

import javax.validation.Valid;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Past;
import javax.validation.constraints.Size;

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonPropertyOrder;

@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonPropertyOrder( "cin", "firstName")
public class Customer 

    @JsonProperty("cin")
    private String cin;

    @JsonProperty("firstName")
    @NotEmpty(message = "First Name must have some values")
    @Size(min = 2, message = "First Name must greater or equal to 2 characters")
    private String firstName;


//getters and setters


错误类 - 将错误包装在一个对象中。

public class Errors 

    private Integer status;
    private String message;
    private List<String> details;

    public Errors(Integer status, String message, List<String> details) 
        super();
        this.status = status;
        this.message = message;
        this.details = details;
    
    
// Getters and Setters



ControllerAdvice 类

import java.util.List;
import java.util.stream.Collectors;

import javax.validation.ConstraintViolationException;

import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.context.request.WebRequest;
import org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler;

import com.ecommerce.ms.customer.model.Errors;

@ControllerAdvice
@ResponseBody
public class CustomerExceptionHandler extends ResponseEntityExceptionHandler 

    @ExceptionHandler(value=ConstraintViolationException.class)
    public final ResponseEntity<Errors> handleConstraintViolation(ConstraintViolationException ex, WebRequest request) 
        List<String> details = ex.getConstraintViolations().parallelStream().map(e -> e.getMessage())
                .collect(Collectors.toList());
        Errors error = new Errors(HttpStatus.BAD_REQUEST.value(), "Request Validation Error", details);
        return ResponseEntity.badRequest().body(error);
    


CustomerController.java

 * 
 */

import java.util.ArrayList;
import java.util.List;

import javax.validation.Valid;
import javax.ws.rs.core.MediaType;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import com.ecommerce.ms.customer.api.service.CustomerService;
import com.ecommerce.ms.customer.model.Customer;

@RestController
@RequestMapping("/api/customers")
public class CustomerController 

    @Autowired
    private CustomerService customerService;

    @GetMapping("/status")
    public String getStatus() 
        return "ok";
    

    @PostMapping(consumes = MediaType.APPLICATION_JSON, produces = MediaType.APPLICATION_JSON)
    public ResponseEntity<Customer> addCustomer(@Valid @RequestBody Customer customer) 
        return ResponseEntity.accepted().body(customerService.addCustomer(customer));
    


Hibernate-validator 已添加 pom.xml,我期待以下原因。


   "status":400,
"message": "Request Validation Error",
"details":["First Name must greater or equal to 2 characters"]

我正在尝试获得正确的响应正文,但在邮递员中找不到它。

【问题讨论】:

【参考方案1】:

查看 ResponseEntityExceptionHandler 没有这样的方法来处理 ConstraintValidationExceptions,因此您创建的自定义方法没有被调用。

https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/servlet/mvc/method/annotation/ResponseEntityExceptionHandler.html

还有:

您无法捕获 ConstraintViolationException.class,因为它不是 传播到代码的那一层,它被较低的 层,包裹并在另一种类型下重新抛出。这样异常 击中您的 Web 层的不是 ConstraintViolationException。

参考:SpringBoot doesn't handle org.hibernate.exception.ConstraintViolationException

正确使用的一个例子是使用方法handleMethodArgumentNotValid并将错误作为正文返回:

@RestControllerAdvice
public class ExceptionHandler extends ResponseEntityExceptionHandler

    @Override
    protected ResponseEntity<Object> handleMethodArgumentNotValid(MethodArgumentNotValidException ex,
                                                                  HttpHeaders headers, HttpStatus status, WebRequest request) 

        Map<String, Object> responseBody = new LinkedHashMap<>();

        List<String> allErrors = new ArrayList<>();
        ex.getBindingResult().getAllErrors().forEach(error -> allErrors.add(error.getDefaultMessage()));
        responseBody.put("Errors:", allErrors);

        return new ResponseEntity<>(responseBody, headers, status);
    


【讨论】:

以上是关于在 Spring Boot 中进行请求验证时,响应不符合预期的主要内容,如果未能解决你的问题,请参考以下文章

Spring boot - 一些 POST 请求严重延迟

如何在 Spring Boot 应用程序中使用 Hibernate 验证进行 Bean 验证?

Spring Boot Keycloak - 每个请求验证 3 次

Spring Boot + Spring Security Restful 登录

Spring Boot如何返回我自己的验证约束错误信息

Spring Boot使用AOP在控制台打印请求响应信息