Commit 14b07dfc authored by Joe Tsai's avatar Joe Tsai Committed by Joe Tsai

net/http/httptest: allow creation of Server manually

The Server struct has exported fields, which allows users to manually
create a Server object without using using NewServer or NewTLSServer
and directly call Start or StartTLS on their object.

In order to ensure that manual creation of Server works, the
NewUnstartedServer function should not initialize Server in any way
that the user was not able to do themselves. For example, the setting
of a unexported filed, client, is not something a user can do.
Thus, rather than setting the client field in NewUnstartedServer,
we lazily initialize it when Start or StartTLS is called.

Otherwise, the Server logic can nil panic later when it assumes that this
field has been initialized.

Fixes #20871

Change-Id: I65c6a9f893ea963b0fbad0990b33af08007c1140
Reviewed-on: https://go-review.googlesource.com/47353Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent a776087e
...@@ -93,9 +93,6 @@ func NewUnstartedServer(handler http.Handler) *Server { ...@@ -93,9 +93,6 @@ func NewUnstartedServer(handler http.Handler) *Server {
return &Server{ return &Server{
Listener: newLocalListener(), Listener: newLocalListener(),
Config: &http.Server{Handler: handler}, Config: &http.Server{Handler: handler},
client: &http.Client{
Transport: &http.Transport{},
},
} }
} }
...@@ -104,6 +101,9 @@ func (s *Server) Start() { ...@@ -104,6 +101,9 @@ func (s *Server) Start() {
if s.URL != "" { if s.URL != "" {
panic("Server already started") panic("Server already started")
} }
if s.client == nil {
s.client = &http.Client{Transport: &http.Transport{}}
}
s.URL = "http://" + s.Listener.Addr().String() s.URL = "http://" + s.Listener.Addr().String()
s.wrap() s.wrap()
s.goServe() s.goServe()
...@@ -118,6 +118,9 @@ func (s *Server) StartTLS() { ...@@ -118,6 +118,9 @@ func (s *Server) StartTLS() {
if s.URL != "" { if s.URL != "" {
panic("Server already started") panic("Server already started")
} }
if s.client == nil {
s.client = &http.Client{Transport: &http.Transport{}}
}
cert, err := tls.X509KeyPair(internal.LocalhostCert, internal.LocalhostKey) cert, err := tls.X509KeyPair(internal.LocalhostCert, internal.LocalhostKey)
if err != nil { if err != nil {
panic(fmt.Sprintf("httptest: NewTLSServer: %v", err)) panic(fmt.Sprintf("httptest: NewTLSServer: %v", err))
......
...@@ -12,8 +12,48 @@ import ( ...@@ -12,8 +12,48 @@ import (
"testing" "testing"
) )
type newServerFunc func(http.Handler) *Server
var newServers = map[string]newServerFunc{
"NewServer": NewServer,
"NewTLSServer": NewTLSServer,
// The manual variants of newServer create a Server manually by only filling
// in the exported fields of Server.
"NewServerManual": func(h http.Handler) *Server {
ts := &Server{Listener: newLocalListener(), Config: &http.Server{Handler: h}}
ts.Start()
return ts
},
"NewTLSServerManual": func(h http.Handler) *Server {
ts := &Server{Listener: newLocalListener(), Config: &http.Server{Handler: h}}
ts.StartTLS()
return ts
},
}
func TestServer(t *testing.T) { func TestServer(t *testing.T) {
ts := NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { for _, name := range []string{"NewServer", "NewServerManual"} {
t.Run(name, func(t *testing.T) {
newServer := newServers[name]
t.Run("Server", func(t *testing.T) { testServer(t, newServer) })
t.Run("GetAfterClose", func(t *testing.T) { testGetAfterClose(t, newServer) })
t.Run("ServerCloseBlocking", func(t *testing.T) { testServerCloseBlocking(t, newServer) })
t.Run("ServerCloseClientConnections", func(t *testing.T) { testServerCloseClientConnections(t, newServer) })
t.Run("ServerClientTransportType", func(t *testing.T) { testServerClientTransportType(t, newServer) })
})
}
for _, name := range []string{"NewTLSServer", "NewTLSServerManual"} {
t.Run(name, func(t *testing.T) {
newServer := newServers[name]
t.Run("ServerClient", func(t *testing.T) { testServerClient(t, newServer) })
t.Run("TLSServerClientTransportType", func(t *testing.T) { testTLSServerClientTransportType(t, newServer) })
})
}
}
func testServer(t *testing.T, newServer newServerFunc) {
ts := newServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello")) w.Write([]byte("hello"))
})) }))
defer ts.Close() defer ts.Close()
...@@ -32,8 +72,8 @@ func TestServer(t *testing.T) { ...@@ -32,8 +72,8 @@ func TestServer(t *testing.T) {
} }
// Issue 12781 // Issue 12781
func TestGetAfterClose(t *testing.T) { func testGetAfterClose(t *testing.T, newServer newServerFunc) {
ts := NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ts := newServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello")) w.Write([]byte("hello"))
})) }))
...@@ -58,8 +98,8 @@ func TestGetAfterClose(t *testing.T) { ...@@ -58,8 +98,8 @@ func TestGetAfterClose(t *testing.T) {
} }
} }
func TestServerCloseBlocking(t *testing.T) { func testServerCloseBlocking(t *testing.T, newServer newServerFunc) {
ts := NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ts := newServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello")) w.Write([]byte("hello"))
})) }))
dial := func() net.Conn { dial := func() net.Conn {
...@@ -87,9 +127,9 @@ func TestServerCloseBlocking(t *testing.T) { ...@@ -87,9 +127,9 @@ func TestServerCloseBlocking(t *testing.T) {
} }
// Issue 14290 // Issue 14290
func TestServerCloseClientConnections(t *testing.T) { func testServerCloseClientConnections(t *testing.T, newServer newServerFunc) {
var s *Server var s *Server
s = NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { s = newServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
s.CloseClientConnections() s.CloseClientConnections()
})) }))
defer s.Close() defer s.Close()
...@@ -102,8 +142,8 @@ func TestServerCloseClientConnections(t *testing.T) { ...@@ -102,8 +142,8 @@ func TestServerCloseClientConnections(t *testing.T) {
// Tests that the Server.Client method works and returns an http.Client that can hit // Tests that the Server.Client method works and returns an http.Client that can hit
// NewTLSServer without cert warnings. // NewTLSServer without cert warnings.
func TestServerClient(t *testing.T) { func testServerClient(t *testing.T, newTLSServer newServerFunc) {
ts := NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ts := newTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("hello")) w.Write([]byte("hello"))
})) }))
defer ts.Close() defer ts.Close()
...@@ -124,8 +164,8 @@ func TestServerClient(t *testing.T) { ...@@ -124,8 +164,8 @@ func TestServerClient(t *testing.T) {
// Tests that the Server.Client.Transport interface is implemented // Tests that the Server.Client.Transport interface is implemented
// by a *http.Transport. // by a *http.Transport.
func TestServerClientTransportType(t *testing.T) { func testServerClientTransportType(t *testing.T, newServer newServerFunc) {
ts := NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ts := newServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
})) }))
defer ts.Close() defer ts.Close()
client := ts.Client() client := ts.Client()
...@@ -136,8 +176,8 @@ func TestServerClientTransportType(t *testing.T) { ...@@ -136,8 +176,8 @@ func TestServerClientTransportType(t *testing.T) {
// Tests that the TLS Server.Client.Transport interface is implemented // Tests that the TLS Server.Client.Transport interface is implemented
// by a *http.Transport. // by a *http.Transport.
func TestTLSServerClientTransportType(t *testing.T) { func testTLSServerClientTransportType(t *testing.T, newTLSServer newServerFunc) {
ts := NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ts := newTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
})) }))
defer ts.Close() defer ts.Close()
client := ts.Client() client := ts.Client()
......
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