mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
http3: fix connection re-dialing logic for non-QUIC errors (#4879)
This commit is contained in:
parent
259fd92306
commit
c017def433
2 changed files with 102 additions and 14 deletions
|
@ -225,22 +225,30 @@ func (t *Transport) roundTripOpt(req *http.Request, opt RoundTripOpt) (*http.Res
|
||||||
defer cl.useCount.Add(-1)
|
defer cl.useCount.Add(-1)
|
||||||
rsp, err := cl.rt.RoundTrip(req)
|
rsp, err := cl.rt.RoundTrip(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
// request aborted due to context cancellation
|
||||||
select {
|
select {
|
||||||
case <-req.Context().Done():
|
case <-req.Context().Done():
|
||||||
return nil, err
|
return nil, err
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Non-nil errors are likely might be due to a problem with the connection,
|
// Retry the request on a new connection if:
|
||||||
// so we remove the client from the cache so that subsequent calls reconnect.
|
// 1. it was sent on a reused connection,
|
||||||
t.removeClient(hostname)
|
// 2. this connection is now closed,
|
||||||
if isReused {
|
// 3. and the error is a timeout error.
|
||||||
if nerr, ok := err.(net.Error); ok && nerr.Timeout() {
|
select {
|
||||||
return t.RoundTripOpt(req, opt)
|
case <-cl.conn.Context().Done():
|
||||||
|
t.removeClient(hostname)
|
||||||
|
if isReused {
|
||||||
|
var nerr net.Error
|
||||||
|
if errors.As(err, &nerr) && nerr.Timeout() {
|
||||||
|
return t.RoundTripOpt(req, opt)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return nil, err
|
||||||
|
default:
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
return rsp, nil
|
return rsp, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -253,18 +253,31 @@ func TestTransportConnectionReuse(t *testing.T) {
|
||||||
require.Equal(t, 1, dialCount)
|
require.Equal(t, 1, dialCount)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Requests reuse the same underlying QUIC connection.
|
||||||
|
// If a request experiences an error, the behavior depends on the nature of that error.
|
||||||
func TestTransportConnectionRedial(t *testing.T) {
|
func TestTransportConnectionRedial(t *testing.T) {
|
||||||
|
// If it's connection error that is a timeout error, we re-dial a new connection.
|
||||||
|
// No error will be returned to the caller.
|
||||||
t.Run("timeout error", func(t *testing.T) {
|
t.Run("timeout error", func(t *testing.T) {
|
||||||
testTransportConnectionRedial(t, &qerr.IdleTimeoutError{}, nil)
|
testTransportConnectionRedial(t, true, &qerr.IdleTimeoutError{}, nil)
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("other error", func(t *testing.T) {
|
// If it's a different connection error, the error is returned to the caller.
|
||||||
testErr := errors.New("test error")
|
// The connection is not redialed.
|
||||||
testTransportConnectionRedial(t, testErr, testErr)
|
t.Run("other error from the connection", func(t *testing.T) {
|
||||||
|
testErr := &quic.TransportError{ErrorCode: quic.ConnectionIDLimitError}
|
||||||
|
testTransportConnectionRedial(t, true, testErr, testErr)
|
||||||
|
})
|
||||||
|
|
||||||
|
// If the error is not related to the connection, we return that error.
|
||||||
|
// The underlying connection remains open and is reused for subsequent requests.
|
||||||
|
t.Run("other error not from the connection", func(t *testing.T) {
|
||||||
|
testErr := &quic.TransportError{ErrorCode: quic.ConnectionIDLimitError}
|
||||||
|
testTransportConnectionRedial(t, false, testErr, testErr)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func testTransportConnectionRedial(t *testing.T, dialErr, expectedErr error) {
|
func testTransportConnectionRedial(t *testing.T, connClosed bool, roundtripErr, expectedErr error) {
|
||||||
mockCtrl := gomock.NewController(t)
|
mockCtrl := gomock.NewController(t)
|
||||||
cl := NewMockSingleRoundTripper(mockCtrl)
|
cl := NewMockSingleRoundTripper(mockCtrl)
|
||||||
conn := mockquic.NewMockEarlyConnection(mockCtrl)
|
conn := mockquic.NewMockEarlyConnection(mockCtrl)
|
||||||
|
@ -280,6 +293,7 @@ func testTransportConnectionRedial(t *testing.T, dialErr, expectedErr error) {
|
||||||
newClient: func(quic.EarlyConnection) singleRoundTripper { return cl },
|
newClient: func(quic.EarlyConnection) singleRoundTripper { return cl },
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// the first request succeeds
|
||||||
req1 := mustNewRequest("GET", "https://quic-go.net/file1.html", nil)
|
req1 := mustNewRequest("GET", "https://quic-go.net/file1.html", nil)
|
||||||
cl.EXPECT().RoundTrip(req1).Return(&http.Response{Request: req1}, nil)
|
cl.EXPECT().RoundTrip(req1).Return(&http.Response{Request: req1}, nil)
|
||||||
rsp, err := tr.RoundTrip(req1)
|
rsp, err := tr.RoundTrip(req1)
|
||||||
|
@ -287,8 +301,15 @@ func testTransportConnectionRedial(t *testing.T, dialErr, expectedErr error) {
|
||||||
require.Equal(t, req1, rsp.Request)
|
require.Equal(t, req1, rsp.Request)
|
||||||
require.Equal(t, 1, dialCount)
|
require.Equal(t, 1, dialCount)
|
||||||
|
|
||||||
|
// the second request reuses the QUIC connection, and encounters an error
|
||||||
req2 := mustNewRequest("GET", "https://quic-go.net/file2.html", nil)
|
req2 := mustNewRequest("GET", "https://quic-go.net/file2.html", nil)
|
||||||
cl.EXPECT().RoundTrip(req2).Return(nil, dialErr)
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
if connClosed {
|
||||||
|
cancel()
|
||||||
|
}
|
||||||
|
conn.EXPECT().Context().Return(ctx)
|
||||||
|
cl.EXPECT().RoundTrip(req2).Return(nil, roundtripErr)
|
||||||
if expectedErr == nil {
|
if expectedErr == nil {
|
||||||
cl.EXPECT().RoundTrip(req2).Return(&http.Response{Request: req2}, nil)
|
cl.EXPECT().RoundTrip(req2).Return(&http.Response{Request: req2}, nil)
|
||||||
}
|
}
|
||||||
|
@ -301,6 +322,65 @@ func testTransportConnectionRedial(t *testing.T, dialErr, expectedErr error) {
|
||||||
require.ErrorIs(t, err, expectedErr)
|
require.ErrorIs(t, err, expectedErr)
|
||||||
require.Equal(t, 1, dialCount)
|
require.Equal(t, 1, dialCount)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// if the error was not a connection error, the next request reuses the connection
|
||||||
|
if connClosed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
currentDialCount := dialCount
|
||||||
|
req3 := mustNewRequest("GET", "https://quic-go.net/file3.html", nil)
|
||||||
|
cl.EXPECT().RoundTrip(req3).Return(&http.Response{Request: req3}, nil)
|
||||||
|
rsp, err = tr.RoundTrip(req3)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, req3, rsp.Request)
|
||||||
|
require.Equal(t, currentDialCount, dialCount) // no new connection was dialed
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTransportRequestContextCancellation(t *testing.T) {
|
||||||
|
mockCtrl := gomock.NewController(t)
|
||||||
|
cl := NewMockSingleRoundTripper(mockCtrl)
|
||||||
|
conn := mockquic.NewMockEarlyConnection(mockCtrl)
|
||||||
|
handshakeChan := make(chan struct{})
|
||||||
|
close(handshakeChan)
|
||||||
|
conn.EXPECT().HandshakeComplete().Return(handshakeChan).MaxTimes(3)
|
||||||
|
var dialCount int
|
||||||
|
tr := &Transport{
|
||||||
|
Dial: func(context.Context, string, *tls.Config, *quic.Config) (quic.EarlyConnection, error) {
|
||||||
|
dialCount++
|
||||||
|
return conn, nil
|
||||||
|
},
|
||||||
|
newClient: func(quic.EarlyConnection) singleRoundTripper { return cl },
|
||||||
|
}
|
||||||
|
|
||||||
|
// the first request succeeds
|
||||||
|
req1 := mustNewRequest("GET", "https://quic-go.net/file1.html", nil)
|
||||||
|
cl.EXPECT().RoundTrip(req1).Return(&http.Response{Request: req1}, nil)
|
||||||
|
rsp, err := tr.RoundTrip(req1)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, req1, rsp.Request)
|
||||||
|
require.Equal(t, 1, dialCount)
|
||||||
|
|
||||||
|
// the second request reuses the QUIC connection, and runs into the cancelled context
|
||||||
|
req2 := mustNewRequest("GET", "https://quic-go.net/file2.html", nil)
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
req2 = req2.WithContext(ctx)
|
||||||
|
cl.EXPECT().RoundTrip(req2).DoAndReturn(
|
||||||
|
func(r *http.Request) (*http.Response, error) {
|
||||||
|
cancel()
|
||||||
|
return nil, context.Canceled
|
||||||
|
},
|
||||||
|
)
|
||||||
|
_, err = tr.RoundTrip(req2)
|
||||||
|
require.ErrorIs(t, err, context.Canceled)
|
||||||
|
require.Equal(t, 1, dialCount)
|
||||||
|
|
||||||
|
// the next request reuses the QUIC connection
|
||||||
|
req3 := mustNewRequest("GET", "https://quic-go.net/file2.html", nil)
|
||||||
|
cl.EXPECT().RoundTrip(req3).Return(&http.Response{Request: req3}, nil)
|
||||||
|
rsp, err = tr.RoundTrip(req3)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, req3, rsp.Request)
|
||||||
|
require.Equal(t, 1, dialCount)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestTransportConnetionRedialHandshakeError(t *testing.T) {
|
func TestTransportConnetionRedialHandshakeError(t *testing.T) {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue