Spring @ControllerAdvice messageSource不能与Hibernate消息参数一起使用

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

我正在尝试在Spring Boot REST控制器中添加一些自定义bean验证,使用ResponseEntityExceptionHandler注释扩展@ControllerAdvice类并覆盖#handleMethodArgumentNotValid(MethodArgumentNotValidException e, HttpHeaders headers, HttpStatus status, WebRequest request)方法。在这个方法中,我试图通过FieldError将给定的messageSource转换为本地化的消息。虽然我在尝试使用可通过Hibernate验证器实现的消息参数时收到NumberFormatException

我正在使用以下依赖项:

  • org.hibernate.validator:hibernate-validator(6.0.11.Final)
  • org.springframework:spring-web(5.0.8.RELEASE)
  • org.springframework:spring-webmvc(5.0.8.RELEASE)

全部包含在org.springframework.boot:spring-boot-starter-web(2.0.4.RELEASE)中。

考虑使用以下REST控制器:

@RestController
public class FooController {

    @PostMapping(value = "/foo")
    public void submitFooRequest(@Validated @RequestBody FooRequest fooRequest) {
        // ....
    }
}

FooRequest bean有一个自定义bean验证注释和约束验证器:

bean FooRequest:

@Getter
@Setter
@ValidBarRequest
public class FooRequest {

    private String fieldFoo;
    private BarRequest barRequest;
}

豆BarRequest:

@Getter
@Setter
public class BarRequest {

    private String fieldBar;
}

验证注释:

@Target({ElementType.TYPE, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = BarRequestValidator.class)
@Documented
public @interface ValidBarRequest {

    String message() default "{org.example.validation.constraints.ValidBarRequest.message}";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};

    String fieldFoo() default "fieldFoo";

    String barRequestFieldBar() default "barRequest.fieldBar";
}

验证约束验证器:

@Log4j2
public class BarRequestValidator implements ConstraintValidator<ValidBarRequest, Object> {

    // ....

    @Override
    public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) {
        if (/* some condition */) {
            HibernateConstraintValidatorContext hibernateValidatorContext = constraintValidatorContext.unwrap(HibernateConstraintValidatorContext.class);
            hibernateValidatorContext.disableDefaultConstraintViolation();
            hibernateValidatorContext.addMessageParameter("fieldFoo", "some value...").buildConstraintViolationWithTemplate("{org.example.validation.constraints.ValidBarRequest.message}")
                .addPropertyNode("barRequest.fieldBar").addConstraintViolation();
            return false;
        }

        return true;
    }
}

但是,通过@ControllerAdvice注释bean并使用Spring messageSource,会在以下消息上抛出NumberFormatException(在messages.properties中):

ValidBarRequest.fooRequest.barRequest.fieldBar=must be lower or equal than {fieldFoo}

@ControllerAdvice bean:

@ControllerAdvice
public class ControllerExceptionHandler extends ResponseEntityExceptionHandler {

    @Autowired private MessageSource messageSource;

    @Override
    protected ResponseEntity<Object> handleMethodArgumentNotValid(MethodArgumentNotValidException e, HttpHeaders headers, HttpStatus status, WebRequest request) {
        List<ErrorDetails> errorDetails = new ArrayList<>();
        for (FieldError fieldError : e.getBindingResult().getFieldErrors()) {
            errorDetails.add(new ErrorDetails(fieldError.getField(), messageSource.getMessage(fieldError, Locale.getDefault())));
        }

        return new ResponseEntity<>(errorDetails, HttpStatus.BAD_REQUEST);
    }

    @Getter
    @AllArgsConstructor
    class ErrorDetails {

        private String field;
        private String message;
    }
}

这会导致以下异常:Caused by: java.lang.NumberFormatException: For input string: "fieldFoo"

我究竟做错了什么?我还在我的@SpringBootApplication中包含了以下bean:

@Bean
public LocalValidatorFactoryBean validator(MessageSource messageSource) {
    LocalValidatorFactoryBean localValidatorFactory = new LocalValidatorFactoryBean();
    localValidatorFactory.setValidationMessageSource(messageSource);

    return localValidatorFactory;
}
spring-mvc spring-boot hibernate-validator
1个回答
0
投票

填充消息源MessageFormat.format中的变量是必须使用花括号中的数字。

ValidBarRequest.fooRequest.barRequest.fieldBar=must be lower or equal than {0}
© www.soinside.com 2019 - 2024. All rights reserved.