在GZIPInputStream中包装BodySubscriber <inputstream>导致挂起

时间:2018-11-19 16:39:04

标签: java gzipinputstream java-http-client

我正在使用新的java.net.http类来处理异步HTTP请求和响应交换,并且我试图找到一种方法使BodySubscriber处理不同的编码类型,例如gzip。

但是,映射BodySubsriber<InputStream>使得基础流被GZIPInputStream包装(当在响应标头中找到“ Content-Encoding:gzip”时)会导致挂起。没有例外,只是完全停止了活动。

映射BodySubscriber的代码如下:

private HttpResponse.BodySubscriber<InputStream> gzippedBodySubscriber(
        HttpResponse.ResponseInfo responseInfo) {
    return HttpResponse.BodySubscribers.mapping(
            HttpResponse.BodySubscribers.ofInputStream(),
            this::decodeGzipStream);
}

private InputStream decodeGzipStream(InputStream gzippedStream) {
    System.out.println("Entered decodeGzipStream method.");
    try {
        InputStream decodedStream = new GZIPInputStream(gzippedStream);
        System.out.println(
                "Created GZIPInputStream to handle response body stream.");
        return decodedStream;
    } catch (IOException ex) {
        System.out.println("IOException occurred while trying to create GZIPInputStream.");
        throw new UncheckedIOException(ex);
    }
}

接收具有“ gzip”编码的HTTP响应会导致控制台显示以下内容:

  

输入了EncodedBodyHandler.apply方法。
  输入了encodeGzipStream方法。

什么也看不到,因此GZIPInputStream构造函数调用之后的行将永远不会执行。

有人知道为什么将InputStream中的BodySubscriber<InputStream>包装到GZIPInputStream中的尝试挂起了吗?

注意:未编码(原始文本)HTTP响应正文的等效方法仅包含对BodySubscribers.ofInputStream()的调用,而没有任何映射,因此可以毫无问题地接收和显示响应。

2 个答案:

答案 0 :(得分:2)

这确实是一个错误。我已登录JDK-8217264。我可以建议两个解决方法:

一种解决方法

请勿使用BodySubscribers.mapping-在获取HttpResponse的正文后,将InputStream转换为GZIPInputStream

GZIPInputStream gzin = new GZIPInputStream(resp.getBody());

解决方法二

让映射函数返回Supplier<InputStream>,注意不要在调用GZIPInputStream之前创建Supplier::get

static final class ISS implements Supplier<InputStream> {
    final InputStream in;
    GZIPInputStream gz;
    ISS(InputStream in) {
        this.in = in;
    }
    public synchronized InputStream get() {
        if (gz == null) {
            try {
                gz = new GZIPInputStream(in);
            } catch (IOException t) {
                throw new UncheckedIOException(t);
            }
        }
        return gz;
    }
}

答案 1 :(得分:1)

遇到了完全相同的问题。我在BodySubscribers.mapping方法的Javadoc中尝试了该示例。行为相同,应用程序挂起,没有任何错误。

可能是一个错误,因为这是Javadoc中的官方示例。

  public static <W> BodySubscriber<W> asJSON(Class<W> targetType) {
     BodySubscriber<InputStream> upstream = BodySubscribers.ofInputStream();

     BodySubscriber<W> downstream = BodySubscribers.mapping(
           upstream,
           (InputStream is) -> {
               try (InputStream stream = is) {
                   ObjectMapper objectMapper = new ObjectMapper();
                   return objectMapper.readValue(stream, targetType);
               } catch (IOException e) {
                   throw new UncheckedIOException(e);
               }
           });
    return downstream;
 } }