在分块数据的 HTTP 响应中如何设置 Content-Length

In HTTP response for a chunked data how to set Content-Length

我们编写了一个服务,它将一些编码数据分块发送到需要设置 Content-Length header 的代理服务,以便它可以向端点发送正确的响应。即使我设置了 Content-Length header 它仍然会作为对客户端的响应的一部分被剥离。 下面是设置 header

的代码
func HTTPSuccessResponse(rw http.ResponseWriter, bufferLen int, media []byte) {
        rw.WriteHeader(http.StatusOK)

        rw.Header().Set("Content-Type", "opus/ogg; audio/ogg; codec=opus")
        length := strconv.Itoa(len(media));
        rw.Header().Set("Content-Length", length)
        rw.Write(media)
}

下面是我尝试使用 curl 请求时得到的响应

bash-4.2# curl -v -X GET -k -H  -i 'http://127.0.0.1:8090/preview'
* About to connect() to 127.0.0.1 port 8090 (#0)
*   Trying 127.0.0.1...
* Connected to 127.0.0.1 (127.0.0.1) port 8090 (#0)
> GET /preview HTTP/1.1
> User-Agent: curl/7.29.0
> Host: 127.0.0.1:8090
> Accept: */*
>
< HTTP/1.1 200 OK
< Date: Tue, 14 May 2019 13:08:20 GMT
< Content-Type: text/plain; charset=utf-8
< Transfer-Encoding: chunked
<

我正在使用 Gorrila Mux 库来设置 HTTP 服务器。任何想法如何使 header 作为响应的一部分。

删除顶部的 WriteHeader 调用。您只能将 headers 写入一次响应。调用 WriteHeader 后,您将无法再设置 headers.

Per the ResponseWriter documentation:

    // Changing the header map after a call to WriteHeader (or
    // Write) has no effect unless the modified headers are
    // trailers.

所以不能先调用;但您也根本不需要调用它 - 来自相同的文档:

    // If WriteHeader is not called explicitly, the first call to Write
    // will trigger an implicit WriteHeader(http.StatusOK).
    // Thus explicit calls to WriteHeader are mainly used to
    // send error codes.