在 Apache TomEE 8.0.10 上使用 CDI 时,Jax-RS 资源上的 Bean 验证停止工作
Bean Validation on Jax-RS Resource stops working while using CDI on Apache TomEE 8.0.10
我在使 bean 验证与仅由这三个 java 文件加上 pom.xml 组成的以下最小化项目一起工作时遇到了麻烦。我正在使用 Apache TomEE 8.0.10。
LoginMessage.java
package org.example;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
import javax.validation.constraints.NotBlank;
@Getter
@Setter
@ToString
public class LoginMessage {
@NotBlank
private String username;
@NotBlank
private String password;
}
SessionService.java
package org.example;
import lombok.extern.java.Log;
import javax.enterprise.context.RequestScoped;
@Log
@RequestScoped
public class SessionService {
public void login(final LoginMessage loginMessage) {
log.info(loginMessage.toString());
}
}
SessionController.java
package org.example;
import lombok.extern.java.Log;
import javax.inject.Inject;
import javax.validation.Valid;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.core.MediaType;
@Log
@Path("/session")
public class SessionController {
@Inject
private SessionService sessionService;
@POST
@Consumes(MediaType.APPLICATION_JSON)
public void postLoginMessage(@Valid final LoginMessage loginMessage) {
sessionService.login(loginMessage);
}
}
pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<packaging>war</packaging>
<groupId>org.example</groupId>
<artifactId>beanval</artifactId>
<version>1.0-SNAPSHOT</version>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>11</source>
<target>11</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>3.3.2</version>
</plugin>
</plugins>
</build>
<dependencies>
<dependency>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
<version>8.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.22</version>
</dependency>
</dependencies>
</project>
如果你 post 一个空的 JSON 对象,它会忽略 SessionController#postLoginMessage() 中的 @Valid
注释,并直接输出包含 toString( ) 通过 SessionService#login() 方法获取 LoginMessage 对象的内容。
POST http://localhost:8080/beanval-1.0-SNAPSHOT/session
Content-Type: application/json
{
}
13-Mar-2022 01:30:39.700 INFORMATION [http-nio-8080-exec-6] SessionService.login LoginMessage(username=null, password=null)
如果您在重启 TomEE 后删除或注释掉 SessionService 中的 @RequestScoped
注释和 post 空的 JSON-Object,则 bean 验证工作并记录:
13-Mar-2022 01:52:51.594 WARNUNG [http-nio-8080-exec-6] org.apache.cxf.jaxrs.validation.ValidationExceptionMapper.toResponse Value (null) of SessionController.postLoginMessage.arg0.password: must not be blank
13-Mar-2022 01:52:51.595 WARNUNG [http-nio-8080-exec-6] org.apache.cxf.jaxrs.validation.ValidationExceptionMapper.toResponse Value (null) of SessionController.postLoginMessage.arg0.username: must not be blank
我想在 JAX-RS 资源中结合使用 CDI 和 Bean-Validation。
我做错了什么?
提前致谢。
这似乎是 OpenWebBeans 或 TomEE 中的错误。因此,首先,bean 的实际实例由 JAX-RS 管理,其次,bean 由 CDI 容器管理。在第二种情况下,需要有某种拦截器来调用 Bean 验证框架。
我会在邮件列表上发起讨论,并在 JIRA 上打开一个错误。如果您可以创建一个重现该问题的示例项目,它将极大地帮助开发人员。
作为解决方法,您可以 @Inject private Validator validator
如果返回任何约束违规,throw new ConstraintViolationException(constraintViolations);
。
毕竟要找到某种解决方案,我将停止在控制器层使用 bean 验证。它在服务层工作,所以我可以继续在我的网络应用程序上工作。
解决方案是在 SessionService#login() 方法中使用 @Valid
注释,并将其从 SessionController#postLoginMessage() 方法中删除。
如果它确实是 TomEE 中的一个错误,也可以使用另一个应用程序服务器,直到它被修复。
我在使 bean 验证与仅由这三个 java 文件加上 pom.xml 组成的以下最小化项目一起工作时遇到了麻烦。我正在使用 Apache TomEE 8.0.10。
LoginMessage.java
package org.example;
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
import javax.validation.constraints.NotBlank;
@Getter
@Setter
@ToString
public class LoginMessage {
@NotBlank
private String username;
@NotBlank
private String password;
}
SessionService.java
package org.example;
import lombok.extern.java.Log;
import javax.enterprise.context.RequestScoped;
@Log
@RequestScoped
public class SessionService {
public void login(final LoginMessage loginMessage) {
log.info(loginMessage.toString());
}
}
SessionController.java
package org.example;
import lombok.extern.java.Log;
import javax.inject.Inject;
import javax.validation.Valid;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.core.MediaType;
@Log
@Path("/session")
public class SessionController {
@Inject
private SessionService sessionService;
@POST
@Consumes(MediaType.APPLICATION_JSON)
public void postLoginMessage(@Valid final LoginMessage loginMessage) {
sessionService.login(loginMessage);
}
}
pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<packaging>war</packaging>
<groupId>org.example</groupId>
<artifactId>beanval</artifactId>
<version>1.0-SNAPSHOT</version>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>11</source>
<target>11</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>3.3.2</version>
</plugin>
</plugins>
</build>
<dependencies>
<dependency>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
<version>8.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.22</version>
</dependency>
</dependencies>
</project>
如果你 post 一个空的 JSON 对象,它会忽略 SessionController#postLoginMessage() 中的 @Valid
注释,并直接输出包含 toString( ) 通过 SessionService#login() 方法获取 LoginMessage 对象的内容。
POST http://localhost:8080/beanval-1.0-SNAPSHOT/session
Content-Type: application/json
{
}
13-Mar-2022 01:30:39.700 INFORMATION [http-nio-8080-exec-6] SessionService.login LoginMessage(username=null, password=null)
如果您在重启 TomEE 后删除或注释掉 SessionService 中的 @RequestScoped
注释和 post 空的 JSON-Object,则 bean 验证工作并记录:
13-Mar-2022 01:52:51.594 WARNUNG [http-nio-8080-exec-6] org.apache.cxf.jaxrs.validation.ValidationExceptionMapper.toResponse Value (null) of SessionController.postLoginMessage.arg0.password: must not be blank
13-Mar-2022 01:52:51.595 WARNUNG [http-nio-8080-exec-6] org.apache.cxf.jaxrs.validation.ValidationExceptionMapper.toResponse Value (null) of SessionController.postLoginMessage.arg0.username: must not be blank
我想在 JAX-RS 资源中结合使用 CDI 和 Bean-Validation。 我做错了什么?
提前致谢。
这似乎是 OpenWebBeans 或 TomEE 中的错误。因此,首先,bean 的实际实例由 JAX-RS 管理,其次,bean 由 CDI 容器管理。在第二种情况下,需要有某种拦截器来调用 Bean 验证框架。
我会在邮件列表上发起讨论,并在 JIRA 上打开一个错误。如果您可以创建一个重现该问题的示例项目,它将极大地帮助开发人员。
作为解决方法,您可以 @Inject private Validator validator
如果返回任何约束违规,throw new ConstraintViolationException(constraintViolations);
。
毕竟要找到某种解决方案,我将停止在控制器层使用 bean 验证。它在服务层工作,所以我可以继续在我的网络应用程序上工作。
解决方案是在 SessionService#login() 方法中使用 @Valid
注释,并将其从 SessionController#postLoginMessage() 方法中删除。
如果它确实是 TomEE 中的一个错误,也可以使用另一个应用程序服务器,直到它被修复。