我尝试将Bean Validation用于我的REST API与Apache CXF。我阅读了Apache CXF Documentation,它适用于根资源,但它不适用于sub resource locators。子资源会忽略约束。
Maven依赖项:
.js
Java代码:
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-frontend-jaxrs</artifactId>
<version>3.1.5</version>
</dependency>
<dependency>
<groupId>javax.validation</groupId>
<artifactId>validation-api</artifactId>
<version>1.1.0.Final</version>
</dependency>
<dependency>
<groupId>org.apache.bval</groupId>
<artifactId>bval-jsr</artifactId>
<version>1.1.0</version>
</dependency>
CXF配置:
@Named
@Path("test")
public class TestResource {
@Inject
private TestSubResource subResource;
@Path("sub")
public TestSubResource getSubResource() {
return subResource;
}
@GET
public void find(@NotNull @QueryParam("value") String value) {
}
}
@Named
public class TestSubResource {
@GET
public void find(@NotNull @QueryParam("value") String value) {
}
}
我发现了以下问题:
但两者都是关于返回值的验证,而不是关于请求参数的验证。
问题CXF in Karaf: how to configure bean validation on subresources (preferably using Blueprint)?类似,但关于卡拉夫和蓝图,并没有解决我的问题。
答案 0 :(得分:2)
我会说自定义调用程序是要走的路。但我想你已经根据你的评论知道了这一点。
<bean id="validationProvider" class="org.apache.cxf.validation.BeanValidationProvider" />
<bean id="validationInInterceptor"
class="org.apache.cxf.jaxrs.validation.JAXRSBeanValidationInInterceptor">
<property name="provider" ref="validationProvider" />
</bean>
<bean id="validationInvoker" class="org.apache.cxf.jaxrs.validation.JAXRSBeanValidationInvoker"></bean>
<jaxrs:server address="/rest" id="test"
staticSubresourceResolution="true">
<jaxrs:serviceBeans>
<ref bean="testResource" />
</jaxrs:serviceBeans>
<jaxrs:inInterceptors>
<ref bean="validationInInterceptor" />
</jaxrs:inInterceptors>
<jaxrs:invoker>
<ref bean="validationInvoker" />
</jaxrs:invoker>
</jaxrs:server>
我在你的例子上尝试过,它有效。但我认为你需要使用ExceptionMapper
,因为你没有得到适当的例外。我认为ValidationExceptionMapper
是正确的。
编辑 - 基于评论更新了异常映射器类
答案 1 :(得分:1)
我认为bean验证不会对子资源有效,请参考BeanValidationInInterceptor
的源代码,它由JAXRSBeanValidationInInterceptor
扩展并用作验证拦截器。如果您注意到方法handleValidation
,则很明显验证发生在资源对象的参数上。现在,这个验证是在资源上调用实际服务方法之前发生的(至少它是如何查看的),此时,子资源实例是未知的(不存在),所以一般来说,这里的子资源参数关键时刻无法验证。
我的印象是,对于正在讨论的CXF中的当前机制(拦截器),子资源参数可能无法以正常方式进行bean验证。
你可以采取一种解决方法,参考JAX-RS Reference,你可以尝试bean验证参数(特别是如果它是一个请求JSON),手动即实例化Validator对象然后用它验证它例如
package validator;
import java.util.Set;
import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
import javax.validation.constraints.Max;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Pattern;
import org.apache.bval.jsr303.ApacheValidationProvider;
public class DataBean {
@Max(10)
public int x = 0;
@Pattern(regexp= "this_string")
public String y = "this_string"; // also matches null
public static void main(String[] args) {
ValidatorFactory avf =
Validation.byProvider(ApacheValidationProvider.class).configure().buildValidatorFactory();
DataBean bean = new DataBean();
Validator validator = avf.getValidator();
Set<ConstraintViolation<DataBean>> violations = validator.validate(bean);
System.out.println(violations);
}
} // calss closing