mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 05:07:36 +03:00
respect minimum idle timeout of 3 PTOs (#3909)
This commit is contained in:
parent
85764da3b6
commit
f57f876446
3 changed files with 56 additions and 6 deletions
|
@ -194,6 +194,7 @@ type connection struct {
|
||||||
versionNegotiated bool
|
versionNegotiated bool
|
||||||
receivedFirstPacket bool
|
receivedFirstPacket bool
|
||||||
|
|
||||||
|
// the minimum of the max_idle_timeout values advertised by both endpoints
|
||||||
idleTimeout time.Duration
|
idleTimeout time.Duration
|
||||||
creationTime time.Time
|
creationTime time.Time
|
||||||
// The idle timeout is set based on the max of the time we received the last packet...
|
// The idle timeout is set based on the max of the time we received the last packet...
|
||||||
|
@ -662,7 +663,7 @@ runLoop:
|
||||||
} else {
|
} else {
|
||||||
idleTimeoutStartTime := s.idleTimeoutStartTime()
|
idleTimeoutStartTime := s.idleTimeoutStartTime()
|
||||||
if (!s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.config.HandshakeIdleTimeout) ||
|
if (!s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.config.HandshakeIdleTimeout) ||
|
||||||
(s.handshakeComplete && now.Sub(idleTimeoutStartTime) >= s.idleTimeout) {
|
(s.handshakeComplete && now.After(s.nextIdleTimeoutTime())) {
|
||||||
s.destroyImpl(qerr.ErrIdleTimeout)
|
s.destroyImpl(qerr.ErrIdleTimeout)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -720,13 +721,20 @@ func (s *connection) ConnectionState() ConnectionState {
|
||||||
return s.connState
|
return s.connState
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Time when the connection should time out
|
||||||
|
func (s *connection) nextIdleTimeoutTime() time.Time {
|
||||||
|
idleTimeout := utils.Max(s.idleTimeout, s.rttStats.PTO(true)*3)
|
||||||
|
return s.idleTimeoutStartTime().Add(idleTimeout)
|
||||||
|
}
|
||||||
|
|
||||||
// Time when the next keep-alive packet should be sent.
|
// Time when the next keep-alive packet should be sent.
|
||||||
// It returns a zero time if no keep-alive should be sent.
|
// It returns a zero time if no keep-alive should be sent.
|
||||||
func (s *connection) nextKeepAliveTime() time.Time {
|
func (s *connection) nextKeepAliveTime() time.Time {
|
||||||
if s.config.KeepAlivePeriod == 0 || s.keepAlivePingSent || !s.firstAckElicitingPacketAfterIdleSentTime.IsZero() {
|
if s.config.KeepAlivePeriod == 0 || s.keepAlivePingSent || !s.firstAckElicitingPacketAfterIdleSentTime.IsZero() {
|
||||||
return time.Time{}
|
return time.Time{}
|
||||||
}
|
}
|
||||||
return s.lastPacketReceivedTime.Add(s.keepAliveInterval)
|
keepAliveInterval := utils.Max(s.keepAliveInterval, s.rttStats.PTO(true)*3/2)
|
||||||
|
return s.lastPacketReceivedTime.Add(keepAliveInterval)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *connection) maybeResetTimer() {
|
func (s *connection) maybeResetTimer() {
|
||||||
|
@ -740,7 +748,7 @@ func (s *connection) maybeResetTimer() {
|
||||||
if keepAliveTime := s.nextKeepAliveTime(); !keepAliveTime.IsZero() {
|
if keepAliveTime := s.nextKeepAliveTime(); !keepAliveTime.IsZero() {
|
||||||
deadline = keepAliveTime
|
deadline = keepAliveTime
|
||||||
} else {
|
} else {
|
||||||
deadline = s.idleTimeoutStartTime().Add(s.idleTimeout)
|
deadline = s.nextIdleTimeoutTime()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2137,6 +2137,20 @@ var _ = Describe("Connection", func() {
|
||||||
// don't EXPECT() any calls to mconn.Write()
|
// don't EXPECT() any calls to mconn.Write()
|
||||||
time.Sleep(50 * time.Millisecond)
|
time.Sleep(50 * time.Millisecond)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("send PING as keep-alive earliest after 1.5 times the PTO", func() {
|
||||||
|
conn.config.KeepAlivePeriod = time.Microsecond
|
||||||
|
pto := conn.rttStats.PTO(true)
|
||||||
|
conn.lastPacketReceivedTime = time.Now()
|
||||||
|
sentPingTimeChan := make(chan time.Time)
|
||||||
|
packer.EXPECT().PackCoalescedPacket(false, gomock.Any(), conn.version).Do(func(bool, protocol.ByteCount, protocol.VersionNumber) (*coalescedPacket, error) {
|
||||||
|
sentPingTimeChan <- time.Now()
|
||||||
|
return nil, nil
|
||||||
|
})
|
||||||
|
runConn()
|
||||||
|
sentPingTime := <-sentPingTimeChan
|
||||||
|
Expect(sentPingTime.Sub(conn.lastPacketReceivedTime)).To(BeNumerically(">", pto*3/2))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("timeouts", func() {
|
Context("timeouts", func() {
|
||||||
|
@ -2305,6 +2319,34 @@ var _ = Describe("Connection", func() {
|
||||||
conn.shutdown()
|
conn.shutdown()
|
||||||
Eventually(conn.Context().Done()).Should(BeClosed())
|
Eventually(conn.Context().Done()).Should(BeClosed())
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("time out earliest after 3 times the PTO", func() {
|
||||||
|
packer.EXPECT().PackCoalescedPacket(false, gomock.Any(), conn.version).AnyTimes()
|
||||||
|
connRunner.EXPECT().Retire(clientDestConnID)
|
||||||
|
connRunner.EXPECT().Remove(gomock.Any())
|
||||||
|
cryptoSetup.EXPECT().Close()
|
||||||
|
closeTimeChan := make(chan time.Time)
|
||||||
|
tracer.EXPECT().ClosedConnection(gomock.Any()).Do(func(e error) {
|
||||||
|
Expect(e).To(MatchError(&IdleTimeoutError{}))
|
||||||
|
closeTimeChan <- time.Now()
|
||||||
|
})
|
||||||
|
tracer.EXPECT().Close()
|
||||||
|
conn.idleTimeout = time.Millisecond
|
||||||
|
done := make(chan struct{})
|
||||||
|
pto := conn.rttStats.PTO(true)
|
||||||
|
go func() {
|
||||||
|
defer GinkgoRecover()
|
||||||
|
cryptoSetup.EXPECT().RunHandshake().MaxTimes(1)
|
||||||
|
cryptoSetup.EXPECT().GetSessionTicket().MaxTimes(1)
|
||||||
|
cryptoSetup.EXPECT().SetHandshakeConfirmed().MaxTimes(1)
|
||||||
|
close(conn.handshakeCompleteChan)
|
||||||
|
conn.run()
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
closeTime := <-closeTimeChan
|
||||||
|
Expect(closeTime.Sub(conn.lastPacketReceivedTime)).To(BeNumerically(">", pto*3))
|
||||||
|
Eventually(done).Should(BeClosed())
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
It("stores up to MaxConnUnprocessedPackets packets", func() {
|
It("stores up to MaxConnUnprocessedPackets packets", func() {
|
||||||
|
|
|
@ -105,7 +105,7 @@ var _ = Describe("Timeout tests", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("returns net.Error timeout errors when an idle timeout occurs", func() {
|
It("returns net.Error timeout errors when an idle timeout occurs", func() {
|
||||||
const idleTimeout = 100 * time.Millisecond
|
const idleTimeout = 500 * time.Millisecond
|
||||||
|
|
||||||
server, err := quic.ListenAddr(
|
server, err := quic.ListenAddr(
|
||||||
"localhost:0",
|
"localhost:0",
|
||||||
|
@ -173,7 +173,7 @@ var _ = Describe("Timeout tests", func() {
|
||||||
var idleTimeout time.Duration
|
var idleTimeout time.Duration
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
idleTimeout = scaleDuration(100 * time.Millisecond)
|
idleTimeout = scaleDuration(500 * time.Millisecond)
|
||||||
})
|
})
|
||||||
|
|
||||||
It("times out after inactivity", func() {
|
It("times out after inactivity", func() {
|
||||||
|
@ -315,7 +315,7 @@ var _ = Describe("Timeout tests", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("does not time out if keepalive is set", func() {
|
It("does not time out if keepalive is set", func() {
|
||||||
const idleTimeout = 100 * time.Millisecond
|
const idleTimeout = 500 * time.Millisecond
|
||||||
|
|
||||||
server, err := quic.ListenAddr(
|
server, err := quic.ListenAddr(
|
||||||
"localhost:0",
|
"localhost:0",
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue