首页 > 解决方案 > ConstraintValidator 两次调用 isValid() 方法

问题描述

任何机构向我解释为什么 ConstraintValidator 类的 isValid() 方法被调用两次?
例如这是我的示例代码:

@POST
@Path("/json/dog")
@Produces("application/json")
@Consumes("application/json")
public Response getDogByJson(@ValidAnimal JsonObject jsonObject) {
      return Response.ok("ok").build();
}



@Constraint(validatedBy = {AnimalValidation.class})
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.PARAMETER)
public @interface ValidAnimal {

    String message() default "This is not valid !";

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

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



public class AnimalValidation implements ConstraintValidator<ValidAnimal, JsonObject> {

    @Override
    public void initialize(ValidAnimal constraintAnnotation) {

    }

    @Override
    public boolean isValid(JsonObject jsonObject, ConstraintValidatorContext context) {
        System.out.println(">>>>>> : " + jsonObject);
        return true;
    }
}

当我向此 Web 服务发送请求时,isValid() 方法调用两次。
为什么 ?

注意:我使用 OpenLiberty 18.0.0.3 和 JavaEE 8

标签: jakarta-eejerseyjax-rsopen-libertyjava-ee-8

解决方案


我进行了快速测试并确认了您注意到的行为。看起来第一次是从 JAX-RS 运行时调用 validate,然后 CDI 运行时也调用 validate(因为 JAX-RS 资源也是 CDI bean)。我正在研究如果 OpenLiberty 可以检测到启用了 CDI 功能,我们是否可以让 JAX-RS 避免调用。

当我启用这些功能时,isValid()JAX-RS 只会调用一次:

<featureManager>
    <feature>jaxrs-2.1</feature>
    <feature>jsonp-1.1</feature>
    <feature>beanValidation-2.0</feature>
</featureManager>

在 com.ibm.ws.jaxrs20.server.LibertyJaxRsInvoker.callValidationMethod(LibertyJaxRsInvoker.java:371)

但是当我们启用 CDI 功能时,CDI 运行时isValid()也会调用:

<featureManager>
    <feature>jaxrs-2.1</feature>
    <feature>jsonp-1.1</feature>
    <feature>beanValidation-2.0</feature>
    <feature>cdi-2.0</feature>
</featureManager>

在 org.hibernate.validator.cdi.internal.interceptor.ValidationInterceptor.validateMethodInvocation(ValidationInterceptor.java:66)
...
在 com.ibm.ws.jaxrs20.server.LibertyJaxRsInvoker.performInvocation(LibertyJaxRsInvoker.java:160)

总而言之,这看起来像是 OpenLiberty 中的一个错误。应该进行更新,以便isValid()仅在启用 JAX-RS 和 CDI 功能时调用一次。


推荐阅读