forked from quic-go/quic-go
add basic AckHandler
This commit is contained in:
52
ackhandler/ack_handler.go
Normal file
52
ackhandler/ack_handler.go
Normal file
@@ -0,0 +1,52 @@
|
||||
package ackhandler
|
||||
|
||||
import "github.com/lucas-clemente/quic-go/protocol"
|
||||
|
||||
// The AckHandler handles ACKs
|
||||
type AckHandler struct {
|
||||
LargestObserved protocol.PacketNumber
|
||||
Observed map[protocol.PacketNumber]bool
|
||||
}
|
||||
|
||||
// NewAckHandler creates a new AckHandler
|
||||
func NewAckHandler() (*AckHandler, error) {
|
||||
ackHandler := &AckHandler{
|
||||
Observed: make(map[protocol.PacketNumber]bool),
|
||||
}
|
||||
return ackHandler, nil
|
||||
}
|
||||
|
||||
// HandlePacket handles a packet
|
||||
func (h *AckHandler) HandlePacket(packetNumber protocol.PacketNumber) {
|
||||
if packetNumber > h.LargestObserved {
|
||||
h.LargestObserved = packetNumber
|
||||
}
|
||||
h.Observed[packetNumber] = true
|
||||
}
|
||||
|
||||
// GetNackRanges gets all the NACK ranges
|
||||
func (h *AckHandler) GetNackRanges() []*NackRange {
|
||||
// ToDo: improve performance
|
||||
var ranges []*NackRange
|
||||
inRange := false
|
||||
// ToDo: fix types
|
||||
for i := 0; i < int(h.LargestObserved); i++ {
|
||||
packetNumber := protocol.PacketNumber(i)
|
||||
_, ok := h.Observed[packetNumber]
|
||||
if !ok {
|
||||
if !inRange {
|
||||
r := &NackRange{
|
||||
FirstPacketNumber: packetNumber,
|
||||
Length: 1,
|
||||
}
|
||||
ranges = append(ranges, r)
|
||||
inRange = true
|
||||
} else {
|
||||
ranges[len(ranges)-1].Length++
|
||||
}
|
||||
} else {
|
||||
inRange = false
|
||||
}
|
||||
}
|
||||
return ranges
|
||||
}
|
||||
84
ackhandler/ack_handler_test.go
Normal file
84
ackhandler/ack_handler_test.go
Normal file
@@ -0,0 +1,84 @@
|
||||
package ackhandler
|
||||
|
||||
import (
|
||||
"github.com/lucas-clemente/quic-go/protocol"
|
||||
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
var _ = Describe("AckHandler", func() {
|
||||
It("Returns no NACK ranges for continously received packets", func() {
|
||||
ackHandler, _ := NewAckHandler()
|
||||
for i := 0; i < 100; i++ {
|
||||
ackHandler.HandlePacket(protocol.PacketNumber(i))
|
||||
}
|
||||
Expect(ackHandler.LargestObserved).To(Equal(protocol.PacketNumber(99)))
|
||||
Expect(len(ackHandler.GetNackRanges())).To(Equal(0))
|
||||
})
|
||||
|
||||
It("handles a single lost package", func() {
|
||||
ackHandler, _ := NewAckHandler()
|
||||
for i := 0; i < 10; i++ {
|
||||
if i == 5 {
|
||||
continue
|
||||
}
|
||||
ackHandler.HandlePacket(protocol.PacketNumber(i))
|
||||
}
|
||||
Expect(ackHandler.LargestObserved).To(Equal(protocol.PacketNumber(9)))
|
||||
nackRanges := ackHandler.GetNackRanges()
|
||||
Expect(len(nackRanges)).To(Equal(1))
|
||||
Expect(nackRanges[0].FirstPacketNumber).To(Equal(protocol.PacketNumber(5)))
|
||||
Expect(nackRanges[0].Length).To(Equal(uint8(1)))
|
||||
})
|
||||
|
||||
It("handles two consecutive lost packages", func() {
|
||||
ackHandler, _ := NewAckHandler()
|
||||
for i := 0; i < 10; i++ {
|
||||
if i == 5 || i == 6 {
|
||||
continue
|
||||
}
|
||||
ackHandler.HandlePacket(protocol.PacketNumber(i))
|
||||
}
|
||||
Expect(ackHandler.LargestObserved).To(Equal(protocol.PacketNumber(9)))
|
||||
nackRanges := ackHandler.GetNackRanges()
|
||||
Expect(len(nackRanges)).To(Equal(1))
|
||||
Expect(nackRanges[0].FirstPacketNumber).To(Equal(protocol.PacketNumber(5)))
|
||||
Expect(nackRanges[0].Length).To(Equal(uint8(2)))
|
||||
})
|
||||
|
||||
It("handles two non-consecutively lost packages", func() {
|
||||
ackHandler, _ := NewAckHandler()
|
||||
for i := 0; i < 10; i++ {
|
||||
if i == 3 || i == 7 {
|
||||
continue
|
||||
}
|
||||
ackHandler.HandlePacket(protocol.PacketNumber(i))
|
||||
}
|
||||
Expect(ackHandler.LargestObserved).To(Equal(protocol.PacketNumber(9)))
|
||||
nackRanges := ackHandler.GetNackRanges()
|
||||
Expect(len(nackRanges)).To(Equal(2))
|
||||
Expect(nackRanges[0].FirstPacketNumber).To(Equal(protocol.PacketNumber(3)))
|
||||
Expect(nackRanges[0].Length).To(Equal(uint8(1)))
|
||||
Expect(nackRanges[1].FirstPacketNumber).To(Equal(protocol.PacketNumber(7)))
|
||||
Expect(nackRanges[1].Length).To(Equal(uint8(1)))
|
||||
})
|
||||
|
||||
It("handles two sequences of lost packages", func() {
|
||||
ackHandler, _ := NewAckHandler()
|
||||
for i := 0; i < 10; i++ {
|
||||
if i == 2 || i == 3 || i == 4 || i == 7 || i == 8 {
|
||||
continue
|
||||
}
|
||||
ackHandler.HandlePacket(protocol.PacketNumber(i))
|
||||
}
|
||||
Expect(ackHandler.LargestObserved).To(Equal(protocol.PacketNumber(9)))
|
||||
nackRanges := ackHandler.GetNackRanges()
|
||||
Expect(len(nackRanges)).To(Equal(2))
|
||||
Expect(nackRanges[0].FirstPacketNumber).To(Equal(protocol.PacketNumber(2)))
|
||||
Expect(nackRanges[0].Length).To(Equal(uint8(3)))
|
||||
Expect(nackRanges[1].FirstPacketNumber).To(Equal(protocol.PacketNumber(7)))
|
||||
Expect(nackRanges[1].Length).To(Equal(uint8(2)))
|
||||
})
|
||||
|
||||
})
|
||||
13
ackhandler/ackhandler_suite_test.go
Normal file
13
ackhandler/ackhandler_suite_test.go
Normal file
@@ -0,0 +1,13 @@
|
||||
package ackhandler
|
||||
|
||||
import (
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestCrypto(t *testing.T) {
|
||||
RegisterFailHandler(Fail)
|
||||
RunSpecs(t, "AckHandler Suite")
|
||||
}
|
||||
Reference in New Issue
Block a user