Bump google.golang.org/api
This commit is contained in:
290
vendor/github.com/google/martian/messageview/messageview.go
generated
vendored
Normal file
290
vendor/github.com/google/martian/messageview/messageview.go
generated
vendored
Normal file
@@ -0,0 +1,290 @@
|
||||
// Copyright 2015 Google Inc. All rights reserved.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
// Package messageview provides no-op snapshots for HTTP requests and
|
||||
// responses.
|
||||
package messageview
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"compress/flate"
|
||||
"compress/gzip"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"net/http"
|
||||
"net/http/httputil"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// MessageView is a static view of an HTTP request or response.
|
||||
type MessageView struct {
|
||||
message []byte
|
||||
cts []string
|
||||
chunked bool
|
||||
skipBody bool
|
||||
compress string
|
||||
bodyoffset int64
|
||||
traileroffset int64
|
||||
}
|
||||
|
||||
type config struct {
|
||||
decode bool
|
||||
}
|
||||
|
||||
// Option is a configuration option for a MessageView.
|
||||
type Option func(*config)
|
||||
|
||||
// Decode sets an option to decode the message body for logging purposes.
|
||||
func Decode() Option {
|
||||
return func(c *config) {
|
||||
c.decode = true
|
||||
}
|
||||
}
|
||||
|
||||
// New returns a new MessageView.
|
||||
func New() *MessageView {
|
||||
return &MessageView{}
|
||||
}
|
||||
|
||||
// SkipBody will skip reading the body when the view is loaded with a request
|
||||
// or response.
|
||||
func (mv *MessageView) SkipBody(skipBody bool) {
|
||||
mv.skipBody = skipBody
|
||||
}
|
||||
|
||||
// SkipBodyUnlessContentType will skip reading the body unless the
|
||||
// Content-Type matches one in cts.
|
||||
func (mv *MessageView) SkipBodyUnlessContentType(cts ...string) {
|
||||
mv.skipBody = true
|
||||
mv.cts = cts
|
||||
}
|
||||
|
||||
// SnapshotRequest reads the request into the MessageView. If mv.skipBody is false
|
||||
// it will also read the body into memory and replace the existing body with
|
||||
// the in-memory copy. This method is semantically a no-op.
|
||||
func (mv *MessageView) SnapshotRequest(req *http.Request) error {
|
||||
buf := new(bytes.Buffer)
|
||||
|
||||
fmt.Fprintf(buf, "%s %s HTTP/%d.%d\r\n", req.Method,
|
||||
req.URL, req.ProtoMajor, req.ProtoMinor)
|
||||
|
||||
if req.Host != "" {
|
||||
fmt.Fprintf(buf, "Host: %s\r\n", req.Host)
|
||||
}
|
||||
|
||||
if tec := len(req.TransferEncoding); tec > 0 {
|
||||
mv.chunked = req.TransferEncoding[tec-1] == "chunked"
|
||||
fmt.Fprintf(buf, "Transfer-Encoding: %s\r\n", strings.Join(req.TransferEncoding, ", "))
|
||||
}
|
||||
if !mv.chunked && req.ContentLength >= 0 {
|
||||
fmt.Fprintf(buf, "Content-Length: %d\r\n", req.ContentLength)
|
||||
}
|
||||
|
||||
mv.compress = req.Header.Get("Content-Encoding")
|
||||
|
||||
req.Header.WriteSubset(buf, map[string]bool{
|
||||
"Host": true,
|
||||
"Content-Length": true,
|
||||
"Transfer-Encoding": true,
|
||||
})
|
||||
|
||||
fmt.Fprint(buf, "\r\n")
|
||||
|
||||
mv.bodyoffset = int64(buf.Len())
|
||||
mv.traileroffset = int64(buf.Len())
|
||||
|
||||
ct := req.Header.Get("Content-Type")
|
||||
if mv.skipBody && !mv.matchContentType(ct) || req.Body == nil {
|
||||
mv.message = buf.Bytes()
|
||||
return nil
|
||||
}
|
||||
|
||||
data, err := ioutil.ReadAll(req.Body)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Body.Close()
|
||||
|
||||
if mv.chunked {
|
||||
cw := httputil.NewChunkedWriter(buf)
|
||||
cw.Write(data)
|
||||
cw.Close()
|
||||
} else {
|
||||
buf.Write(data)
|
||||
}
|
||||
|
||||
mv.traileroffset = int64(buf.Len())
|
||||
|
||||
req.Body = ioutil.NopCloser(bytes.NewReader(data))
|
||||
|
||||
if req.Trailer != nil {
|
||||
req.Trailer.Write(buf)
|
||||
} else if mv.chunked {
|
||||
fmt.Fprint(buf, "\r\n")
|
||||
}
|
||||
|
||||
mv.message = buf.Bytes()
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// SnapshotResponse reads the response into the MessageView. If mv.headersOnly
|
||||
// is false it will also read the body into memory and replace the existing
|
||||
// body with the in-memory copy. This method is semantically a no-op.
|
||||
func (mv *MessageView) SnapshotResponse(res *http.Response) error {
|
||||
buf := new(bytes.Buffer)
|
||||
|
||||
fmt.Fprintf(buf, "HTTP/%d.%d %s\r\n", res.ProtoMajor, res.ProtoMinor, res.Status)
|
||||
|
||||
if tec := len(res.TransferEncoding); tec > 0 {
|
||||
mv.chunked = res.TransferEncoding[tec-1] == "chunked"
|
||||
fmt.Fprintf(buf, "Transfer-Encoding: %s\r\n", strings.Join(res.TransferEncoding, ", "))
|
||||
}
|
||||
if !mv.chunked && res.ContentLength >= 0 {
|
||||
fmt.Fprintf(buf, "Content-Length: %d\r\n", res.ContentLength)
|
||||
}
|
||||
|
||||
mv.compress = res.Header.Get("Content-Encoding")
|
||||
// Do not uncompress if we have don't have the full contents.
|
||||
if res.StatusCode == http.StatusNoContent || res.StatusCode == http.StatusPartialContent {
|
||||
mv.compress = ""
|
||||
}
|
||||
|
||||
res.Header.WriteSubset(buf, map[string]bool{
|
||||
"Content-Length": true,
|
||||
"Transfer-Encoding": true,
|
||||
})
|
||||
|
||||
fmt.Fprint(buf, "\r\n")
|
||||
|
||||
mv.bodyoffset = int64(buf.Len())
|
||||
mv.traileroffset = int64(buf.Len())
|
||||
|
||||
ct := res.Header.Get("Content-Type")
|
||||
if mv.skipBody && !mv.matchContentType(ct) || res.Body == nil {
|
||||
mv.message = buf.Bytes()
|
||||
return nil
|
||||
}
|
||||
|
||||
data, err := ioutil.ReadAll(res.Body)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
res.Body.Close()
|
||||
|
||||
if mv.chunked {
|
||||
cw := httputil.NewChunkedWriter(buf)
|
||||
cw.Write(data)
|
||||
cw.Close()
|
||||
} else {
|
||||
buf.Write(data)
|
||||
}
|
||||
|
||||
mv.traileroffset = int64(buf.Len())
|
||||
|
||||
res.Body = ioutil.NopCloser(bytes.NewReader(data))
|
||||
|
||||
if res.Trailer != nil {
|
||||
res.Trailer.Write(buf)
|
||||
} else if mv.chunked {
|
||||
fmt.Fprint(buf, "\r\n")
|
||||
}
|
||||
|
||||
mv.message = buf.Bytes()
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Reader returns the an io.ReadCloser that reads the full HTTP message.
|
||||
func (mv *MessageView) Reader(opts ...Option) (io.ReadCloser, error) {
|
||||
hr := mv.HeaderReader()
|
||||
br, err := mv.BodyReader(opts...)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
tr := mv.TrailerReader()
|
||||
|
||||
return struct {
|
||||
io.Reader
|
||||
io.Closer
|
||||
}{
|
||||
Reader: io.MultiReader(hr, br, tr),
|
||||
Closer: br,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// HeaderReader returns an io.Reader that reads the HTTP Status-Line or
|
||||
// HTTP Request-Line and headers.
|
||||
func (mv *MessageView) HeaderReader() io.Reader {
|
||||
r := bytes.NewReader(mv.message)
|
||||
return io.NewSectionReader(r, 0, mv.bodyoffset)
|
||||
}
|
||||
|
||||
// BodyReader returns an io.ReadCloser that reads the HTTP request or response
|
||||
// body. If mv.skipBody was set the reader will immediately return io.EOF.
|
||||
//
|
||||
// If the Decode option is passed the body will be unchunked if
|
||||
// Transfer-Encoding is set to "chunked", and will decode the following
|
||||
// Content-Encodings: gzip, deflate.
|
||||
func (mv *MessageView) BodyReader(opts ...Option) (io.ReadCloser, error) {
|
||||
var r io.Reader
|
||||
|
||||
conf := &config{}
|
||||
for _, o := range opts {
|
||||
o(conf)
|
||||
}
|
||||
|
||||
br := bytes.NewReader(mv.message)
|
||||
r = io.NewSectionReader(br, mv.bodyoffset, mv.traileroffset-mv.bodyoffset)
|
||||
|
||||
if !conf.decode {
|
||||
return ioutil.NopCloser(r), nil
|
||||
}
|
||||
|
||||
if mv.chunked {
|
||||
r = httputil.NewChunkedReader(r)
|
||||
}
|
||||
switch mv.compress {
|
||||
case "gzip":
|
||||
gr, err := gzip.NewReader(r)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return gr, nil
|
||||
case "deflate":
|
||||
return flate.NewReader(r), nil
|
||||
default:
|
||||
return ioutil.NopCloser(r), nil
|
||||
}
|
||||
}
|
||||
|
||||
// TrailerReader returns an io.Reader that reads the HTTP request or response
|
||||
// trailers, if present.
|
||||
func (mv *MessageView) TrailerReader() io.Reader {
|
||||
r := bytes.NewReader(mv.message)
|
||||
end := int64(len(mv.message)) - mv.traileroffset
|
||||
|
||||
return io.NewSectionReader(r, mv.traileroffset, end)
|
||||
}
|
||||
|
||||
func (mv *MessageView) matchContentType(mct string) bool {
|
||||
for _, ct := range mv.cts {
|
||||
if strings.HasPrefix(mct, ct) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
759
vendor/github.com/google/martian/messageview/messageview_test.go
generated
vendored
Normal file
759
vendor/github.com/google/martian/messageview/messageview_test.go
generated
vendored
Normal file
@@ -0,0 +1,759 @@
|
||||
// Copyright 2015 Google Inc. All rights reserved.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
package messageview
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"bytes"
|
||||
"compress/flate"
|
||||
"compress/gzip"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"net/http"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/google/martian/proxyutil"
|
||||
)
|
||||
|
||||
func TestRequestViewHeadersOnly(t *testing.T) {
|
||||
body := strings.NewReader("body content")
|
||||
req, err := http.NewRequest("GET", "http://example.com/path?k=v", body)
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.ContentLength = int64(body.Len())
|
||||
req.Header.Set("Request-Header", "true")
|
||||
|
||||
mv := New()
|
||||
mv.SkipBody(true)
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "GET http://example.com/path?k=v HTTP/1.1\r\n" +
|
||||
"Host: example.com\r\n" +
|
||||
"Content-Length: 12\r\n" +
|
||||
"Request-Header: true\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if _, err := br.Read(nil); err != io.EOF {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want io.EOF", err)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRequestView(t *testing.T) {
|
||||
body := strings.NewReader("body content")
|
||||
req, err := http.NewRequest("GET", "http://example.com/path?k=v", body)
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.Header.Set("Request-Header", "true")
|
||||
|
||||
// Force Content Length to be unset to simulate lack of Content-Length and
|
||||
// Transfer-Encoding which is valid.
|
||||
req.ContentLength = -1
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "GET http://example.com/path?k=v HTTP/1.1\r\n" +
|
||||
"Host: example.com\r\n" +
|
||||
"Request-Header: true\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
|
||||
// Sanity check to ensure it still parses.
|
||||
if _, err := http.ReadRequest(bufio.NewReader(bytes.NewReader(got))); err != nil {
|
||||
t.Fatalf("http.ReadRequest(): got %v, want no error", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRequestViewSkipBodyUnlessContentType(t *testing.T) {
|
||||
req, err := http.NewRequest("GET", "http://example.com", strings.NewReader("body content"))
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.ContentLength = 12
|
||||
req.Header.Set("Content-Type", "text/plain; charset=utf-8")
|
||||
|
||||
mv := New()
|
||||
mv.SkipBodyUnlessContentType("text/plain")
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
req.Header.Set("Content-Type", "image/png")
|
||||
mv = New()
|
||||
mv.SkipBodyUnlessContentType("text/plain")
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
br, err = mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if _, err := br.Read(nil); err != io.EOF {
|
||||
t.Fatalf("br.Read(): got %v, want io.EOF", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRequestViewChunkedTransferEncoding(t *testing.T) {
|
||||
req, err := http.NewRequest("GET", "http://example.com/path?k=v", strings.NewReader("body content"))
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.TransferEncoding = []string{"chunked"}
|
||||
req.Header.Set("Trailer", "Trailer-Header")
|
||||
req.Trailer = http.Header{
|
||||
"Trailer-Header": []string{"true"},
|
||||
}
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "GET http://example.com/path?k=v HTTP/1.1\r\n" +
|
||||
"Host: example.com\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Trailer: Trailer-Header\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "c\r\nbody content\r\n0\r\n"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(mv.TrailerReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.TrailerReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
trailerwant := "Trailer-Header: true\r\n"
|
||||
if !bytes.Equal(got, []byte(trailerwant)) {
|
||||
t.Fatalf("mv.TrailerReader(): got %q, want %q", got, trailerwant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + trailerwant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
|
||||
// Sanity check to ensure it still parses.
|
||||
if _, err := http.ReadRequest(bufio.NewReader(bytes.NewReader(got))); err != nil {
|
||||
t.Fatalf("http.ReadRequest(): got %v, want no error", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRequestViewDecodeGzipContentEncoding(t *testing.T) {
|
||||
body := new(bytes.Buffer)
|
||||
gw := gzip.NewWriter(body)
|
||||
gw.Write([]byte("body content"))
|
||||
gw.Flush()
|
||||
gw.Close()
|
||||
|
||||
req, err := http.NewRequest("GET", "http://example.com/path?k=v", body)
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.TransferEncoding = []string{"chunked"}
|
||||
req.Header.Set("Content-Encoding", "gzip")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "GET http://example.com/path?k=v HTTP/1.1\r\n" +
|
||||
"Host: example.com\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Content-Encoding: gzip\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, wt o error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + "\r\n"); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRequestViewDecodeDeflateContentEncoding(t *testing.T) {
|
||||
body := new(bytes.Buffer)
|
||||
dw, err := flate.NewWriter(body, -1)
|
||||
if err != nil {
|
||||
t.Fatalf("flate.NewWriter(): got %v, want no error", err)
|
||||
}
|
||||
dw.Write([]byte("body content"))
|
||||
dw.Flush()
|
||||
dw.Close()
|
||||
|
||||
req, err := http.NewRequest("GET", "http://example.com/path?k=v", body)
|
||||
if err != nil {
|
||||
t.Fatalf("http.NewRequest(): got %v, want no error", err)
|
||||
}
|
||||
req.TransferEncoding = []string{"chunked"}
|
||||
req.Header.Set("Content-Encoding", "deflate")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotRequest(req); err != nil {
|
||||
t.Fatalf("SnapshotRequest(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "GET http://example.com/path?k=v HTTP/1.1\r\n" +
|
||||
"Host: example.com\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Content-Encoding: deflate\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + "\r\n"); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewHeadersOnly(t *testing.T) {
|
||||
body := strings.NewReader("body content")
|
||||
res := proxyutil.NewResponse(200, body, nil)
|
||||
res.ContentLength = 12
|
||||
res.Header.Set("Response-Header", "true")
|
||||
|
||||
mv := New()
|
||||
mv.SkipBody(true)
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "HTTP/1.1 200 OK\r\n" +
|
||||
"Content-Length: 12\r\n" +
|
||||
"Response-Header: true\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if _, err := br.Read(nil); err != io.EOF {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want io.EOF", err)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseView(t *testing.T) {
|
||||
body := strings.NewReader("body content")
|
||||
res := proxyutil.NewResponse(200, body, nil)
|
||||
res.ContentLength = 12
|
||||
res.Header.Set("Response-Header", "true")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "HTTP/1.1 200 OK\r\n" +
|
||||
"Content-Length: 12\r\n" +
|
||||
"Response-Header: true\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
|
||||
// Sanity check to ensure it still parses.
|
||||
if _, err := http.ReadResponse(bufio.NewReader(bytes.NewReader(got)), nil); err != nil {
|
||||
t.Fatalf("http.ReadResponse(): got %v, want no error", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewSkipBodyUnlessContentType(t *testing.T) {
|
||||
res := proxyutil.NewResponse(200, strings.NewReader("body content"), nil)
|
||||
res.ContentLength = 12
|
||||
res.Header.Set("Content-Type", "text/plain; charset=utf-8")
|
||||
|
||||
mv := New()
|
||||
mv.SkipBodyUnlessContentType("text/plain")
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
res.Header.Set("Content-Type", "image/png")
|
||||
mv = New()
|
||||
mv.SkipBodyUnlessContentType("text/plain")
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
br, err = mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if _, err := br.Read(nil); err != io.EOF {
|
||||
t.Fatalf("br.Read(): got %v, want io.EOF", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewChunkedTransferEncoding(t *testing.T) {
|
||||
body := strings.NewReader("body content")
|
||||
res := proxyutil.NewResponse(200, body, nil)
|
||||
res.TransferEncoding = []string{"chunked"}
|
||||
res.Header.Set("Trailer", "Trailer-Header")
|
||||
res.Trailer = http.Header{
|
||||
"Trailer-Header": []string{"true"},
|
||||
}
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "HTTP/1.1 200 OK\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Trailer: Trailer-Header\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
bodywant := "c\r\nbody content\r\n0\r\n"
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(mv.TrailerReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.TrailerReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
trailerwant := "Trailer-Header: true\r\n"
|
||||
if !bytes.Equal(got, []byte(trailerwant)) {
|
||||
t.Fatalf("mv.TrailerReader(): got %q, want %q", got, trailerwant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader()
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + trailerwant); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
|
||||
// Sanity check to ensure it still parses.
|
||||
if _, err := http.ReadResponse(bufio.NewReader(bytes.NewReader(got)), nil); err != nil {
|
||||
t.Fatalf("http.ReadResponse(): got %v, want no error", err)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewDecodeGzipContentEncoding(t *testing.T) {
|
||||
body := new(bytes.Buffer)
|
||||
gw := gzip.NewWriter(body)
|
||||
gw.Write([]byte("body content"))
|
||||
gw.Flush()
|
||||
gw.Close()
|
||||
|
||||
res := proxyutil.NewResponse(200, body, nil)
|
||||
res.TransferEncoding = []string{"chunked"}
|
||||
res.Header.Set("Content-Encoding", "gzip")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "HTTP/1.1 200 OK\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Content-Encoding: gzip\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, wt o error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + "\r\n"); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewDecodeGzipContentEncodingPartial(t *testing.T) {
|
||||
bodywant := "partial content"
|
||||
res := proxyutil.NewResponse(206, strings.NewReader(bodywant), nil)
|
||||
res.TransferEncoding = []string{"chunked"}
|
||||
res.Header.Set("Content-Encoding", "gzip")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
br, err := mv.BodyReader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, wt o error", err)
|
||||
}
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
}
|
||||
|
||||
func TestResponseViewDecodeDeflateContentEncoding(t *testing.T) {
|
||||
body := new(bytes.Buffer)
|
||||
dw, err := flate.NewWriter(body, -1)
|
||||
if err != nil {
|
||||
t.Fatalf("flate.NewWriter(): got %v, want no error", err)
|
||||
}
|
||||
dw.Write([]byte("body content"))
|
||||
dw.Flush()
|
||||
dw.Close()
|
||||
|
||||
res := proxyutil.NewResponse(200, body, nil)
|
||||
res.TransferEncoding = []string{"chunked"}
|
||||
res.Header.Set("Content-Encoding", "deflate")
|
||||
|
||||
mv := New()
|
||||
if err := mv.SnapshotResponse(res); err != nil {
|
||||
t.Fatalf("SnapshotResponse(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err := ioutil.ReadAll(mv.HeaderReader())
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.HeaderReader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
hdrwant := "HTTP/1.1 200 OK\r\n" +
|
||||
"Transfer-Encoding: chunked\r\n" +
|
||||
"Content-Encoding: deflate\r\n\r\n"
|
||||
|
||||
if !bytes.Equal(got, []byte(hdrwant)) {
|
||||
t.Fatalf("mv.HeaderReader(): got %q, want %q", got, hdrwant)
|
||||
}
|
||||
|
||||
br, err := mv.BodyReader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.BodyReader(): got %v, want no error", err)
|
||||
}
|
||||
|
||||
got, err = ioutil.ReadAll(br)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.BodyReader()): got %v, wt o error", err)
|
||||
}
|
||||
|
||||
bodywant := "body content"
|
||||
|
||||
if !bytes.Equal(got, []byte(bodywant)) {
|
||||
t.Fatalf("mv.BodyReader(): got %q, want %q", got, bodywant)
|
||||
}
|
||||
|
||||
r, err := mv.Reader(Decode())
|
||||
if err != nil {
|
||||
t.Fatalf("mv.Reader(): got %v, want no error", err)
|
||||
}
|
||||
got, err = ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
t.Fatalf("ioutil.ReadAll(mv.Reader()): got %v, want no error", err)
|
||||
}
|
||||
|
||||
if want := []byte(hdrwant + bodywant + "\r\n"); !bytes.Equal(got, want) {
|
||||
t.Fatalf("mv.Read(): got %q, want %q", got, want)
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user