route/vendor/github.com/lucas-clemente/quic-go/streams_map_test.go

416 lines
12 KiB
Go
Raw Normal View History

2017-12-12 02:51:45 +00:00
package quic
import (
"errors"
2018-01-20 18:07:01 +00:00
"github.com/golang/mock/gomock"
"github.com/lucas-clemente/quic-go/internal/handshake"
2018-01-03 19:19:49 +00:00
"github.com/lucas-clemente/quic-go/internal/protocol"
2018-01-20 18:07:01 +00:00
"github.com/lucas-clemente/quic-go/internal/wire"
2018-01-03 19:19:49 +00:00
2017-12-12 02:51:45 +00:00
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
)
2018-01-20 18:07:01 +00:00
var _ = Describe("Streams Map (for IETF QUIC)", func() {
var m *streamsMap
2017-12-12 02:51:45 +00:00
2018-01-03 19:19:49 +00:00
newStream := func(id protocol.StreamID) streamI {
2018-01-20 18:07:01 +00:00
str := NewMockStreamI(mockCtrl)
2018-01-03 19:19:49 +00:00
str.EXPECT().StreamID().Return(id).AnyTimes()
return str
}
2018-01-20 18:07:01 +00:00
setNewStreamsMap := func(p protocol.Perspective) {
m = newStreamsMap(newStream, p).(*streamsMap)
2017-12-12 02:51:45 +00:00
}
2018-01-03 19:19:49 +00:00
deleteStream := func(id protocol.StreamID) {
2018-01-20 18:07:01 +00:00
ExpectWithOffset(1, m.DeleteStream(id)).To(Succeed())
2018-01-03 19:19:49 +00:00
}
2017-12-12 02:51:45 +00:00
Context("getting and creating streams", func() {
Context("as a server", func() {
BeforeEach(func() {
2018-01-20 18:07:01 +00:00
setNewStreamsMap(protocol.PerspectiveServer)
2017-12-12 02:51:45 +00:00
})
Context("client-side streams", func() {
It("gets new streams", func() {
s, err := m.GetOrOpenStream(1)
Expect(err).NotTo(HaveOccurred())
2018-01-20 18:07:01 +00:00
Expect(s).ToNot(BeNil())
2017-12-12 02:51:45 +00:00
Expect(s.StreamID()).To(Equal(protocol.StreamID(1)))
2018-01-20 18:07:01 +00:00
Expect(m.streams).To(HaveLen(1))
2017-12-12 02:51:45 +00:00
})
It("rejects streams with even IDs", func() {
_, err := m.GetOrOpenStream(6)
2018-01-03 19:19:49 +00:00
Expect(err).To(MatchError("InvalidStreamID: peer attempted to open stream 6"))
})
It("rejects streams with even IDs, which are lower thatn the highest client-side stream", func() {
_, err := m.GetOrOpenStream(5)
Expect(err).NotTo(HaveOccurred())
_, err = m.GetOrOpenStream(4)
Expect(err).To(MatchError("InvalidStreamID: peer attempted to open stream 4"))
2017-12-12 02:51:45 +00:00
})
It("gets existing streams", func() {
s, err := m.GetOrOpenStream(5)
Expect(err).NotTo(HaveOccurred())
2018-01-20 18:07:01 +00:00
numStreams := len(m.streams)
2017-12-12 02:51:45 +00:00
s, err = m.GetOrOpenStream(5)
Expect(err).NotTo(HaveOccurred())
Expect(s.StreamID()).To(Equal(protocol.StreamID(5)))
2018-01-20 18:07:01 +00:00
Expect(m.streams).To(HaveLen(numStreams))
2017-12-12 02:51:45 +00:00
})
It("returns nil for closed streams", func() {
2018-01-03 19:19:49 +00:00
_, err := m.GetOrOpenStream(5)
2017-12-12 02:51:45 +00:00
Expect(err).NotTo(HaveOccurred())
2018-01-03 19:19:49 +00:00
deleteStream(5)
s, err := m.GetOrOpenStream(5)
2017-12-12 02:51:45 +00:00
Expect(err).NotTo(HaveOccurred())
Expect(s).To(BeNil())
})
It("opens skipped streams", func() {
2018-01-20 18:07:01 +00:00
_, err := m.GetOrOpenStream(7)
2017-12-12 02:51:45 +00:00
Expect(err).NotTo(HaveOccurred())
Expect(m.streams).To(HaveKey(protocol.StreamID(3)))
Expect(m.streams).To(HaveKey(protocol.StreamID(5)))
2018-01-20 18:07:01 +00:00
Expect(m.streams).To(HaveKey(protocol.StreamID(7)))
2017-12-12 02:51:45 +00:00
})
It("doesn't reopen an already closed stream", func() {
_, err := m.GetOrOpenStream(5)
Expect(err).ToNot(HaveOccurred())
2018-01-03 19:19:49 +00:00
deleteStream(5)
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
str, err := m.GetOrOpenStream(5)
Expect(err).ToNot(HaveOccurred())
Expect(str).To(BeNil())
})
})
Context("server-side streams", func() {
It("opens a stream 2 first", func() {
s, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
Expect(s).ToNot(BeNil())
Expect(s.StreamID()).To(Equal(protocol.StreamID(2)))
})
2018-01-03 19:19:49 +00:00
It("returns the error when the streamsMap was closed", func() {
2017-12-12 02:51:45 +00:00
testErr := errors.New("test error")
2018-01-03 19:19:49 +00:00
m.CloseWithError(testErr)
2017-12-12 02:51:45 +00:00
_, err := m.OpenStream()
Expect(err).To(MatchError(testErr))
})
2018-01-03 19:19:49 +00:00
It("doesn't reopen an already closed stream", func() {
str, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
Expect(str.StreamID()).To(Equal(protocol.StreamID(2)))
deleteStream(2)
Expect(err).ToNot(HaveOccurred())
str, err = m.GetOrOpenStream(2)
Expect(err).ToNot(HaveOccurred())
Expect(str).To(BeNil())
})
2017-12-12 02:51:45 +00:00
Context("opening streams synchronously", func() {
It("immediately returns when OpenStreamSync is called after an error was registered", func() {
testErr := errors.New("test error")
m.CloseWithError(testErr)
_, err := m.OpenStreamSync()
Expect(err).To(MatchError(testErr))
})
})
})
Context("accepting streams", func() {
It("does nothing if no stream is opened", func() {
var accepted bool
go func() {
_, _ = m.AcceptStream()
accepted = true
}()
Consistently(func() bool { return accepted }).Should(BeFalse())
})
2018-01-20 18:07:01 +00:00
It("starts with stream 1", func() {
var str Stream
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
_, err := m.GetOrOpenStream(1)
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
2017-12-12 02:51:45 +00:00
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
})
It("returns an implicitly opened stream, if a stream number is skipped", func() {
2018-01-20 18:07:01 +00:00
var str Stream
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
_, err := m.GetOrOpenStream(3)
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
2017-12-12 02:51:45 +00:00
})
It("returns to multiple accepts", func() {
2018-01-20 18:07:01 +00:00
var str1, str2 Stream
done1 := make(chan struct{})
done2 := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str1, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done1)
2017-12-12 02:51:45 +00:00
}()
go func() {
defer GinkgoRecover()
var err error
str2, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done2)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
_, err := m.GetOrOpenStream(3) // opens stream 1 and 3
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done1).Should(BeClosed())
Eventually(done2).Should(BeClosed())
2017-12-12 02:51:45 +00:00
Expect(str1.StreamID()).ToNot(Equal(str2.StreamID()))
2018-01-20 18:07:01 +00:00
Expect(str1.StreamID() + str2.StreamID()).To(BeEquivalentTo(1 + 3))
2017-12-12 02:51:45 +00:00
})
2018-01-20 18:07:01 +00:00
It("waits until a new stream is available", func() {
var str Stream
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
Consistently(done).ShouldNot(BeClosed())
_, err := m.GetOrOpenStream(1)
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
2017-12-12 02:51:45 +00:00
})
It("returns multiple streams on subsequent Accept calls, if available", func() {
2018-01-20 18:07:01 +00:00
var str Stream
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
_, err := m.GetOrOpenStream(3)
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
2017-12-12 02:51:45 +00:00
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Expect(str.StreamID()).To(Equal(protocol.StreamID(3)))
2017-12-12 02:51:45 +00:00
})
It("blocks after accepting a stream", func() {
2018-01-20 18:07:01 +00:00
_, err := m.GetOrOpenStream(1)
2017-12-12 02:51:45 +00:00
Expect(err).ToNot(HaveOccurred())
str, err := m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
_, _ = m.AcceptStream()
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
Consistently(done).ShouldNot(BeClosed())
// make the go routine return
str.(*MockStreamI).EXPECT().closeForShutdown(gomock.Any())
m.CloseWithError(errors.New("shut down"))
Eventually(done).Should(BeClosed())
2017-12-12 02:51:45 +00:00
})
It("stops waiting when an error is registered", func() {
testErr := errors.New("testErr")
2018-01-20 18:07:01 +00:00
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
2018-01-20 18:07:01 +00:00
defer GinkgoRecover()
_, err := m.AcceptStream()
Expect(err).To(MatchError(testErr))
close(done)
2017-12-12 02:51:45 +00:00
}()
2018-01-20 18:07:01 +00:00
Consistently(done).ShouldNot(BeClosed())
2017-12-12 02:51:45 +00:00
m.CloseWithError(testErr)
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
2017-12-12 02:51:45 +00:00
})
It("immediately returns when Accept is called after an error was registered", func() {
testErr := errors.New("testErr")
m.CloseWithError(testErr)
_, err := m.AcceptStream()
Expect(err).To(MatchError(testErr))
})
})
})
Context("as a client", func() {
BeforeEach(func() {
2018-01-20 18:07:01 +00:00
setNewStreamsMap(protocol.PerspectiveClient)
2017-12-12 02:51:45 +00:00
})
2018-01-03 19:19:49 +00:00
Context("server-side streams", func() {
2017-12-12 02:51:45 +00:00
It("rejects streams with odd IDs", func() {
_, err := m.GetOrOpenStream(5)
2018-01-03 19:19:49 +00:00
Expect(err).To(MatchError("InvalidStreamID: peer attempted to open stream 5"))
})
2018-01-20 18:07:01 +00:00
It("rejects streams with odds IDs, which are lower than the highest server-side stream", func() {
2018-01-03 19:19:49 +00:00
_, err := m.GetOrOpenStream(6)
Expect(err).NotTo(HaveOccurred())
_, err = m.GetOrOpenStream(5)
Expect(err).To(MatchError("InvalidStreamID: peer attempted to open stream 5"))
2017-12-12 02:51:45 +00:00
})
It("gets new streams", func() {
s, err := m.GetOrOpenStream(2)
Expect(err).NotTo(HaveOccurred())
Expect(s.StreamID()).To(Equal(protocol.StreamID(2)))
2018-01-20 18:07:01 +00:00
Expect(m.streams).To(HaveLen(1))
2017-12-12 02:51:45 +00:00
})
It("opens skipped streams", func() {
_, err := m.GetOrOpenStream(6)
Expect(err).NotTo(HaveOccurred())
Expect(m.streams).To(HaveKey(protocol.StreamID(2)))
Expect(m.streams).To(HaveKey(protocol.StreamID(4)))
Expect(m.streams).To(HaveKey(protocol.StreamID(6)))
2018-01-03 19:19:49 +00:00
})
It("doesn't reopen an already closed stream", func() {
str, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Expect(str.StreamID()).To(Equal(protocol.StreamID(1)))
deleteStream(1)
2018-01-03 19:19:49 +00:00
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
str, err = m.GetOrOpenStream(1)
2018-01-03 19:19:49 +00:00
Expect(err).ToNot(HaveOccurred())
Expect(str).To(BeNil())
2017-12-12 02:51:45 +00:00
})
})
2018-01-03 19:19:49 +00:00
Context("client-side streams", func() {
2018-01-20 18:07:01 +00:00
It("starts with stream 1", func() {
setNewStreamsMap(protocol.PerspectiveClient)
2017-12-12 02:51:45 +00:00
s, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
Expect(s).ToNot(BeNil())
Expect(s.StreamID()).To(BeEquivalentTo(1))
})
It("opens multiple streams", func() {
s1, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
s2, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
Expect(s2.StreamID()).To(Equal(s1.StreamID() + 2))
})
2018-01-03 19:19:49 +00:00
It("doesn't reopen an already closed stream", func() {
_, err := m.GetOrOpenStream(4)
Expect(err).ToNot(HaveOccurred())
deleteStream(4)
Expect(err).ToNot(HaveOccurred())
str, err := m.GetOrOpenStream(4)
Expect(err).ToNot(HaveOccurred())
Expect(str).To(BeNil())
})
2017-12-12 02:51:45 +00:00
})
Context("accepting streams", func() {
It("accepts stream 2 first", func() {
2018-01-20 18:07:01 +00:00
var str Stream
done := make(chan struct{})
2017-12-12 02:51:45 +00:00
go func() {
defer GinkgoRecover()
var err error
str, err = m.AcceptStream()
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
close(done)
2017-12-12 02:51:45 +00:00
}()
_, err := m.GetOrOpenStream(2)
Expect(err).ToNot(HaveOccurred())
2018-01-20 18:07:01 +00:00
Eventually(done).Should(BeClosed())
2017-12-12 02:51:45 +00:00
Expect(str.StreamID()).To(Equal(protocol.StreamID(2)))
})
})
})
})
2018-01-20 18:07:01 +00:00
Context("deleting streams", func() {
2017-12-12 02:51:45 +00:00
BeforeEach(func() {
2018-01-20 18:07:01 +00:00
setNewStreamsMap(protocol.PerspectiveServer)
2017-12-12 02:51:45 +00:00
})
2018-01-20 18:07:01 +00:00
It("deletes an incoming stream", func() {
_, err := m.GetOrOpenStream(3) // open stream 1 and 3
Expect(err).ToNot(HaveOccurred())
err = m.DeleteStream(1)
Expect(err).ToNot(HaveOccurred())
Expect(m.streams).To(HaveLen(1))
Expect(m.streams).To(HaveKey(protocol.StreamID(3)))
2017-12-12 02:51:45 +00:00
})
2018-01-20 18:07:01 +00:00
It("deletes an outgoing stream", func() {
_, err := m.OpenStream() // open stream 2
Expect(err).ToNot(HaveOccurred())
_, err = m.OpenStream()
Expect(err).ToNot(HaveOccurred())
err = m.DeleteStream(2)
Expect(err).ToNot(HaveOccurred())
2017-12-12 02:51:45 +00:00
})
2018-01-20 18:07:01 +00:00
It("errors when the stream doesn't exist", func() {
err := m.DeleteStream(1337)
Expect(err).To(MatchError(errMapAccess))
})
})
2017-12-12 02:51:45 +00:00
2018-01-20 18:07:01 +00:00
It("sets the flow control limit", func() {
setNewStreamsMap(protocol.PerspectiveServer)
_, err := m.GetOrOpenStream(3)
Expect(err).ToNot(HaveOccurred())
m.streams[1].(*MockStreamI).EXPECT().handleMaxStreamDataFrame(&wire.MaxStreamDataFrame{
StreamID: 1,
ByteOffset: 321,
})
m.streams[3].(*MockStreamI).EXPECT().handleMaxStreamDataFrame(&wire.MaxStreamDataFrame{
StreamID: 3,
ByteOffset: 321,
2017-12-12 02:51:45 +00:00
})
2018-01-20 18:07:01 +00:00
m.UpdateLimits(&handshake.TransportParameters{StreamFlowControlWindow: 321})
2017-12-12 02:51:45 +00:00
})
})