Grizzly JAX-RS没有返回400 Bad Request

时间:2017-02-09 06:30:01

标签: java rest maven jax-rs grizzly

我正在使用JAX-RS开发REST API。我跟着this tutorial,现在我正在运行应用程序。但我的URL路径有问题。 Grizzly在main方法中自动创建了BASE_URI,我在这里添加了自己的路径:

// Base URI the Grizzly HTTP server will listen on
public static final String BASE_URI = "http://localhost:8080/app/api/1.0

如果用户输入的BASE_URI错误,例如"http://localhost:8080/ap/ap/1.0/path/to/myResourse/123" Grizzly返回

Not Found
Resource identified by path '/app/api/1.0/whatever/the/user/entered, does not exist.
Grizzly 2.3.28.

问题是,如果用户输入BASE_URI正确,但输入的资源路径错误,Grizzly就不会显示"资源未找到&#34 ;消息,但只显示一个HTTP标题为404的空白屏幕。

那么如何显示400 Bad Request,告诉用户他向无效网址发出了请求?如何更改Grizzly提供的默认错误消息?

我已经尝试过搜索创建自定义错误消息,包括ExceptionMappers,但我认为这不是我正在搜索的内容。我能想到的一个解决方案是在URL路径中为每个/创建一个新类,但这不是一个非常优雅的方法......?

下面是我的Resource类,它连接到另一个REST API,从中获取资源,然后我在API中显示

@Path("/path/to/myResourse")
public class ResourceService {

  @GET
  @Path("{id}")
  @Produces(MediaType.APPLICATION_JSON + "; charset=UTF-8")
  public Response getBuildingSite(@PathParam("id") String id) throws Exception {

    StringBuilder sb = new StringBuilder();
    sb.append("https://www.exmaple.com/rest/api/resources");
    sb.append(id);
    sb.append(".xml");
    String url = sb.toString();

    try {
      Resource resource = Connector.fetch(Resource.class, url);
      return JSONMapper.asOkJsonResponse(resource);
    } catch (Exception e) {
      return JSONMapper.asErrorJsonResponse(
        new ErrorResponse(404,"Resource '" + id + "' not found"));
    }
  }
}

我的pom.xml文件

    <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/maven-v4_0_0.xsd">

  <modelVersion>4.0.0</modelVersion>

  <groupId>com.example.app.exampleApp</groupId>
  <artifactId>exampleApp</artifactId>
  <packaging>jar</packaging>
  <version>1.0-SNAPSHOT</version>
  <name>exampleApp</name>

  <dependencyManagement>
    <dependencies>
      <dependency>
        <groupId>org.glassfish.jersey</groupId>
        <artifactId>jersey-bom</artifactId>
        <version>${jersey.version}</version>
        <type>pom</type>
        <scope>import</scope>
      </dependency>
    </dependencies>
  </dependencyManagement>

  <dependencies>
    <dependency>
      <groupId>org.glassfish.jersey.containers</groupId>
      <artifactId>jersey-container-grizzly2-http</artifactId>
    </dependency>
    <dependency>
      <groupId>org.glassfish.jersey.media</groupId>
      <artifactId>jersey-media-json-jackson</artifactId>
    </dependency>
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.9</version>
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>org.mariadb.jdbc</groupId>
      <artifactId>mariadb-java-client</artifactId>
      <version>1.5.7</version>
    </dependency>
  </dependencies>

  <build>
    <plugins>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-compiler-plugin</artifactId>
        <version>2.5.1</version>
        <inherited>true</inherited>
        <configuration>
          <source>1.8</source>
          <target>1.8</target>
        </configuration>
      </plugin>
      <plugin>
        <groupId>org.codehaus.mojo</groupId>
        <artifactId>exec-maven-plugin</artifactId>
        <version>1.2.1</version>
        <executions>
          <execution>
            <goals>
              <goal>java</goal>
            </goals>
          </execution>
        </executions>
        <configuration>
          <mainClass>com.example.app.exampleApp.Main</mainClass>
        </configuration>
      </plugin>
    </plugins>
  </build>

  <properties>
    <jersey.version>2.25</jersey.version>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  </properties>
</project>

1 个答案:

答案 0 :(得分:0)

我自己找到了解决问题的方法。我发现我可以使用由具有匹配所有字符串的正则表达式的路径注释的资源。我对它进行了测试,发现只有在找不到其他“工作”资源的情况下,灰熊才会匹配它。

感谢peeskillet's answer derabbink's question,我发现了这个想法。

以下是我创建的资源:

@Path("{any: .*}")
public class NotFoundService {

  @GET
  @Produces(MediaType.APPLICATION_JSON + ";charset=UTF-8")
  public Response getNotFound(@Context UriInfo uriInfo) {
    // My custom response where I can use uriInfo to tell what went wrong
  }
}

现在,如果用户输入的路径具有正确的BASE URI,但资源路径无效,例如"http://localhost:8080/app/api/1.0"/invalid/path它会返回我在自定义Response中添加的内容。