Commit aae81d94 authored by Brad Fitzpatrick's avatar Brad Fitzpatrick

net/http: start of making all relevant tests test both http1 and http2

This CL adds skipped failing tests, showing differences between HTTP/1
and HTTP/2 behavior. They'll be fixed in later commits.

Only a tiny fraction of the net/http tests have been split into their
"_h1" and "_h2" variants. That will also continue. (help welcome)

Updates #6891
Updates #13315
Updates #13316
Updates #13317

Change-Id: I16c3c381dbe267a3098fb266ab0d804c36473a64
Reviewed-on: https://go-review.googlesource.com/17046
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
Reviewed-by: default avatarAndrew Gerrand <adg@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent e4a1acce
...@@ -133,4 +133,8 @@ var ExportErrRequestCanceled = errRequestCanceled ...@@ -133,4 +133,8 @@ var ExportErrRequestCanceled = errRequestCanceled
var ExportServeFile = serveFile var ExportServeFile = serveFile
var ExportHttp2ConfigureTransport = http2ConfigureTransport
var ExportHttp2ConfigureServer = http2ConfigureServer
func SetTestHookServerServe(fn func(*Server, net.Listener)) { testHookServerServe = fn } func SetTestHookServerServe(fn func(*Server, net.Listener)) { testHookServerServe = fn }
...@@ -5,8 +5,10 @@ ...@@ -5,8 +5,10 @@
package http_test package http_test
import ( import (
"crypto/tls"
"fmt" "fmt"
"net/http" "net/http"
"net/http/httptest"
"os" "os"
"runtime" "runtime"
"sort" "sort"
...@@ -111,3 +113,43 @@ func afterTest(t testing.TB) { ...@@ -111,3 +113,43 @@ func afterTest(t testing.TB) {
} }
t.Errorf("Test appears to have leaked %s:\n%s", bad, stacks) t.Errorf("Test appears to have leaked %s:\n%s", bad, stacks)
} }
type clientServerTest struct {
t *testing.T
h2 bool
h http.Handler
ts *httptest.Server
tr *http.Transport
c *http.Client
}
func (t *clientServerTest) close() {
t.tr.CloseIdleConnections()
t.ts.Close()
}
func newClientServerTest(t *testing.T, h2 bool, h http.Handler) *clientServerTest {
cst := &clientServerTest{
t: t,
h2: h2,
h: h,
tr: &http.Transport{},
}
cst.c = &http.Client{Transport: cst.tr}
if !h2 {
cst.ts = httptest.NewServer(h)
return cst
}
cst.ts = httptest.NewUnstartedServer(h)
http.ExportHttp2ConfigureServer(cst.ts.Config, nil)
cst.ts.TLS = cst.ts.Config.TLSConfig
cst.ts.StartTLS()
cst.tr.TLSClientConfig = &tls.Config{
InsecureSkipVerify: true,
}
if err := http.ExportHttp2ConfigureTransport(cst.tr); err != nil {
t.Fatal(err)
}
return cst
}
...@@ -857,19 +857,24 @@ func TestServerAllowsBlockingRemoteAddr(t *testing.T) { ...@@ -857,19 +857,24 @@ func TestServerAllowsBlockingRemoteAddr(t *testing.T) {
} }
} }
func TestChunkedResponseHeaders(t *testing.T) { func TestChunkedResponseHeaders_h1(t *testing.T) { testChunkedResponseHeaders(t, false) }
func TestChunkedResponseHeaders_h2(t *testing.T) { testChunkedResponseHeaders(t, true) }
func testChunkedResponseHeaders(t *testing.T, h2 bool) {
if h2 {
t.Skip("known failing test; golang.org/issue/13316")
}
defer afterTest(t) defer afterTest(t)
log.SetOutput(ioutil.Discard) // is noisy otherwise log.SetOutput(ioutil.Discard) // is noisy otherwise
defer log.SetOutput(os.Stderr) defer log.SetOutput(os.Stderr)
cst := newClientServerTest(t, h2, HandlerFunc(func(w ResponseWriter, r *Request) {
ts := httptest.NewServer(HandlerFunc(func(w ResponseWriter, r *Request) {
w.Header().Set("Content-Length", "intentional gibberish") // we check that this is deleted w.Header().Set("Content-Length", "intentional gibberish") // we check that this is deleted
w.(Flusher).Flush() w.(Flusher).Flush()
fmt.Fprintf(w, "I am a chunked response.") fmt.Fprintf(w, "I am a chunked response.")
})) }))
defer ts.Close() defer cst.close()
res, err := Get(ts.URL) res, err := cst.c.Get(cst.ts.URL)
if err != nil { if err != nil {
t.Fatalf("Get error: %v", err) t.Fatalf("Get error: %v", err)
} }
...@@ -880,8 +885,8 @@ func TestChunkedResponseHeaders(t *testing.T) { ...@@ -880,8 +885,8 @@ func TestChunkedResponseHeaders(t *testing.T) {
if g, e := res.TransferEncoding, []string{"chunked"}; !reflect.DeepEqual(g, e) { if g, e := res.TransferEncoding, []string{"chunked"}; !reflect.DeepEqual(g, e) {
t.Errorf("expected TransferEncoding of %v; got %v", e, g) t.Errorf("expected TransferEncoding of %v; got %v", e, g)
} }
if _, haveCL := res.Header["Content-Length"]; haveCL { if got, haveCL := res.Header["Content-Length"]; haveCL {
t.Errorf("Unexpected Content-Length") t.Errorf("Unexpected Content-Length: %q", got)
} }
} }
...@@ -914,22 +919,52 @@ func TestIdentityResponseHeaders(t *testing.T) { ...@@ -914,22 +919,52 @@ func TestIdentityResponseHeaders(t *testing.T) {
} }
} }
// Testing the newClientServerTest helper.
func TestNewClientServerTest(t *testing.T) {
var got struct {
sync.Mutex
log []string
}
h := HandlerFunc(func(w ResponseWriter, r *Request) {
got.Lock()
defer got.Unlock()
got.log = append(got.log, r.Proto)
})
for _, v := range [2]bool{false, true} {
cst := newClientServerTest(t, v, h)
if _, err := cst.c.Head(cst.ts.URL); err != nil {
t.Fatal(err)
}
cst.close()
}
got.Lock() // no need to unlock
if want := []string{"HTTP/1.1", "HTTP/2.0"}; !reflect.DeepEqual(got.log, want) {
t.Errorf("got %q; want %q", got.log, want)
}
}
// Test304Responses verifies that 304s don't declare that they're // Test304Responses verifies that 304s don't declare that they're
// chunking in their response headers and aren't allowed to produce // chunking in their response headers and aren't allowed to produce
// output. // output.
func Test304Responses(t *testing.T) { func Test304Responses_h1(t *testing.T) { test304Responses(t, false) }
func Test304Responses_h2(t *testing.T) { test304Responses(t, true) }
func test304Responses(t *testing.T, h2 bool) {
if h2 {
t.Skip("known failing test; golang.org/issue/13317")
}
defer afterTest(t) defer afterTest(t)
ts := httptest.NewServer(HandlerFunc(func(w ResponseWriter, r *Request) { cst := newClientServerTest(t, h2, HandlerFunc(func(w ResponseWriter, r *Request) {
w.WriteHeader(StatusNotModified) w.WriteHeader(StatusNotModified)
_, err := w.Write([]byte("illegal body")) _, err := w.Write([]byte("illegal body"))
if err != ErrBodyNotAllowed { if err != ErrBodyNotAllowed {
t.Errorf("on Write, expected ErrBodyNotAllowed, got %v", err) t.Errorf("on Write, expected ErrBodyNotAllowed, got %v", err)
} }
})) }))
defer ts.Close() defer cst.close()
res, err := Get(ts.URL) res, err := cst.c.Get(cst.ts.URL)
if err != nil { if err != nil {
t.Error(err) t.Fatal(err)
} }
if len(res.TransferEncoding) > 0 { if len(res.TransferEncoding) > 0 {
t.Errorf("expected no TransferEncoding; got %v", res.TransferEncoding) t.Errorf("expected no TransferEncoding; got %v", res.TransferEncoding)
......
...@@ -2844,6 +2844,70 @@ func TestTransportPrefersResponseOverWriteError(t *testing.T) { ...@@ -2844,6 +2844,70 @@ func TestTransportPrefersResponseOverWriteError(t *testing.T) {
} }
} }
func TestTransportResponse_h12(t *testing.T) {
t.Skip("known failing test; golang.org/issue/13315")
tests := []Handler{
HandlerFunc(func(w ResponseWriter, r *Request) {
// no body.
}),
HandlerFunc(func(w ResponseWriter, r *Request) {
io.WriteString(w, "small body")
}),
HandlerFunc(func(w ResponseWriter, r *Request) {
w.Header().Set("Content-Length", "3") // w/ content length
io.WriteString(w, "foo")
}),
HandlerFunc(func(w ResponseWriter, r *Request) {
w.(Flusher).Flush()
io.WriteString(w, "foo")
}),
}
handlerc := make(chan Handler, 1)
testHandler := HandlerFunc(func(w ResponseWriter, r *Request) {
(<-handlerc).ServeHTTP(w, r)
})
normalizeRes := func(res *Response, wantProto string) {
if res.Proto == wantProto {
res.Proto, res.ProtoMajor, res.ProtoMinor = "", 0, 0
} else {
t.Errorf("got %q response; want %q", res.Proto, wantProto)
}
slurp, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err != nil {
t.Errorf("ReadAll(Body) = %v", err)
}
res.Body = ioutil.NopCloser(bytes.NewReader(slurp))
}
cst1 := newClientServerTest(t, false, testHandler)
defer cst1.close()
cst2 := newClientServerTest(t, true, testHandler)
defer cst2.close()
for i, h := range tests {
handlerc <- h
res1, err := cst1.c.Get(cst1.ts.URL)
if err != nil {
t.Errorf("%d. HTTP/1 get: %v", i, err)
continue
}
normalizeRes(res1, "HTTP/1.1")
handlerc <- h
res2, err := cst2.c.Get(cst2.ts.URL)
if err != nil {
t.Errorf("%d. HTTP/2 get: %v", i, err)
continue
}
normalizeRes(res2, "HTTP/2.0")
if !reflect.DeepEqual(res1, res2) {
t.Errorf("\nhttp/1 (%v): %#v\nhttp/2 (%v): %#v", cst1.ts.URL, res1, cst2.ts.URL, res2)
}
}
}
func wantBody(res *Response, err error, want string) error { func wantBody(res *Response, err error, want string) error {
if err != nil { if err != nil {
return err return err
......
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