java - 带有图像响应的 Spring Boot FeignClient 给出错误 'stream is closed'

标签 java spring rest spring-boot

我的 Spring REST Controller 使用 FeignClient 来消费另一个图像/Blob 存储微服务失败,并出现错误“流已关闭”。使用 blob 数据服务的正确方法是什么。

REST Controller

@RestController
@RequestMapping("/blob")
public class BlobDataRestController  extends BaseRESTController{

    @Autowired
    private IBlobService blobService;

    @CrossOrigin
    @RequestMapping(value = "/{filename:.+}", method = RequestMethod.GET)
     public ResponseEntity<InputStreamResource> getFile(@PathVariable(value = "filename") String filename){
        return blobService.getFile(filename);
    }
}

FeignClient接口(interface)

@FeignClient(name="blob-service", url="${blob-service.url}")
public interface IBlobService {

     @RequestMapping(value = "/blob/{filename:.+}", method = RequestMethod.GET)
     public ResponseEntity<InputStreamResource> getFile(@PathVariable(value = "filename") String filename);
}

调用微服务时出错/blob/filename.jpg

2016-07-27T20:31:41.73+0530 [App/0]      OUT 2016-07-27 15:01:41.732 ERROR 29 --- [io-61779-exec-2] o.a.c.c.C.[.[.[/].[dispatcherServlet]    : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception
2016-07-27T20:31:41.73+0530 [App/0]      OUT java.io.IOException: stream is closed
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at sun.net.www.protocol.http.HttpURLConnection$HttpInputStream.ensureOpen(HttpURLConnection.java:3309) ~[na:1.8.0_91]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at sun.net.www.protocol.http.HttpURLConnection$HttpInputStream.read(HttpURLConnection.java:3334) ~[na:1.8.0_91]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at java.io.FilterInputStream.read(FilterInputStream.java:133) ~[na:1.8.0_91]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at java.io.PushbackInputStream.read(PushbackInputStream.java:186) ~[na:1.8.0_91]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at java.io.FilterInputStream.read(FilterInputStream.java:107) ~[na:1.8.0_91]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.util.StreamUtils.copy(StreamUtils.java:126) ~[spring-core-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.http.converter.ResourceHttpMessageConverter.writeInternal(ResourceHttpMessageConverter.java:102) ~[spring-web-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.http.converter.ResourceHttpMessageConverter.writeInternal(ResourceHttpMessageConverter.java:47) ~[spring-web-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.http.converter.AbstractHttpMessageConverter.write(AbstractHttpMessageConverter.java:195) ~[spring-web-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.annotation.AbstractMessageConverterMethodProcessor.writeWithMessageConverters(AbstractMessageConverterMethodProcessor.java:238) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.annotation.HttpEntityMethodProcessor.handleReturnValue(HttpEntityMethodProcessor.java:183) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.method.support.HandlerMethodReturnValueHandlerComposite.handleReturnValue(HandlerMethodReturnValueHandlerComposite.java:81) ~[spring-web-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:126) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:832) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:743) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:85) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:961) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:895) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]
2016-07-27T20:31:41.73+0530 [App/0]      OUT    at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:967) ~[spring-webmvc-4.2.6.RELEASE.jar!/:4.2.6.RELEASE]

最佳答案

如果我使用带有 byte[] 的 ResponseEntity 作为响应,一切都运行良好:

@FeignClient(name="blob-service", url="${blob-service.url}")
public interface IBlobService {
     @RequestMapping(value = "/blob/{filename:.+}", method = RequestMethod.GET)
     public ResponseEntity<byte[]> getFile(@PathVariable(value = "filename") String filename);
}

但它可能会导致OutOfMemoryError,所以我进行了调查并找到了更好的解决方案:

@FeignClient(name="blob-service", url="${blob-service.url}")
public interface IBlobService {
     @RequestMapping(value = "/blob/{filename:.+}", method = RequestMethod.GET)
     public feign.Response getFile(@PathVariable(value = "filename") String filename);
}

我正在以这种方式使用 blobService:

try (Response response = blobService.getFile(fileName)) {
    InputStream is = response.body().asInputStream();
  // work with is
}

我还在 Feign sources 中找到了如果响应大小超过 8192 字节,则 response.body().asInputStream() 返回原始 sun.net.www.http.KeepAliveStream

if (response.body().length() == null ||
            response.body().length() > MAX_RESPONSE_BUFFER_SIZE) {
    shouldClose = false;
    return response;
}

否则将创建 feign.Response.ByteArrayBody。如果您尝试访问 InputStream it will return ByteArrayInputStream . 所有的东西都解决了可能的 OutOfMemoryError。

关于java - 带有图像响应的 Spring Boot FeignClient 给出错误 'stream is closed',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38616808/

相关文章:

java - 具有多个 ArrayList 的基本 Java 编程

java - 你如何为 MySQL X DevAPI 使用 Java 连接池?

django - Tastypie REST 上的外键 - 模型 '' 具有空属性

javascript - 如何将数据从Java方法发送到javascript函数?

java - 通过 SpringBoot 调用 REST api 时不支持 POST 异常

Java读取字符串的一部分

Java Scanner 接收不存在的输入

java - 如何在Spring中应用MVC概念(非Web应用程序)

java - 如何将 hawtio 与我的 spring 应用程序(和 ActiveMQ)集成

java - 尽管文件位于类路径中,但无法找到资源