make the client send a packet when the handshake completes (in gQUIC)

In gQUIC, there's no equivalent to the Finished message in TLS. The
server knows that the handshake is complete when it receives the first
forward-secure packet sent by the client. If the protocol demands that
the server sends the first data, this would never happen. We need to
make sure that the client actually sends such a packet. Queueing a PING
frame is an easy way to do so.
This commit is contained in:
Marten Seemann 2018-01-10 17:10:50 +07:00
parent ca0f9f4a12
commit 3818fa16b3
2 changed files with 22 additions and 0 deletions

View file

@ -394,6 +394,12 @@ runLoop:
s.handshakeComplete = true
handshakeEvent = nil // prevent this case from ever being selected again
s.sentPacketHandler.SetHandshakeComplete()
if !s.version.UsesTLS() && s.perspective == protocol.PerspectiveClient {
// In gQUIC, there's no equivalent to the Finished message in TLS
// The server knows that the handshake is complete when it receives the first forward-secure packet sent by the client.
// We need to make sure that the client actually sends such a packet.
s.packer.QueueControlFrame(&wire.PingFrame{})
}
close(s.handshakeChan)
} else {
s.tryDecryptingQueuedPackets()

View file

@ -1470,6 +1470,22 @@ var _ = Describe("Client Session", func() {
newCryptoSetupClient = handshake.NewCryptoSetupClient
})
It("sends a forward-secure packet when the handshake completes", func() {
sess.packer.hasSentPacket = true
done := make(chan struct{})
go func() {
defer GinkgoRecover()
err := sess.run()
Expect(err).ToNot(HaveOccurred())
close(done)
}()
close(handshakeChan)
Eventually(mconn.written).Should(Receive())
//make sure the go routine returns
Expect(sess.Close(nil)).To(Succeed())
Eventually(done).Should(BeClosed())
})
Context("receiving packets", func() {
var hdr *wire.Header