@ -560,10 +560,11 @@ func (t *Transport) RoundTripOpt(req *http.Request, opt RoundTripOpt) (*http.Res
traceGotConn ( req , cc , reused )
traceGotConn ( req , cc , reused )
res , err := cc . RoundTrip ( req )
res , err := cc . RoundTrip ( req )
if err != nil && retry <= 6 {
if err != nil && retry <= 6 {
roundTripErr := err
if req , err = shouldRetryRequest ( req , err ) ; err == nil {
if req , err = shouldRetryRequest ( req , err ) ; err == nil {
// After the first retry, do exponential backoff with 10% jitter.
// After the first retry, do exponential backoff with 10% jitter.
if retry == 0 {
if retry == 0 {
t . vlogf ( "RoundTrip retrying after failure: %v" , e rr)
t . vlogf ( "RoundTrip retrying after failure: %v" , roundTripE rr)
continue
continue
}
}
backoff := float64 ( uint ( 1 ) << ( uint ( retry ) - 1 ) )
backoff := float64 ( uint ( 1 ) << ( uint ( retry ) - 1 ) )
@ -572,7 +573,7 @@ func (t *Transport) RoundTripOpt(req *http.Request, opt RoundTripOpt) (*http.Res
timer := backoffNewTimer ( d )
timer := backoffNewTimer ( d )
select {
select {
case <- timer . C :
case <- timer . C :
t . vlogf ( "RoundTrip retrying after failure: %v" , e rr)
t . vlogf ( "RoundTrip retrying after failure: %v" , roundTripE rr)
continue
continue
case <- req . Context ( ) . Done ( ) :
case <- req . Context ( ) . Done ( ) :
timer . Stop ( )
timer . Stop ( )
@ -1265,6 +1266,27 @@ func (cc *ClientConn) RoundTrip(req *http.Request) (*http.Response, error) {
return res , nil
return res , nil
}
}
cancelRequest := func ( cs * clientStream , err error ) error {
cs . cc . mu . Lock ( )
defer cs . cc . mu . Unlock ( )
cs . abortStreamLocked ( err )
if cs . ID != 0 {
// This request may have failed because of a problem with the connection,
// or for some unrelated reason. (For example, the user might have canceled
// the request without waiting for a response.) Mark the connection as
// not reusable, since trying to reuse a dead connection is worse than
// unnecessarily creating a new one.
//
// If cs.ID is 0, then the request was never allocated a stream ID and
// whatever went wrong was unrelated to the connection. We might have
// timed out waiting for a stream slot when StrictMaxConcurrentStreams
// is set, for example, in which case retrying on a different connection
// will not help.
cs . cc . doNotReuse = true
}
return err
}
for {
for {
select {
select {
case <- cs . respHeaderRecv :
case <- cs . respHeaderRecv :
@ -1279,15 +1301,12 @@ func (cc *ClientConn) RoundTrip(req *http.Request) (*http.Response, error) {
return handleResponseHeaders ( )
return handleResponseHeaders ( )
default :
default :
waitDone ( )
waitDone ( )
return nil , cs . abortErr
return nil , cancelRequest ( cs , cs . abortErr )
}
}
case <- ctx . Done ( ) :
case <- ctx . Done ( ) :
err := ctx . Err ( )
return nil , cancelRequest ( cs , ctx . Err ( ) )
cs . abortStream ( err )
return nil , err
case <- cs . reqCancel :
case <- cs . reqCancel :
cs . abortStream ( errRequestCanceled )
return nil , cancelRequest ( cs , errRequestCanceled )
return nil , errRequestCanceled
}
}
}
}
}
}
@ -2555,6 +2574,9 @@ func (b transportResponseBody) Close() error {
cs := b . cs
cs := b . cs
cc := cs . cc
cc := cs . cc
cs . bufPipe . BreakWithError ( errClosedResponseBody )
cs . abortStream ( errClosedResponseBody )
unread := cs . bufPipe . Len ( )
unread := cs . bufPipe . Len ( )
if unread > 0 {
if unread > 0 {
cc . mu . Lock ( )
cc . mu . Lock ( )
@ -2573,9 +2595,6 @@ func (b transportResponseBody) Close() error {
cc . wmu . Unlock ( )
cc . wmu . Unlock ( )
}
}
cs . bufPipe . BreakWithError ( errClosedResponseBody )
cs . abortStream ( errClosedResponseBody )
select {
select {
case <- cs . donec :
case <- cs . donec :
case <- cs . ctx . Done ( ) :
case <- cs . ctx . Done ( ) :