如何在传输编码分块时使用HttpsUrlconnection读取响应

时间:2016-02-28 00:55:13

标签: java http servlets urlconnection httpsurlconnection

我正在编写一个servlet,它在内部使用HttpsUrlConnection来调用另一个Url。从servlet,我必须返回从HttpsUrlConnection调用收到的相同响应和相同的响应头。

为此,我使用getHeaderFields()读取所有标题:将返回的标题复制到我的servlet的Httpservletresponse对象。

此外,之后我尝试使用connection.getErrorStream()读取响应(这主要发生在服务器返回状态> 400时),这就是我使用getErrorStream的原因。然后我读取字节并将字节复制到我的servlet的HttpServletResponse输出流。

现在这种情况很正常。

但是当我使用连接调用的服务器返回带有传输编码的响应时,那么当我调用我的servlet API时,如果我将响应和响应头复制到Httpservlet响应,它将返回无效块编码的错误。

在这种情况下,在日志中我看到当我使用getHeaderFields读取头文件时,我看到的第一个头是传输编码块,然后下一个头名称为null,它的值为HTTP 1/1。

另外,我看到的反应也不合适。

如何从httpsurlconnection中读取响应头和响应体,并从我的servlet中正确返回它?

1 个答案:

答案 0 :(得分:1)

我遇到了与分块编码类似的问题。与您不同的是,我的servlet在内部使用HTTP而不是HTTPS。

我也使用一种方法来复制标题字段。但是我提供了一个由servlet容器管理的头文件列表,所以我不创建,复制或编辑它们:

 private static final Set forbiddenCopyHeaders = new HashSet<>(Arrays.asList(new String[]{
            "connection"
            , "transfer-encoding"
            , "content-length"
            , "via"
            , "x-forwarded-for"
            , "x-forwarded-host"
            , "x-forwarded-server"
    }));

这些是我用于复制响应标头的方法:

  private void copyResponseHeaders(CloseableHttpResponse internResponse, HttpServletResponse response)
    {
        Header[] headers = internResponse.getAllHeaders();
        Header[] connHeaders = internResponse.getHeaders("connection");
        StringBuilder connectionValue = new StringBuilder();

        for (Header connHeader : connHeaders)
        {
            connectionValue.append(connHeader.getValue()).append(", ");
        }

        for (Header header : headers)
        {
            String headerName = header.getName();

            boolean copyAllowed = !forbiddenCopyHeaders.contains(headerName.toLowerCase())
                    && !StringUtils.containsIgnoreCase(connectionValue.toString(), headerName);

            if (copyAllowed)
            {
                if (response.containsHeader(headerName))
                {
                    response.addHeader(headerName, header.getValue());
                }
                else
                {
                    response.setHeader(headerName, header.getValue());
                }
            }
        }

        setViaHeader(internResponse, response);
    }

方法setViaHeader():

private void setViaHeader(CloseableHttpResponse response, HttpServletResponse customerResponse)
    {
        String serverHostName = "companyServer";
        try
        {
            serverHostName = InetAddress.getLocalHost().getHostName();
        }
        catch (UnknownHostException e)
        {
            logger.error("für den VIA-Header kann der Hostname nicht ermittelt werden", e);
            System.err.println("für den VIA-Header kann der Hostname nicht ermittelt werden: " +
                    ExceptionUtils.getStackTrace(e));
        }

        Header[] originalViaHeaders = response.getHeaders("via");
        StringBuilder via = new StringBuilder("");
        if ((originalViaHeaders != null) && (originalViaHeaders.length > 0))
        {
            for (Header viaHeader : originalViaHeaders)
            {
                via.append(viaHeader.getValue()).append(", ");
            }
        }
        via.append(response.getStatusLine().getProtocolVersion().toString()).append(" ").append(serverHostName);

        customerResponse.setHeader("via", via.toString());
    }