save timestamp when a packet arrives in ReceivedPacketHandler

work towards #10, #81
This commit is contained in:
Marten Seemann
2016-05-12 15:12:51 +07:00
parent 33c0f3a5df
commit e782f28b2b
2 changed files with 30 additions and 10 deletions

View File

@@ -2,6 +2,7 @@ package ackhandler
import ( import (
"errors" "errors"
"time"
"github.com/lucas-clemente/quic-go/frames" "github.com/lucas-clemente/quic-go/frames"
"github.com/lucas-clemente/quic-go/protocol" "github.com/lucas-clemente/quic-go/protocol"
@@ -10,18 +11,23 @@ import (
// ErrDuplicatePacket occurres when a duplicate packet is received // ErrDuplicatePacket occurres when a duplicate packet is received
var ErrDuplicatePacket = errors.New("ReceivedPacketHandler: Duplicate Packet") var ErrDuplicatePacket = errors.New("ReceivedPacketHandler: Duplicate Packet")
type packetHistoryEntry struct {
EntropyBit bool
TimeReceived time.Time
}
type receivedPacketHandler struct { type receivedPacketHandler struct {
highestInOrderObserved protocol.PacketNumber highestInOrderObserved protocol.PacketNumber
highestInOrderObservedEntropy EntropyAccumulator highestInOrderObservedEntropy EntropyAccumulator
largestObserved protocol.PacketNumber largestObserved protocol.PacketNumber
packetHistory map[protocol.PacketNumber]bool // the bool is the EntropyBit of the packet packetHistory map[protocol.PacketNumber]packetHistoryEntry
stateChanged bool // has an ACK for this state already been sent? Will be set to false every time a new packet arrives, and to false every time an ACK is sent stateChanged bool // has an ACK for this state already been sent? Will be set to false every time a new packet arrives, and to false every time an ACK is sent
} }
// NewReceivedPacketHandler creates a new receivedPacketHandler // NewReceivedPacketHandler creates a new receivedPacketHandler
func NewReceivedPacketHandler() ReceivedPacketHandler { func NewReceivedPacketHandler() ReceivedPacketHandler {
return &receivedPacketHandler{ return &receivedPacketHandler{
packetHistory: make(map[protocol.PacketNumber]bool), packetHistory: make(map[protocol.PacketNumber]packetHistoryEntry),
} }
} }
@@ -29,7 +35,8 @@ func (h *receivedPacketHandler) ReceivedPacket(packetNumber protocol.PacketNumbe
if packetNumber == 0 { if packetNumber == 0 {
return errors.New("Invalid packet number") return errors.New("Invalid packet number")
} }
if packetNumber <= h.highestInOrderObserved || h.packetHistory[packetNumber] { _, ok := h.packetHistory[packetNumber]
if packetNumber <= h.highestInOrderObserved || ok {
return ErrDuplicatePacket return ErrDuplicatePacket
} }
@@ -44,7 +51,10 @@ func (h *receivedPacketHandler) ReceivedPacket(packetNumber protocol.PacketNumbe
h.highestInOrderObservedEntropy.Add(packetNumber, entropyBit) h.highestInOrderObservedEntropy.Add(packetNumber, entropyBit)
} }
h.packetHistory[packetNumber] = entropyBit h.packetHistory[packetNumber] = packetHistoryEntry{
EntropyBit: entropyBit,
TimeReceived: time.Now(),
}
return nil return nil
} }
@@ -68,7 +78,7 @@ func (h *receivedPacketHandler) getNackRanges() ([]frames.NackRange, EntropyAccu
inRange := false inRange := false
entropy := h.highestInOrderObservedEntropy entropy := h.highestInOrderObservedEntropy
for i := h.largestObserved; i > h.highestInOrderObserved; i-- { for i := h.largestObserved; i > h.highestInOrderObserved; i-- {
entropyBit, ok := h.packetHistory[i] p, ok := h.packetHistory[i]
if !ok { if !ok {
if !inRange { if !inRange {
r := frames.NackRange{ r := frames.NackRange{
@@ -82,7 +92,7 @@ func (h *receivedPacketHandler) getNackRanges() ([]frames.NackRange, EntropyAccu
} }
} else { } else {
inRange = false inRange = false
entropy.Add(i, entropyBit) entropy.Add(i, p.EntropyBit)
} }
} }
return ranges, entropy return ranges, entropy

View File

@@ -1,6 +1,8 @@
package ackhandler package ackhandler
import ( import (
"time"
"github.com/lucas-clemente/quic-go/frames" "github.com/lucas-clemente/quic-go/frames"
"github.com/lucas-clemente/quic-go/protocol" "github.com/lucas-clemente/quic-go/protocol"
@@ -19,7 +21,7 @@ var _ = Describe("receivedPacketHandler", func() {
expectedEntropy = EntropyAccumulator(0) expectedEntropy = EntropyAccumulator(0)
}) })
Context("accepting and rejecting packets", func() { Context("accepting packets", func() {
It("handles a packet that arrives late", func() { It("handles a packet that arrives late", func() {
err := handler.ReceivedPacket(protocol.PacketNumber(1), false) err := handler.ReceivedPacket(protocol.PacketNumber(1), false)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
@@ -27,8 +29,9 @@ var _ = Describe("receivedPacketHandler", func() {
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
err = handler.ReceivedPacket(protocol.PacketNumber(2), false) err = handler.ReceivedPacket(protocol.PacketNumber(2), false)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
nackRanges, _ := handler.getNackRanges() Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(1)))
Expect(len(nackRanges)).To(Equal(0)) Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(2)))
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(3)))
}) })
It("rejects a duplicate package with PacketNumber equal to LargestObserved", func() { It("rejects a duplicate package with PacketNumber equal to LargestObserved", func() {
@@ -50,6 +53,13 @@ var _ = Describe("receivedPacketHandler", func() {
Expect(err).To(HaveOccurred()) Expect(err).To(HaveOccurred())
Expect(err).To(Equal(ErrDuplicatePacket)) Expect(err).To(Equal(ErrDuplicatePacket))
}) })
It("saves the time when each packet arrived", func() {
err := handler.ReceivedPacket(protocol.PacketNumber(3), false)
Expect(err).ToNot(HaveOccurred())
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(3)))
Expect(handler.packetHistory[3].TimeReceived).To(BeTemporally("~", time.Now(), 10*time.Millisecond))
})
}) })
Context("Entropy calculation", func() { Context("Entropy calculation", func() {