mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
145 lines
4.5 KiB
Go
145 lines
4.5 KiB
Go
package wire
|
|
|
|
import (
|
|
"bytes"
|
|
"io"
|
|
|
|
"github.com/quic-go/quic-go/internal/protocol"
|
|
"github.com/quic-go/quic-go/quicvarint"
|
|
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
)
|
|
|
|
var _ = Describe("CRYPTO frame", func() {
|
|
Context("when parsing", func() {
|
|
It("parses", func() {
|
|
data := encodeVarInt(0xdecafbad) // offset
|
|
data = append(data, encodeVarInt(6)...) // length
|
|
data = append(data, []byte("foobar")...)
|
|
r := bytes.NewReader(data)
|
|
frame, err := parseCryptoFrame(r, protocol.Version1)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(frame.Offset).To(Equal(protocol.ByteCount(0xdecafbad)))
|
|
Expect(frame.Data).To(Equal([]byte("foobar")))
|
|
Expect(r.Len()).To(BeZero())
|
|
})
|
|
|
|
It("errors on EOFs", func() {
|
|
data := encodeVarInt(0xdecafbad) // offset
|
|
data = append(data, encodeVarInt(6)...) // data length
|
|
data = append(data, []byte("foobar")...)
|
|
r := bytes.NewReader(data)
|
|
_, err := parseCryptoFrame(r, protocol.Version1)
|
|
Expect(err).NotTo(HaveOccurred())
|
|
for i := range data {
|
|
_, err := parseCryptoFrame(bytes.NewReader(data[:i]), protocol.Version1)
|
|
Expect(err).To(MatchError(io.EOF))
|
|
}
|
|
})
|
|
})
|
|
|
|
Context("when writing", func() {
|
|
It("writes a frame", func() {
|
|
f := &CryptoFrame{
|
|
Offset: 0x123456,
|
|
Data: []byte("foobar"),
|
|
}
|
|
b, err := f.Append(nil, protocol.Version1)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
expected := []byte{cryptoFrameType}
|
|
expected = append(expected, encodeVarInt(0x123456)...) // offset
|
|
expected = append(expected, encodeVarInt(6)...) // length
|
|
expected = append(expected, []byte("foobar")...)
|
|
Expect(b).To(Equal(expected))
|
|
})
|
|
})
|
|
|
|
Context("max data length", func() {
|
|
const maxSize = 3000
|
|
|
|
It("always returns a data length such that the resulting frame has the right size", func() {
|
|
data := make([]byte, maxSize)
|
|
f := &CryptoFrame{
|
|
Offset: 0xdeadbeef,
|
|
}
|
|
var frameOneByteTooSmallCounter int
|
|
for i := 1; i < maxSize; i++ {
|
|
f.Data = nil
|
|
maxDataLen := f.MaxDataLen(protocol.ByteCount(i))
|
|
if maxDataLen == 0 { // 0 means that no valid CRYTPO frame can be written
|
|
// check that writing a minimal size CRYPTO frame (i.e. with 1 byte data) is actually larger than the desired size
|
|
f.Data = []byte{0}
|
|
b, err := f.Append(nil, protocol.Version1)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(len(b)).To(BeNumerically(">", i))
|
|
continue
|
|
}
|
|
f.Data = data[:int(maxDataLen)]
|
|
b, err := f.Append(nil, protocol.Version1)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
// There's *one* pathological case, where a data length of x can be encoded into 1 byte
|
|
// but a data lengths of x+1 needs 2 bytes
|
|
// In that case, it's impossible to create a STREAM frame of the desired size
|
|
if len(b) == i-1 {
|
|
frameOneByteTooSmallCounter++
|
|
continue
|
|
}
|
|
Expect(len(b)).To(Equal(i))
|
|
}
|
|
Expect(frameOneByteTooSmallCounter).To(Equal(1))
|
|
})
|
|
})
|
|
|
|
Context("length", func() {
|
|
It("has the right length for a frame without offset and data length", func() {
|
|
f := &CryptoFrame{
|
|
Offset: 0x1337,
|
|
Data: []byte("foobar"),
|
|
}
|
|
Expect(f.Length(protocol.Version1)).To(Equal(1 + quicvarint.Len(0x1337) + quicvarint.Len(6) + 6))
|
|
})
|
|
})
|
|
|
|
Context("splitting", func() {
|
|
It("splits a frame", func() {
|
|
f := &CryptoFrame{
|
|
Offset: 0x1337,
|
|
Data: []byte("foobar"),
|
|
}
|
|
hdrLen := f.Length(protocol.Version1) - 6
|
|
new, needsSplit := f.MaybeSplitOffFrame(hdrLen+3, protocol.Version1)
|
|
Expect(needsSplit).To(BeTrue())
|
|
Expect(new.Data).To(Equal([]byte("foo")))
|
|
Expect(new.Offset).To(Equal(protocol.ByteCount(0x1337)))
|
|
Expect(f.Data).To(Equal([]byte("bar")))
|
|
Expect(f.Offset).To(Equal(protocol.ByteCount(0x1337 + 3)))
|
|
})
|
|
|
|
It("doesn't split if there's enough space in the frame", func() {
|
|
f := &CryptoFrame{
|
|
Offset: 0x1337,
|
|
Data: []byte("foobar"),
|
|
}
|
|
f, needsSplit := f.MaybeSplitOffFrame(f.Length(protocol.Version1), protocol.Version1)
|
|
Expect(needsSplit).To(BeFalse())
|
|
Expect(f).To(BeNil())
|
|
})
|
|
|
|
It("doesn't split if the size is too small", func() {
|
|
f := &CryptoFrame{
|
|
Offset: 0x1337,
|
|
Data: []byte("foobar"),
|
|
}
|
|
length := f.Length(protocol.Version1) - 6
|
|
for i := protocol.ByteCount(0); i <= length; i++ {
|
|
f, needsSplit := f.MaybeSplitOffFrame(i, protocol.Version1)
|
|
Expect(needsSplit).To(BeTrue())
|
|
Expect(f).To(BeNil())
|
|
}
|
|
f, needsSplit := f.MaybeSplitOffFrame(length+1, protocol.Version1)
|
|
Expect(needsSplit).To(BeTrue())
|
|
Expect(f).ToNot(BeNil())
|
|
})
|
|
})
|
|
})
|