Commit 2566e21f authored by Brad Fitzpatrick's avatar Brad Fitzpatrick

net/http: support disabling built-in HTTP/2 with a new build tag

Fixes #35082
Updates #6853

Change-Id: I4eeb0e15f534cff57fefb6039cd33fadf15b946e
Reviewed-on: https://go-review.googlesource.com/c/go/+/205139Reviewed-by: default avatarEmmanuel Odeke <emm.odeke@gmail.com>
Reviewed-by: default avatarDavid Crawshaw <crawshaw@golang.org>
parent 74af7fc6
...@@ -83,6 +83,9 @@ func optWithServerLog(lg *log.Logger) func(*httptest.Server) { ...@@ -83,6 +83,9 @@ func optWithServerLog(lg *log.Logger) func(*httptest.Server) {
} }
func newClientServerTest(t *testing.T, h2 bool, h Handler, opts ...interface{}) *clientServerTest { func newClientServerTest(t *testing.T, h2 bool, h Handler, opts ...interface{}) *clientServerTest {
if h2 {
CondSkipHTTP2(t)
}
cst := &clientServerTest{ cst := &clientServerTest{
t: t, t: t,
h2: h2, h2: h2,
......
...@@ -60,6 +60,12 @@ func init() { ...@@ -60,6 +60,12 @@ func init() {
} }
} }
func CondSkipHTTP2(t *testing.T) {
if omitBundledHTTP2 {
t.Skip("skipping HTTP/2 test when nethttpomithttp2 build tag in use")
}
}
var ( var (
SetEnterRoundTripHook = hookSetter(&testHookEnterRoundTrip) SetEnterRoundTripHook = hookSetter(&testHookEnterRoundTrip)
SetRoundTripRetried = hookSetter(&testHookRoundTripRetried) SetRoundTripRetried = hookSetter(&testHookRoundTripRetried)
......
// +build !nethttpomithttp2
// Code generated by golang.org/x/tools/cmd/bundle. DO NOT EDIT. // Code generated by golang.org/x/tools/cmd/bundle. DO NOT EDIT.
//go:generate bundle -o h2_bundle.go -prefix http2 golang.org/x/net/http2 // $ bundle -o=h2_bundle.go -prefix=http2 -tags=!nethttpomithttp2 golang.org/x/net/http2
// Package http2 implements the HTTP/2 protocol. // Package http2 implements the HTTP/2 protocol.
// //
......
...@@ -2,6 +2,8 @@ ...@@ -2,6 +2,8 @@
// Use of this source code is governed by a BSD-style // Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file. // license that can be found in the LICENSE file.
//go:generate bundle -o=h2_bundle.go -prefix=http2 -tags=!nethttpomithttp2 golang.org/x/net/http2
package http package http
import ( import (
...@@ -22,6 +24,11 @@ const maxInt64 = 1<<63 - 1 ...@@ -22,6 +24,11 @@ const maxInt64 = 1<<63 - 1
// immediate cancellation of network operations. // immediate cancellation of network operations.
var aLongTimeAgo = time.Unix(1, 0) var aLongTimeAgo = time.Unix(1, 0)
// omitBundledHTTP2 is set by omithttp2.go when the nethttpomithttp2
// build tag is set. That means h2_bundle.go isn't compiled in and we
// shouldn't try to use it.
var omitBundledHTTP2 bool
// TODO(bradfitz): move common stuff here. The other files have accumulated // TODO(bradfitz): move common stuff here. The other files have accumulated
// generic http stuff in random places. // generic http stuff in random places.
......
...@@ -111,6 +111,32 @@ func TestCmdGoNoHTTPServer(t *testing.T) { ...@@ -111,6 +111,32 @@ func TestCmdGoNoHTTPServer(t *testing.T) {
} }
} }
// Tests that the nethttpomithttp2 build tag doesn't rot too much,
// even if there's not a regular builder on it.
func TestOmitHTTP2(t *testing.T) {
if testing.Short() {
t.Skip("skipping in short mode")
}
t.Parallel()
goTool := testenv.GoToolPath(t)
out, err := exec.Command(goTool, "test", "-short", "-tags=nethttpomithttp2", "net/http").CombinedOutput()
if err != nil {
t.Fatalf("go test -short failed: %v, %s", err, out)
}
}
// Tests that the nethttpomithttp2 build tag at least type checks
// in short mode.
// The TestOmitHTTP2 test above actually runs tests (in long mode).
func TestOmitHTTP2Vet(t *testing.T) {
t.Parallel()
goTool := testenv.GoToolPath(t)
out, err := exec.Command(goTool, "vet", "-tags=nethttpomithttp2", "net/http").CombinedOutput()
if err != nil {
t.Fatalf("go vet failed: %v, %s", err, out)
}
}
var valuesCount int var valuesCount int
func BenchmarkCopyValues(b *testing.B) { func BenchmarkCopyValues(b *testing.B) {
......
...@@ -90,6 +90,9 @@ func goroutineLeaked() bool { ...@@ -90,6 +90,9 @@ func goroutineLeaked() bool {
// (all.bash), but as a serial test otherwise. Using t.Parallel isn't // (all.bash), but as a serial test otherwise. Using t.Parallel isn't
// compatible with the afterTest func in non-short mode. // compatible with the afterTest func in non-short mode.
func setParallel(t *testing.T) { func setParallel(t *testing.T) {
if strings.Contains(t.Name(), "HTTP2") {
http.CondSkipHTTP2(t)
}
if testing.Short() { if testing.Short() {
t.Parallel() t.Parallel()
} }
......
// Copyright 2019 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// +build nethttpomithttp2
package http
import (
"errors"
"sync"
"time"
)
func init() {
omitBundledHTTP2 = true
}
const noHTTP2 = "no bundled HTTP/2" // should never see this
var http2errRequestCanceled = errors.New("net/http: request canceled")
var http2goAwayTimeout = 1 * time.Second
const http2NextProtoTLS = "h2"
type http2Transport struct {
MaxHeaderListSize uint32
ConnPool interface{}
}
func (*http2Transport) RoundTrip(*Request) (*Response, error) { panic(noHTTP2) }
func (*http2Transport) CloseIdleConnections() {}
type http2erringRoundTripper struct{}
func (http2erringRoundTripper) RoundTrip(*Request) (*Response, error) { panic(noHTTP2) }
type http2noDialClientConnPool struct {
http2clientConnPool http2clientConnPool
}
type http2clientConnPool struct {
mu *sync.Mutex
conns map[string][]struct{}
}
func http2configureTransport(*Transport) (*http2Transport, error) { panic(noHTTP2) }
func http2isNoCachedConnError(err error) bool {
_, ok := err.(interface{ IsHTTP2NoCachedConnError() })
return ok
}
type http2Server struct {
NewWriteScheduler func() http2WriteScheduler
}
type http2WriteScheduler interface{}
func http2NewPriorityWriteScheduler(interface{}) http2WriteScheduler { panic(noHTTP2) }
func http2ConfigureServer(s *Server, conf *http2Server) error { panic(noHTTP2) }
var http2ErrNoCachedConn = http2noCachedConnError{}
type http2noCachedConnError struct{}
func (http2noCachedConnError) IsHTTP2NoCachedConnError() {}
func (http2noCachedConnError) Error() string { return "http2: no cached connection was available" }
...@@ -1507,6 +1507,7 @@ func TestTLSServer(t *testing.T) { ...@@ -1507,6 +1507,7 @@ func TestTLSServer(t *testing.T) {
} }
func TestServeTLS(t *testing.T) { func TestServeTLS(t *testing.T) {
CondSkipHTTP2(t)
// Not parallel: uses global test hooks. // Not parallel: uses global test hooks.
defer afterTest(t) defer afterTest(t)
defer SetTestHookServerServe(nil) defer SetTestHookServerServe(nil)
...@@ -1657,6 +1658,7 @@ func TestAutomaticHTTP2_ListenAndServe_GetCertificate(t *testing.T) { ...@@ -1657,6 +1658,7 @@ func TestAutomaticHTTP2_ListenAndServe_GetCertificate(t *testing.T) {
} }
func testAutomaticHTTP2_ListenAndServe(t *testing.T, tlsConf *tls.Config) { func testAutomaticHTTP2_ListenAndServe(t *testing.T, tlsConf *tls.Config) {
CondSkipHTTP2(t)
// Not parallel: uses global test hooks. // Not parallel: uses global test hooks.
defer afterTest(t) defer afterTest(t)
defer SetTestHookServerServe(nil) defer SetTestHookServerServe(nil)
......
...@@ -3160,7 +3160,7 @@ func (srv *Server) onceSetNextProtoDefaults_Serve() { ...@@ -3160,7 +3160,7 @@ func (srv *Server) onceSetNextProtoDefaults_Serve() {
// configured otherwise. (by setting srv.TLSNextProto non-nil) // configured otherwise. (by setting srv.TLSNextProto non-nil)
// It must only be called via srv.nextProtoOnce (use srv.setupHTTP2_*). // It must only be called via srv.nextProtoOnce (use srv.setupHTTP2_*).
func (srv *Server) onceSetNextProtoDefaults() { func (srv *Server) onceSetNextProtoDefaults() {
if strings.Contains(os.Getenv("GODEBUG"), "http2server=0") { if omitBundledHTTP2 || strings.Contains(os.Getenv("GODEBUG"), "http2server=0") {
return return
} }
// Enable HTTP/2 by default if the user hasn't otherwise // Enable HTTP/2 by default if the user hasn't otherwise
......
...@@ -286,7 +286,6 @@ func (t *Transport) Clone() *Transport { ...@@ -286,7 +286,6 @@ func (t *Transport) Clone() *Transport {
DialContext: t.DialContext, DialContext: t.DialContext,
Dial: t.Dial, Dial: t.Dial,
DialTLS: t.DialTLS, DialTLS: t.DialTLS,
TLSClientConfig: t.TLSClientConfig.Clone(),
TLSHandshakeTimeout: t.TLSHandshakeTimeout, TLSHandshakeTimeout: t.TLSHandshakeTimeout,
DisableKeepAlives: t.DisableKeepAlives, DisableKeepAlives: t.DisableKeepAlives,
DisableCompression: t.DisableCompression, DisableCompression: t.DisableCompression,
...@@ -302,6 +301,9 @@ func (t *Transport) Clone() *Transport { ...@@ -302,6 +301,9 @@ func (t *Transport) Clone() *Transport {
WriteBufferSize: t.WriteBufferSize, WriteBufferSize: t.WriteBufferSize,
ReadBufferSize: t.ReadBufferSize, ReadBufferSize: t.ReadBufferSize,
} }
if t.TLSClientConfig != nil {
t2.TLSClientConfig = t.TLSClientConfig.Clone()
}
if !t.tlsNextProtoWasNil { if !t.tlsNextProtoWasNil {
npm := map[string]func(authority string, c *tls.Conn) RoundTripper{} npm := map[string]func(authority string, c *tls.Conn) RoundTripper{}
for k, v := range t.TLSNextProto { for k, v := range t.TLSNextProto {
...@@ -359,6 +361,9 @@ func (t *Transport) onceSetNextProtoDefaults() { ...@@ -359,6 +361,9 @@ func (t *Transport) onceSetNextProtoDefaults() {
// However, if ForceAttemptHTTP2 is true, it overrides the above checks. // However, if ForceAttemptHTTP2 is true, it overrides the above checks.
return return
} }
if omitBundledHTTP2 {
return
}
t2, err := http2configureTransport(t) t2, err := http2configureTransport(t)
if err != nil { if err != nil {
log.Printf("Error enabling Transport HTTP/2 support: %v", err) log.Printf("Error enabling Transport HTTP/2 support: %v", err)
......
...@@ -591,6 +591,7 @@ func TestTransportMaxConnsPerHostIncludeDialInProgress(t *testing.T) { ...@@ -591,6 +591,7 @@ func TestTransportMaxConnsPerHostIncludeDialInProgress(t *testing.T) {
func TestTransportMaxConnsPerHost(t *testing.T) { func TestTransportMaxConnsPerHost(t *testing.T) {
defer afterTest(t) defer afterTest(t)
CondSkipHTTP2(t)
h := HandlerFunc(func(w ResponseWriter, r *Request) { h := HandlerFunc(func(w ResponseWriter, r *Request) {
_, err := w.Write([]byte("foo")) _, err := w.Write([]byte("foo"))
...@@ -3994,6 +3995,7 @@ func TestTransportAutomaticHTTP2_DialTLS(t *testing.T) { ...@@ -3994,6 +3995,7 @@ func TestTransportAutomaticHTTP2_DialTLS(t *testing.T) {
} }
func testTransportAutoHTTP(t *testing.T, tr *Transport, wantH2 bool) { func testTransportAutoHTTP(t *testing.T, tr *Transport, wantH2 bool) {
CondSkipHTTP2(t)
_, err := tr.RoundTrip(new(Request)) _, err := tr.RoundTrip(new(Request))
if err == nil { if err == nil {
t.Error("expected error from RoundTrip") t.Error("expected error from RoundTrip")
...@@ -5896,6 +5898,7 @@ func TestDontCacheBrokenHTTP2Conn(t *testing.T) { ...@@ -5896,6 +5898,7 @@ func TestDontCacheBrokenHTTP2Conn(t *testing.T) {
// only be one decrement regardless of the number of failures. // only be one decrement regardless of the number of failures.
func TestTransportDecrementConnWhenIdleConnRemoved(t *testing.T) { func TestTransportDecrementConnWhenIdleConnRemoved(t *testing.T) {
defer afterTest(t) defer afterTest(t)
CondSkipHTTP2(t)
h := HandlerFunc(func(w ResponseWriter, r *Request) { h := HandlerFunc(func(w ResponseWriter, r *Request) {
_, err := w.Write([]byte("foo")) _, err := w.Write([]byte("foo"))
......
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