Commit 71d2fa8b authored by Brad Fitzpatrick's avatar Brad Fitzpatrick

net/http: deflake a non-short test, clean up export_test.go

This makes TestTransportResponseCloseRace much faster and no longer
flaky.

In the process it also cleans up test hooks in net/http which were
inconsistent and scattered.

Change-Id: Ifd0b11dbc7e8915c24eb5bdc36731ed6751dd7ec
Reviewed-on: https://go-review.googlesource.com/17316Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
parent 7a4022ee
...@@ -9,11 +9,23 @@ package http ...@@ -9,11 +9,23 @@ package http
import ( import (
"net" "net"
"net/url"
"sync" "sync"
"time" "time"
) )
var (
DefaultUserAgent = defaultUserAgent
NewLoggingConn = newLoggingConn
ExportAppendTime = appendTime
ExportRefererForURL = refererForURL
ExportServerNewConn = (*Server).newConn
ExportCloseWriteAndWait = (*conn).closeWriteAndWait
ExportErrRequestCanceled = errRequestCanceled
ExportServeFile = serveFile
ExportHttp2ConfigureTransport = http2ConfigureTransport
ExportHttp2ConfigureServer = http2ConfigureServer
)
func init() { func init() {
// We only want to pay for this cost during testing. // We only want to pay for this cost during testing.
// When not under test, these values are always nil // When not under test, these values are always nil
...@@ -21,11 +33,42 @@ func init() { ...@@ -21,11 +33,42 @@ func init() {
testHookMu = new(sync.Mutex) testHookMu = new(sync.Mutex)
} }
func NewLoggingConn(baseName string, c net.Conn) net.Conn { var (
return newLoggingConn(baseName, c) SetInstallConnClosedHook = hookSetter(&testHookPersistConnClosedGotRes)
SetEnterRoundTripHook = hookSetter(&testHookEnterRoundTrip)
SetTestHookWaitResLoop = hookSetter(&testHookWaitResLoop)
SetRoundTripRetried = hookSetter(&testHookRoundTripRetried)
)
func SetReadLoopBeforeNextReadHook(f func()) {
testHookMu.Lock()
defer testHookMu.Unlock()
unnilTestHook(&f)
testHookReadLoopBeforeNextRead = f
}
// SetPendingDialHooks sets the hooks that run before and after handling
// pending dials.
func SetPendingDialHooks(before, after func()) {
unnilTestHook(&before)
unnilTestHook(&after)
testHookPrePendingDial, testHookPostPendingDial = before, after
}
func SetTestHookServerServe(fn func(*Server, net.Listener)) { testHookServerServe = fn }
func NewTestTimeoutHandler(handler Handler, ch <-chan time.Time) Handler {
f := func() <-chan time.Time {
return ch
}
return &timeoutHandler{handler, f, ""}
} }
var ExportAppendTime = appendTime func ResetCachedEnvironment() {
httpProxyEnv.reset()
httpsProxyEnv.reset()
noProxyEnv.reset()
}
func (t *Transport) NumPendingRequestsForTesting() int { func (t *Transport) NumPendingRequestsForTesting() int {
t.reqMu.Lock() t.reqMu.Lock()
...@@ -86,60 +129,17 @@ func (t *Transport) PutIdleTestConn() bool { ...@@ -86,60 +129,17 @@ func (t *Transport) PutIdleTestConn() bool {
}) })
} }
func SetInstallConnClosedHook(f func()) { // All test hooks must be non-nil so they can be called directly,
testHookPersistConnClosedGotRes = f // but the tests use nil to mean hook disabled.
} func unnilTestHook(f *func()) {
if *f == nil {
func SetEnterRoundTripHook(f func()) { *f = nop
testHookEnterRoundTrip = f
}
func SetReadLoopBeforeNextReadHook(f func()) {
testHookMu.Lock()
defer testHookMu.Unlock()
testHookReadLoopBeforeNextRead = f
}
func NewTestTimeoutHandler(handler Handler, ch <-chan time.Time) Handler {
f := func() <-chan time.Time {
return ch
} }
return &timeoutHandler{handler, f, ""}
}
func ResetCachedEnvironment() {
httpProxyEnv.reset()
httpsProxyEnv.reset()
noProxyEnv.reset()
}
var DefaultUserAgent = defaultUserAgent
func ExportRefererForURL(lastReq, newReq *url.URL) string {
return refererForURL(lastReq, newReq)
}
// SetPendingDialHooks sets the hooks that run before and after handling
// pending dials.
func SetPendingDialHooks(before, after func()) {
prePendingDial, postPendingDial = before, after
} }
// SetRetriedHook sets the hook that runs when an idempotent retry occurs. func hookSetter(dst *func()) func(func()) {
func SetRetriedHook(hook func()) { return func(fn func()) {
retried = hook unnilTestHook(&fn)
*dst = fn
}
} }
var ExportServerNewConn = (*Server).newConn
var ExportCloseWriteAndWait = (*conn).closeWriteAndWait
var ExportErrRequestCanceled = errRequestCanceled
var ExportServeFile = serveFile
var ExportHttp2ConfigureTransport = http2ConfigureTransport
var ExportHttp2ConfigureServer = http2ConfigureServer
func SetTestHookServerServe(fn func(*Server, net.Listener)) { testHookServerServe = fn }
...@@ -288,9 +288,7 @@ func (t *Transport) RoundTrip(req *Request) (*Response, error) { ...@@ -288,9 +288,7 @@ func (t *Transport) RoundTrip(req *Request) (*Response, error) {
if err := checkTransportResend(err, req, pconn); err != nil { if err := checkTransportResend(err, req, pconn); err != nil {
return nil, err return nil, err
} }
if retried != nil { testHookRoundTripRetried()
retried()
}
} }
} }
...@@ -600,9 +598,6 @@ func (t *Transport) dial(network, addr string) (c net.Conn, err error) { ...@@ -600,9 +598,6 @@ func (t *Transport) dial(network, addr string) (c net.Conn, err error) {
return net.Dial(network, addr) return net.Dial(network, addr)
} }
// Testing hooks:
var prePendingDial, postPendingDial, retried func()
// getConn dials and creates a new persistConn to the target as // getConn dials and creates a new persistConn to the target as
// specified in the connectMethod. This includes doing a proxy CONNECT // specified in the connectMethod. This includes doing a proxy CONNECT
// and/or setting up TLS. If this doesn't return an error, the persistConn // and/or setting up TLS. If this doesn't return an error, the persistConn
...@@ -624,20 +619,16 @@ func (t *Transport) getConn(req *Request, cm connectMethod) (*persistConn, error ...@@ -624,20 +619,16 @@ func (t *Transport) getConn(req *Request, cm connectMethod) (*persistConn, error
// Copy these hooks so we don't race on the postPendingDial in // Copy these hooks so we don't race on the postPendingDial in
// the goroutine we launch. Issue 11136. // the goroutine we launch. Issue 11136.
prePendingDial := prePendingDial testHookPrePendingDial := testHookPrePendingDial
postPendingDial := postPendingDial testHookPostPendingDial := testHookPostPendingDial
handlePendingDial := func() { handlePendingDial := func() {
if prePendingDial != nil { testHookPrePendingDial()
prePendingDial()
}
go func() { go func() {
if v := <-dialc; v.err == nil { if v := <-dialc; v.err == nil {
t.putIdleConn(v.pc) t.putIdleConn(v.pc)
} }
if postPendingDial != nil { testHookPostPendingDial()
postPendingDial()
}
}() }()
} }
...@@ -1128,10 +1119,7 @@ func (pc *persistConn) readLoop() { ...@@ -1128,10 +1119,7 @@ func (pc *persistConn) readLoop() {
pc.wroteRequest() && pc.wroteRequest() &&
pc.t.putIdleConn(pc) pc.t.putIdleConn(pc)
} }
testHookReadLoopBeforeNextRead()
if hook := testHookReadLoopBeforeNextRead; hook != nil {
hook()
}
} }
pc.close() pc.close()
} }
...@@ -1258,12 +1246,19 @@ var errTimeout error = &httpError{err: "net/http: timeout awaiting response head ...@@ -1258,12 +1246,19 @@ var errTimeout error = &httpError{err: "net/http: timeout awaiting response head
var errClosed error = &httpError{err: "net/http: transport closed before response was received"} var errClosed error = &httpError{err: "net/http: transport closed before response was received"}
var errRequestCanceled = errors.New("net/http: request canceled") var errRequestCanceled = errors.New("net/http: request canceled")
// nil except for tests func nop() {}
// testHooks. Always non-nil.
var ( var (
testHookPersistConnClosedGotRes func() testHookPersistConnClosedGotRes = nop
testHookEnterRoundTrip func() testHookEnterRoundTrip = nop
testHookMu sync.Locker = fakeLocker{} // guards following testHookWaitResLoop = nop
testHookReadLoopBeforeNextRead func() testHookRoundTripRetried = nop
testHookPrePendingDial = nop
testHookPostPendingDial = nop
testHookMu sync.Locker = fakeLocker{} // guards following
testHookReadLoopBeforeNextRead = nop
) )
// beforeRespHeaderError is used to indicate when an IO error has occurred before // beforeRespHeaderError is used to indicate when an IO error has occurred before
...@@ -1273,9 +1268,7 @@ type beforeRespHeaderError struct { ...@@ -1273,9 +1268,7 @@ type beforeRespHeaderError struct {
} }
func (pc *persistConn) roundTrip(req *transportRequest) (resp *Response, err error) { func (pc *persistConn) roundTrip(req *transportRequest) (resp *Response, err error) {
if hook := testHookEnterRoundTrip; hook != nil { testHookEnterRoundTrip()
hook()
}
if !pc.t.replaceReqCanceler(req.Request, pc.cancelRequest) { if !pc.t.replaceReqCanceler(req.Request, pc.cancelRequest) {
pc.t.putIdleConn(pc) pc.t.putIdleConn(pc)
return nil, errRequestCanceled return nil, errRequestCanceled
...@@ -1337,6 +1330,7 @@ func (pc *persistConn) roundTrip(req *transportRequest) (resp *Response, err err ...@@ -1337,6 +1330,7 @@ func (pc *persistConn) roundTrip(req *transportRequest) (resp *Response, err err
cancelChan := req.Request.Cancel cancelChan := req.Request.Cancel
WaitResponse: WaitResponse:
for { for {
testHookWaitResLoop()
select { select {
case err := <-writeErrCh: case err := <-writeErrCh:
if isNetWriteError(err) { if isNetWriteError(err) {
...@@ -1375,9 +1369,7 @@ WaitResponse: ...@@ -1375,9 +1369,7 @@ WaitResponse:
// with a non-blocking receive. // with a non-blocking receive.
select { select {
case re = <-resc: case re = <-resc:
if fn := testHookPersistConnClosedGotRes; fn != nil { testHookPersistConnClosedGotRes()
fn()
}
default: default:
re = responseAndError{err: beforeRespHeaderError{errClosed}} re = responseAndError{err: beforeRespHeaderError{errClosed}}
if pc.isCanceled() { if pc.isCanceled() {
......
...@@ -2431,10 +2431,10 @@ func TestRetryIdempotentRequestsOnError(t *testing.T) { ...@@ -2431,10 +2431,10 @@ func TestRetryIdempotentRequestsOnError(t *testing.T) {
const N = 2 const N = 2
retryc := make(chan struct{}, N) retryc := make(chan struct{}, N)
SetRetriedHook(func() { SetRoundTripRetried(func() {
retryc <- struct{}{} retryc <- struct{}{}
}) })
defer SetRetriedHook(nil) defer SetRoundTripRetried(nil)
for n := 0; n < 100; n++ { for n := 0; n < 100; n++ {
// open 2 conns // open 2 conns
...@@ -2681,6 +2681,15 @@ func TestTransportResponseCloseRace(t *testing.T) { ...@@ -2681,6 +2681,15 @@ func TestTransportResponseCloseRace(t *testing.T) {
sawRace = true sawRace = true
}) })
defer SetInstallConnClosedHook(nil) defer SetInstallConnClosedHook(nil)
SetTestHookWaitResLoop(func() {
// Make the select race much more likely by blocking before
// the select, so both will be ready by the time the
// select runs.
time.Sleep(50 * time.Millisecond)
})
defer SetTestHookWaitResLoop(nil)
tr := &Transport{ tr := &Transport{
DisableKeepAlives: true, DisableKeepAlives: true,
} }
...@@ -2698,6 +2707,7 @@ func TestTransportResponseCloseRace(t *testing.T) { ...@@ -2698,6 +2707,7 @@ func TestTransportResponseCloseRace(t *testing.T) {
} }
resp.Body.Close() resp.Body.Close()
if sawRace { if sawRace {
t.Logf("saw race after %d iterations", i+1)
break break
} }
} }
......
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