使用Swagger文档的异步球衣REST API中的NullPointerException

时间:2014-07-18 20:46:15

标签: java multithreading rest asynchronous jersey-2.0

我在REST API中工作,这是一个Jersey 2.10应用程序,它有一个异步(长轮询)端点和其他几个CRUD操作的正常同步端点。

例如,异步服务如下所示:

@GET
@Path("poll/{groupId}")
@ManagedAsync
public void poll(@Suspended final AsyncResponse asyncResponse, @NotNull @PathParam("groupId") Integer groupId) {
    asyncResponse.setTimeout(SERVICE_TIMEOUT, TimeUnit.SECONDS);

    List<User> users= Collections.emptyList();

    while (users.isEmpty() && asyncResponse.isSuspended()) {
        users= userService.findUsersByGroupId(groupId);

        if (users.isEmpty()) {
            try {
                Thread.sleep(POLL_INTERVAL);
            } catch (InterruptedException ex) {}
        }
    }

    asyncResponse.resume(users.toArray(new TestView[0]));
}   

所有以前工作都很完美,异步请求是在泽西管理线程池的自己的线程中处理的。同步服务按预期工作。

但是有一天我建议团队添加一个文档工具,Swagger非常性感,所以我们安装并添加了一些文档给我们的同步服务。

在那之后,噩梦开始了,我们的服务回复了疯狂的回应。有时它会将响应从一个服务发送到另一个服务,例如运行get用户服务我希望收到一个User对象,但是收到了其他业务对象,如Address或任何其他随机服务,甚至有时候我收到一条html错误消息Tomcat或空响应。

查看我发现的日志:

SEVERE: Error while closing the output stream in order to commit response.
java.lang.NullPointerException
    at org.apache.coyote.http11.InternalOutputBuffer.realWriteBytes(InternalOutputBuffer.java:215)
    at org.apache.tomcat.util.buf.ByteChunk.flushBuffer(ByteChunk.java:480)
    at org.apache.coyote.http11.InternalOutputBuffer.endRequest(InternalOutputBuffer.java:159)
    at org.apache.coyote.http11.AbstractHttp11Processor.action(AbstractHttp11Processor.java:758)
    at org.apache.coyote.Response.action(Response.java:174)
    at org.apache.coyote.Response.finish(Response.java:291)
    at org.apache.catalina.connector.OutputBuffer.close(OutputBuffer.java:320)
    at org.apache.catalina.connector.CoyoteOutputStream.close(CoyoteOutputStream.java:108)
    at org.glassfish.jersey.message.internal.CommittingOutputStream.close(CommittingOutputStream.java:277)
    at org.glassfish.jersey.message.internal.OutboundMessageContext.close(OutboundMessageContext.java:834)
    at org.glassfish.jersey.server.ContainerResponse.close(ContainerResponse.java:411)
    at org.glassfish.jersey.server.ServerRuntime$Responder.writeResponse(ServerRuntime.java:691)
    at org.glassfish.jersey.server.ServerRuntime$Responder.processResponse(ServerRuntime.java:377)
    at org.glassfish.jersey.server.ServerRuntime$Responder.process(ServerRuntime.java:367)
    at org.glassfish.jersey.server.ServerRuntime$AsyncResponder$3.run(ServerRuntime.java:828)
    at org.glassfish.jersey.internal.Errors$1.call(Errors.java:271)
    at org.glassfish.jersey.internal.Errors$1.call(Errors.java:267)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:315)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:297)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:267)
    at org.glassfish.jersey.process.internal.RequestScope.runInScope(RequestScope.java:297)
    at org.glassfish.jersey.server.ServerRuntime$AsyncResponder.resume(ServerRuntime.java:858)
    at org.glassfish.jersey.server.ServerRuntime$AsyncResponder.resume(ServerRuntime.java:820)
    at com.compuware.ruxit.synthetic.api.resource.test.TestDispatcherResource.poll(TestDispatcherResource.java:111)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at org.glassfish.jersey.server.model.internal.ResourceMethodInvocationHandlerFactory$1.invoke(ResourceMethodInvocationHandlerFactory.java:81)
    at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher$1.run(AbstractJavaResourceMethodDispatcher.java:151)
    at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher.invoke(AbstractJavaResourceMethodDispatcher.java:171)
    at org.glassfish.jersey.server.model.internal.JavaResourceMethodDispatcherProvider$VoidOutInvoker.doDispatch(JavaResourceMethodDispatcherProvider.java:136)
    at org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher.dispatch(AbstractJavaResourceMethodDispatcher.java:104)
    at org.glassfish.jersey.server.model.ResourceMethodInvoker.invoke(ResourceMethodInvoker.java:387)
    at org.glassfish.jersey.server.model.ResourceMethodInvoker.access$100(ResourceMethodInvoker.java:103)
    at org.glassfish.jersey.server.model.ResourceMethodInvoker$2.call(ResourceMethodInvoker.java:320)
    at org.glassfish.jersey.server.model.ResourceMethodInvoker$2.call(ResourceMethodInvoker.java:317)
    at org.glassfish.jersey.server.ServerRuntime$AsyncResponder$2$1.run(ServerRuntime.java:791)
    at org.glassfish.jersey.internal.Errors$1.call(Errors.java:271)
    at org.glassfish.jersey.internal.Errors$1.call(Errors.java:267)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:315)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:297)
    at org.glassfish.jersey.internal.Errors.process(Errors.java:267)
    at org.glassfish.jersey.process.internal.RequestScope.runInScope(RequestScope.java:297)
    at org.glassfish.jersey.server.ServerRuntime$AsyncResponder$2.run(ServerRuntime.java:787)Nul
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:471)
    at java.util.concurrent.FutureTask.run(FutureTask.java:262)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
    at java.lang.Thread.run(Thread.java:744)

我最初认为它与新的servlet定义有关,可以让Swagger工作:

<servlet>
    <servlet-name>SwaggerConfig</servlet-name>
    <servlet-class>com.compuware.ruxit.synthetic.api.configuration.SwaggerJaxrsConfig</servlet-class>
    <load-on-startup>2</load-on-startup>
</servlet>

我尝试将<async-supported>true</async-supported>添加到SwaggerConfig servlet和Jersey servlet定义中(当你使用ManagedAsync注释时我不需要这个,但我试了一下)。但它没有帮助。

1 个答案:

答案 0 :(得分:1)

发现我正在使用的swagger maven依赖:

    <dependency>
        <groupId>com.wordnik</groupId>
        <artifactId>swagger-jersey2-jaxrs_2.10</artifactId>
        <version>1.3.5</version>
    </dependency>

依赖于jersey-container-servlet-core,但是对于比我在我的球衣依赖中已有的旧版本,它是2.1,而我使用的是2.10。 Maven然后使用它并覆盖我的球衣应用程序需要的那个。

我所做的只是排除传递依赖:

    <dependency>
        <groupId>com.wordnik</groupId>
        <artifactId>swagger-jersey2-jaxrs_2.10</artifactId>
        <version>1.3.5</version>
        <exclusions>
            <exclusion>
                <artifactId>jersey-container-servlet-core</artifactId>
                <groupId>org.glassfish.jersey.containers</groupId>
            </exclusion>
        </exclusions>
    </dependency>

在更新之后,到目前为止还没有例外。