Commit bdb5e9d1 authored by Harshavardhana's avatar Harshavardhana Committed by Brad Fitzpatrick

net/http/httputil: fix missing Transfer-Encoding header

Current implementation of httputil.DumpRequestOut
incorrectly resets the Request.Body prematurely
before Content-Length/Transfer-Encoding detection
in newTransferWriter()

This fix avoids resetting the Request.Body when
Request.ContentLength is set to '0' by the caller
and Request.Body is set to a custom reader. To allow
newTransferWriter() to treat this situation as
'Transfer-Encoding: chunked'.

Fixes #34504

Change-Id: Ieab6bf876ced28c32c084e0f4c8c4432964181f5
Reviewed-on: https://go-review.googlesource.com/c/go/+/197898Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent afe50c11
...@@ -24,7 +24,7 @@ import ( ...@@ -24,7 +24,7 @@ import (
// It returns an error if the initial slurp of all bytes fails. It does not attempt // It returns an error if the initial slurp of all bytes fails. It does not attempt
// to make the returned ReadClosers have identical error-matching behavior. // to make the returned ReadClosers have identical error-matching behavior.
func drainBody(b io.ReadCloser) (r1, r2 io.ReadCloser, err error) { func drainBody(b io.ReadCloser) (r1, r2 io.ReadCloser, err error) {
if b == http.NoBody { if b == nil || b == http.NoBody {
// No copying needed. Preserve the magic sentinel meaning of NoBody. // No copying needed. Preserve the magic sentinel meaning of NoBody.
return http.NoBody, http.NoBody, nil return http.NoBody, http.NoBody, nil
} }
...@@ -60,16 +60,28 @@ func (b neverEnding) Read(p []byte) (n int, err error) { ...@@ -60,16 +60,28 @@ func (b neverEnding) Read(p []byte) (n int, err error) {
return len(p), nil return len(p), nil
} }
// outGoingLength is a copy of the unexported
// (*http.Request).outgoingLength method.
func outgoingLength(req *http.Request) int64 {
if req.Body == nil || req.Body == http.NoBody {
return 0
}
if req.ContentLength != 0 {
return req.ContentLength
}
return -1
}
// DumpRequestOut is like DumpRequest but for outgoing client requests. It // DumpRequestOut is like DumpRequest but for outgoing client requests. It
// includes any headers that the standard http.Transport adds, such as // includes any headers that the standard http.Transport adds, such as
// User-Agent. // User-Agent.
func DumpRequestOut(req *http.Request, body bool) ([]byte, error) { func DumpRequestOut(req *http.Request, body bool) ([]byte, error) {
save := req.Body save := req.Body
dummyBody := false dummyBody := false
if !body || req.Body == nil { if !body {
req.Body = nil contentLength := outgoingLength(req)
if req.ContentLength != 0 { if contentLength != 0 {
req.Body = ioutil.NopCloser(io.LimitReader(neverEnding('x'), req.ContentLength)) req.Body = ioutil.NopCloser(io.LimitReader(neverEnding('x'), contentLength))
dummyBody = true dummyBody = true
} }
} else { } else {
......
...@@ -17,6 +17,12 @@ import ( ...@@ -17,6 +17,12 @@ import (
"testing" "testing"
) )
type eofReader struct{}
func (n eofReader) Close() error { return nil }
func (n eofReader) Read([]byte) (int, error) { return 0, io.EOF }
type dumpTest struct { type dumpTest struct {
// Either Req or GetReq can be set/nil but not both. // Either Req or GetReq can be set/nil but not both.
Req *http.Request Req *http.Request
...@@ -204,6 +210,29 @@ var dumpTests = []dumpTest{ ...@@ -204,6 +210,29 @@ var dumpTests = []dumpTest{
"Content-Length: 0\r\n" + "Content-Length: 0\r\n" +
"Accept-Encoding: gzip\r\n\r\n", "Accept-Encoding: gzip\r\n\r\n",
}, },
// Issue 34504: a non-nil Body without ContentLength set should be chunked
{
Req: &http.Request{
Method: "PUT",
URL: &url.URL{
Scheme: "http",
Host: "post.tld",
Path: "/test",
},
ContentLength: 0,
Proto: "HTTP/1.1",
ProtoMajor: 1,
ProtoMinor: 1,
Body: &eofReader{},
},
NoBody: true,
WantDumpOut: "PUT /test HTTP/1.1\r\n" +
"Host: post.tld\r\n" +
"User-Agent: Go-http-client/1.1\r\n" +
"Transfer-Encoding: chunked\r\n" +
"Accept-Encoding: gzip\r\n\r\n",
},
} }
func TestDumpRequest(t *testing.T) { func TestDumpRequest(t *testing.T) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment