mirror of
https://github.com/golang/go
synced 2024-11-02 11:50:30 +00:00
http: support Trailers in ReadRequest
Available after closing Request.Body. R=adg, rsc CC=golang-dev https://golang.org/cl/5348041
This commit is contained in:
parent
aac144b120
commit
b14ee23f9b
3 changed files with 119 additions and 23 deletions
|
@ -9,19 +9,22 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
"url"
|
"url"
|
||||||
)
|
)
|
||||||
|
|
||||||
type reqTest struct {
|
type reqTest struct {
|
||||||
Raw string
|
Raw string
|
||||||
Req *Request
|
Req *Request
|
||||||
Body string
|
Body string
|
||||||
Error string
|
Trailer Header
|
||||||
|
Error string
|
||||||
}
|
}
|
||||||
|
|
||||||
var noError = ""
|
var noError = ""
|
||||||
var noBody = ""
|
var noBody = ""
|
||||||
|
var noTrailer Header = nil
|
||||||
|
|
||||||
var reqTests = []reqTest{
|
var reqTests = []reqTest{
|
||||||
// Baseline test; All Request fields included for template use
|
// Baseline test; All Request fields included for template use
|
||||||
|
@ -72,6 +75,7 @@ var reqTests = []reqTest{
|
||||||
|
|
||||||
"abcdef\n",
|
"abcdef\n",
|
||||||
|
|
||||||
|
noTrailer,
|
||||||
noError,
|
noError,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -97,6 +101,7 @@ var reqTests = []reqTest{
|
||||||
},
|
},
|
||||||
|
|
||||||
noBody,
|
noBody,
|
||||||
|
noTrailer,
|
||||||
noError,
|
noError,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -130,6 +135,7 @@ var reqTests = []reqTest{
|
||||||
},
|
},
|
||||||
|
|
||||||
noBody,
|
noBody,
|
||||||
|
noTrailer,
|
||||||
noError,
|
noError,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -139,6 +145,7 @@ var reqTests = []reqTest{
|
||||||
"Host: test\r\n\r\n",
|
"Host: test\r\n\r\n",
|
||||||
nil,
|
nil,
|
||||||
noBody,
|
noBody,
|
||||||
|
noTrailer,
|
||||||
"parse ../../../../etc/passwd: invalid URI for request",
|
"parse ../../../../etc/passwd: invalid URI for request",
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -148,8 +155,42 @@ var reqTests = []reqTest{
|
||||||
"Host: test\r\n\r\n",
|
"Host: test\r\n\r\n",
|
||||||
nil,
|
nil,
|
||||||
noBody,
|
noBody,
|
||||||
|
noTrailer,
|
||||||
"parse : empty url",
|
"parse : empty url",
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// Tests chunked body with trailer:
|
||||||
|
{
|
||||||
|
"POST / HTTP/1.1\r\n" +
|
||||||
|
"Host: foo.com\r\n" +
|
||||||
|
"Transfer-Encoding: chunked\r\n\r\n" +
|
||||||
|
"3\r\nfoo\r\n" +
|
||||||
|
"3\r\nbar\r\n" +
|
||||||
|
"0\r\n" +
|
||||||
|
"Trailer-Key: Trailer-Value\r\n" +
|
||||||
|
"\r\n",
|
||||||
|
&Request{
|
||||||
|
Method: "POST",
|
||||||
|
URL: &url.URL{
|
||||||
|
Raw: "/",
|
||||||
|
Path: "/",
|
||||||
|
RawPath: "/",
|
||||||
|
},
|
||||||
|
TransferEncoding: []string{"chunked"},
|
||||||
|
Proto: "HTTP/1.1",
|
||||||
|
ProtoMajor: 1,
|
||||||
|
ProtoMinor: 1,
|
||||||
|
ContentLength: -1,
|
||||||
|
Host: "foo.com",
|
||||||
|
Form: url.Values{},
|
||||||
|
},
|
||||||
|
|
||||||
|
"foobar",
|
||||||
|
Header{
|
||||||
|
"Trailer-Key": {"Trailer-Value"},
|
||||||
|
},
|
||||||
|
noError,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestReadRequest(t *testing.T) {
|
func TestReadRequest(t *testing.T) {
|
||||||
|
@ -169,12 +210,18 @@ func TestReadRequest(t *testing.T) {
|
||||||
diff(t, fmt.Sprintf("#%d Request", i), req, tt.Req)
|
diff(t, fmt.Sprintf("#%d Request", i), req, tt.Req)
|
||||||
var bout bytes.Buffer
|
var bout bytes.Buffer
|
||||||
if rbody != nil {
|
if rbody != nil {
|
||||||
io.Copy(&bout, rbody)
|
_, err := io.Copy(&bout, rbody)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("#%d. copying body: %v", i, err)
|
||||||
|
}
|
||||||
rbody.Close()
|
rbody.Close()
|
||||||
}
|
}
|
||||||
body := bout.String()
|
body := bout.String()
|
||||||
if body != tt.Body {
|
if body != tt.Body {
|
||||||
t.Errorf("#%d: Body = %q want %q", i, body, tt.Body)
|
t.Errorf("#%d: Body = %q want %q", i, body, tt.Body)
|
||||||
}
|
}
|
||||||
|
if !reflect.DeepEqual(tt.Trailer, req.Trailer) {
|
||||||
|
t.Errorf("%#d. Trailers differ.\n got: %v\nwant: %v", i, req.Trailer, tt.Trailer)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -142,6 +142,8 @@ type Request struct {
|
||||||
// Trailer maps trailer keys to values. Like for Header, if the
|
// Trailer maps trailer keys to values. Like for Header, if the
|
||||||
// response has multiple trailer lines with the same key, they will be
|
// response has multiple trailer lines with the same key, they will be
|
||||||
// concatenated, delimited by commas.
|
// concatenated, delimited by commas.
|
||||||
|
// For server requests, Trailer is only populated after Body has been
|
||||||
|
// closed or fully consumed.
|
||||||
// Trailer support is only partially complete.
|
// Trailer support is only partially complete.
|
||||||
Trailer Header
|
Trailer Header
|
||||||
|
|
||||||
|
@ -464,16 +466,6 @@ func (cr *chunkedReader) beginChunk() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if cr.n == 0 {
|
if cr.n == 0 {
|
||||||
// trailer CRLF
|
|
||||||
for {
|
|
||||||
line, cr.err = readLine(cr.r)
|
|
||||||
if cr.err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if line == "" {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
cr.err = io.EOF
|
cr.err = io.EOF
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"net/textproto"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
@ -532,7 +533,68 @@ func (b *body) Read(p []byte) (n int, err error) {
|
||||||
if b.closed {
|
if b.closed {
|
||||||
return 0, ErrBodyReadAfterClose
|
return 0, ErrBodyReadAfterClose
|
||||||
}
|
}
|
||||||
return b.Reader.Read(p)
|
n, err = b.Reader.Read(p)
|
||||||
|
|
||||||
|
// Read the final trailer once we hit EOF.
|
||||||
|
if err == io.EOF && b.hdr != nil {
|
||||||
|
err = b.readTrailer()
|
||||||
|
b.hdr = nil
|
||||||
|
}
|
||||||
|
return n, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
singleCRLF = []byte("\r\n")
|
||||||
|
doubleCRLF = []byte("\r\n\r\n")
|
||||||
|
)
|
||||||
|
|
||||||
|
func seeUpcomingDoubleCRLF(r *bufio.Reader) bool {
|
||||||
|
for peekSize := 4; ; peekSize++ {
|
||||||
|
// This loop stops when Peek returns an error,
|
||||||
|
// which it does when r's buffer has been filled.
|
||||||
|
buf, err := r.Peek(peekSize)
|
||||||
|
if bytes.HasSuffix(buf, doubleCRLF) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *body) readTrailer() error {
|
||||||
|
// The common case, since nobody uses trailers.
|
||||||
|
buf, _ := b.r.Peek(2)
|
||||||
|
if bytes.Equal(buf, singleCRLF) {
|
||||||
|
b.r.ReadByte()
|
||||||
|
b.r.ReadByte()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure there's a header terminator coming up, to prevent
|
||||||
|
// a DoS with an unbounded size Trailer. It's not easy to
|
||||||
|
// slip in a LimitReader here, as textproto.NewReader requires
|
||||||
|
// a concrete *bufio.Reader. Also, we can't get all the way
|
||||||
|
// back up to our conn's LimitedReader that *might* be backing
|
||||||
|
// this bufio.Reader. Instead, a hack: we iteratively Peek up
|
||||||
|
// to the bufio.Reader's max size, looking for a double CRLF.
|
||||||
|
// This limits the trailer to the underlying buffer size, typically 4kB.
|
||||||
|
if !seeUpcomingDoubleCRLF(b.r) {
|
||||||
|
return errors.New("http: suspiciously long trailer after chunked body")
|
||||||
|
}
|
||||||
|
|
||||||
|
hdr, err := textproto.NewReader(b.r).ReadMIMEHeader()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
switch rr := b.hdr.(type) {
|
||||||
|
case *Request:
|
||||||
|
rr.Trailer = Header(hdr)
|
||||||
|
case *Response:
|
||||||
|
rr.Trailer = Header(hdr)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *body) Close() error {
|
func (b *body) Close() error {
|
||||||
|
@ -557,15 +619,10 @@ func (b *body) Close() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Fully consume the body, which will also lead to us reading
|
||||||
|
// the trailer headers after the body, if present.
|
||||||
if _, err := io.Copy(ioutil.Discard, b); err != nil {
|
if _, err := io.Copy(ioutil.Discard, b); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if b.hdr == nil { // not reading trailer
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO(petar): Put trailer reader code here
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue