implement sending of STREAM_ID_BLOCKED frames

This commit is contained in:
Marten Seemann 2018-02-05 16:03:02 +08:00
parent 4f364a7c24
commit 1ec720f2f2
8 changed files with 120 additions and 26 deletions

View file

@ -568,6 +568,7 @@ func (s *session) handleFrames(fs []wire.Frame, encLevel protocol.EncryptionLeve
err = s.handleMaxStreamIDFrame(frame) err = s.handleMaxStreamIDFrame(frame)
case *wire.BlockedFrame: case *wire.BlockedFrame:
case *wire.StreamBlockedFrame: case *wire.StreamBlockedFrame:
case *wire.StreamIDBlockedFrame:
case *wire.StopSendingFrame: case *wire.StopSendingFrame:
err = s.handleStopSendingFrame(frame) err = s.handleStopSendingFrame(frame)
case *wire.PingFrame: case *wire.PingFrame:

View file

@ -415,6 +415,16 @@ var _ = Describe("Session", func() {
Expect(err).NotTo(HaveOccurred()) Expect(err).NotTo(HaveOccurred())
}) })
It("handles STREAM_BLOCKED frames", func() {
err := sess.handleFrames([]wire.Frame{&wire.StreamBlockedFrame{}}, protocol.EncryptionUnspecified)
Expect(err).NotTo(HaveOccurred())
})
It("handles STREAM_ID_BLOCKED frames", func() {
err := sess.handleFrames([]wire.Frame{&wire.StreamIDBlockedFrame{}}, protocol.EncryptionUnspecified)
Expect(err).NotTo(HaveOccurred())
})
It("errors on GOAWAY frames", func() { It("errors on GOAWAY frames", func() {
err := sess.handleFrames([]wire.Frame{&wire.GoawayFrame{}}, protocol.EncryptionUnspecified) err := sess.handleFrames([]wire.Frame{&wire.GoawayFrame{}}, protocol.EncryptionUnspecified)
Expect(err).To(MatchError("unimplemented: handling GOAWAY frames")) Expect(err).To(MatchError("unimplemented: handling GOAWAY frames"))

View file

@ -64,7 +64,11 @@ func newStreamsMap(
newUniReceiveStream := func(id protocol.StreamID) receiveStreamI { newUniReceiveStream := func(id protocol.StreamID) receiveStreamI {
return newReceiveStream(id, m.sender, m.newFlowController(id), version) return newReceiveStream(id, m.sender, m.newFlowController(id), version)
} }
m.outgoingBidiStreams = newOutgoingBidiStreamsMap(firstOutgoingBidiStream, newBidiStream) m.outgoingBidiStreams = newOutgoingBidiStreamsMap(
firstOutgoingBidiStream,
newBidiStream,
sender.queueControlFrame,
)
// TODO(#523): make these values configurable // TODO(#523): make these values configurable
m.incomingBidiStreams = newIncomingBidiStreamsMap( m.incomingBidiStreams = newIncomingBidiStreamsMap(
firstIncomingBidiStream, firstIncomingBidiStream,
@ -73,7 +77,11 @@ func newStreamsMap(
sender.queueControlFrame, sender.queueControlFrame,
newBidiStream, newBidiStream,
) )
m.outgoingUniStreams = newOutgoingUniStreamsMap(firstOutgoingUniStream, newUniSendStream) m.outgoingUniStreams = newOutgoingUniStreamsMap(
firstOutgoingUniStream,
newUniSendStream,
sender.queueControlFrame,
)
// TODO(#523): make these values configurable // TODO(#523): make these values configurable
m.incomingUniStreams = newIncomingUniStreamsMap( m.incomingUniStreams = newIncomingUniStreamsMap(
firstIncomingUniStream, firstIncomingUniStream,

View file

@ -9,6 +9,7 @@ import (
"sync" "sync"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/wire"
"github.com/lucas-clemente/quic-go/qerr" "github.com/lucas-clemente/quic-go/qerr"
) )
@ -18,18 +19,26 @@ type outgoingBidiStreamsMap struct {
streams map[protocol.StreamID]streamI streams map[protocol.StreamID]streamI
nextStream protocol.StreamID nextStream protocol.StreamID // stream ID of the stream returned by OpenStream(Sync)
maxStream protocol.StreamID maxStream protocol.StreamID // the maximum stream ID we're allowed to open
newStream func(protocol.StreamID) streamI highestBlocked protocol.StreamID // the highest stream ID that we queued a STREAM_ID_BLOCKED frame for
newStream func(protocol.StreamID) streamI
queueStreamIDBlocked func(*wire.StreamIDBlockedFrame)
closeErr error closeErr error
} }
func newOutgoingBidiStreamsMap(nextStream protocol.StreamID, newStream func(protocol.StreamID) streamI) *outgoingBidiStreamsMap { func newOutgoingBidiStreamsMap(
nextStream protocol.StreamID,
newStream func(protocol.StreamID) streamI,
queueControlFrame func(wire.Frame),
) *outgoingBidiStreamsMap {
m := &outgoingBidiStreamsMap{ m := &outgoingBidiStreamsMap{
streams: make(map[protocol.StreamID]streamI), streams: make(map[protocol.StreamID]streamI),
nextStream: nextStream, nextStream: nextStream,
newStream: newStream, newStream: newStream,
queueStreamIDBlocked: func(f *wire.StreamIDBlockedFrame) { queueControlFrame(f) },
} }
m.cond.L = &m.mutex m.cond.L = &m.mutex
return m return m
@ -63,6 +72,10 @@ func (m *outgoingBidiStreamsMap) openStreamImpl() (streamI, error) {
return nil, m.closeErr return nil, m.closeErr
} }
if m.nextStream > m.maxStream { if m.nextStream > m.maxStream {
if m.maxStream == 0 || m.highestBlocked < m.maxStream {
m.queueStreamIDBlocked(&wire.StreamIDBlockedFrame{StreamID: m.maxStream})
m.highestBlocked = m.maxStream
}
return nil, qerr.TooManyOpenStreams return nil, qerr.TooManyOpenStreams
} }
s := m.newStream(m.nextStream) s := m.newStream(m.nextStream)

View file

@ -6,6 +6,7 @@ import (
"github.com/cheekybits/genny/generic" "github.com/cheekybits/genny/generic"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/wire"
"github.com/lucas-clemente/quic-go/qerr" "github.com/lucas-clemente/quic-go/qerr"
) )
@ -19,18 +20,26 @@ type outgoingItemsMap struct {
streams map[protocol.StreamID]item streams map[protocol.StreamID]item
nextStream protocol.StreamID nextStream protocol.StreamID // stream ID of the stream returned by OpenStream(Sync)
maxStream protocol.StreamID maxStream protocol.StreamID // the maximum stream ID we're allowed to open
newStream func(protocol.StreamID) item highestBlocked protocol.StreamID // the highest stream ID that we queued a STREAM_ID_BLOCKED frame for
newStream func(protocol.StreamID) item
queueStreamIDBlocked func(*wire.StreamIDBlockedFrame)
closeErr error closeErr error
} }
func newOutgoingItemsMap(nextStream protocol.StreamID, newStream func(protocol.StreamID) item) *outgoingItemsMap { func newOutgoingItemsMap(
nextStream protocol.StreamID,
newStream func(protocol.StreamID) item,
queueControlFrame func(wire.Frame),
) *outgoingItemsMap {
m := &outgoingItemsMap{ m := &outgoingItemsMap{
streams: make(map[protocol.StreamID]item), streams: make(map[protocol.StreamID]item),
nextStream: nextStream, nextStream: nextStream,
newStream: newStream, newStream: newStream,
queueStreamIDBlocked: func(f *wire.StreamIDBlockedFrame) { queueControlFrame(f) },
} }
m.cond.L = &m.mutex m.cond.L = &m.mutex
return m return m
@ -64,6 +73,10 @@ func (m *outgoingItemsMap) openStreamImpl() (item, error) {
return nil, m.closeErr return nil, m.closeErr
} }
if m.nextStream > m.maxStream { if m.nextStream > m.maxStream {
if m.maxStream == 0 || m.highestBlocked < m.maxStream {
m.queueStreamIDBlocked(&wire.StreamIDBlockedFrame{StreamID: m.maxStream})
m.highestBlocked = m.maxStream
}
return nil, qerr.TooManyOpenStreams return nil, qerr.TooManyOpenStreams
} }
s := m.newStream(m.nextStream) s := m.newStream(m.nextStream)

View file

@ -3,7 +3,9 @@ package quic
import ( import (
"errors" "errors"
"github.com/golang/mock/gomock"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/wire"
"github.com/lucas-clemente/quic-go/qerr" "github.com/lucas-clemente/quic-go/qerr"
. "github.com/onsi/ginkgo" . "github.com/onsi/ginkgo"
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
@ -12,15 +14,17 @@ import (
var _ = Describe("Streams Map (outgoing)", func() { var _ = Describe("Streams Map (outgoing)", func() {
const firstNewStream protocol.StreamID = 10 const firstNewStream protocol.StreamID = 10
var ( var (
m *outgoingItemsMap m *outgoingItemsMap
newItem func(id protocol.StreamID) item newItem func(id protocol.StreamID) item
mockSender *MockStreamSender
) )
BeforeEach(func() { BeforeEach(func() {
newItem = func(id protocol.StreamID) item { newItem = func(id protocol.StreamID) item {
return id return id
} }
m = newOutgoingItemsMap(firstNewStream, newItem) mockSender = NewMockStreamSender(mockCtrl)
m = newOutgoingItemsMap(firstNewStream, newItem, mockSender.queueControlFrame)
}) })
Context("no stream ID limit", func() { Context("no stream ID limit", func() {
@ -84,11 +88,13 @@ var _ = Describe("Streams Map (outgoing)", func() {
Context("with stream ID limits", func() { Context("with stream ID limits", func() {
It("errors when no stream can be opened immediately", func() { It("errors when no stream can be opened immediately", func() {
mockSender.EXPECT().queueControlFrame(gomock.Any())
_, err := m.OpenStream() _, err := m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams)) Expect(err).To(MatchError(qerr.TooManyOpenStreams))
}) })
It("blocks until a stream can be opened synchronously", func() { It("blocks until a stream can be opened synchronously", func() {
mockSender.EXPECT().queueControlFrame(gomock.Any())
done := make(chan struct{}) done := make(chan struct{})
go func() { go func() {
defer GinkgoRecover() defer GinkgoRecover()
@ -104,6 +110,7 @@ var _ = Describe("Streams Map (outgoing)", func() {
}) })
It("stops opening synchronously when it is closed", func() { It("stops opening synchronously when it is closed", func() {
mockSender.EXPECT().queueControlFrame(gomock.Any())
testErr := errors.New("test error") testErr := errors.New("test error")
done := make(chan struct{}) done := make(chan struct{})
go func() { go func() {
@ -125,5 +132,26 @@ var _ = Describe("Streams Map (outgoing)", func() {
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
Expect(str).To(Equal(firstNewStream)) Expect(str).To(Equal(firstNewStream))
}) })
It("queues a STREAM_ID_BLOCKED frame if no stream can be opened", func() {
m.SetMaxStream(firstNewStream)
mockSender.EXPECT().queueControlFrame(&wire.StreamIDBlockedFrame{StreamID: firstNewStream})
_, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
_, err = m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams))
})
It("only sends one STREAM_ID_BLOCKED frame for one stream ID", func() {
m.SetMaxStream(firstNewStream)
mockSender.EXPECT().queueControlFrame(&wire.StreamIDBlockedFrame{StreamID: firstNewStream})
_, err := m.OpenStream()
Expect(err).ToNot(HaveOccurred())
// try to open a stream twice, but expect only one STREAM_ID_BLOCKED to be sent
_, err = m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams))
_, err = m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams))
})
}) })
}) })

View file

@ -9,6 +9,7 @@ import (
"sync" "sync"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/wire"
"github.com/lucas-clemente/quic-go/qerr" "github.com/lucas-clemente/quic-go/qerr"
) )
@ -18,18 +19,26 @@ type outgoingUniStreamsMap struct {
streams map[protocol.StreamID]sendStreamI streams map[protocol.StreamID]sendStreamI
nextStream protocol.StreamID nextStream protocol.StreamID // stream ID of the stream returned by OpenStream(Sync)
maxStream protocol.StreamID maxStream protocol.StreamID // the maximum stream ID we're allowed to open
newStream func(protocol.StreamID) sendStreamI highestBlocked protocol.StreamID // the highest stream ID that we queued a STREAM_ID_BLOCKED frame for
newStream func(protocol.StreamID) sendStreamI
queueStreamIDBlocked func(*wire.StreamIDBlockedFrame)
closeErr error closeErr error
} }
func newOutgoingUniStreamsMap(nextStream protocol.StreamID, newStream func(protocol.StreamID) sendStreamI) *outgoingUniStreamsMap { func newOutgoingUniStreamsMap(
nextStream protocol.StreamID,
newStream func(protocol.StreamID) sendStreamI,
queueControlFrame func(wire.Frame),
) *outgoingUniStreamsMap {
m := &outgoingUniStreamsMap{ m := &outgoingUniStreamsMap{
streams: make(map[protocol.StreamID]sendStreamI), streams: make(map[protocol.StreamID]sendStreamI),
nextStream: nextStream, nextStream: nextStream,
newStream: newStream, newStream: newStream,
queueStreamIDBlocked: func(f *wire.StreamIDBlockedFrame) { queueControlFrame(f) },
} }
m.cond.L = &m.mutex m.cond.L = &m.mutex
return m return m
@ -63,6 +72,10 @@ func (m *outgoingUniStreamsMap) openStreamImpl() (sendStreamI, error) {
return nil, m.closeErr return nil, m.closeErr
} }
if m.nextStream > m.maxStream { if m.nextStream > m.maxStream {
if m.maxStream == 0 || m.highestBlocked < m.maxStream {
m.queueStreamIDBlocked(&wire.StreamIDBlockedFrame{StreamID: m.maxStream})
m.highestBlocked = m.maxStream
}
return nil, qerr.TooManyOpenStreams return nil, qerr.TooManyOpenStreams
} }
s := m.newStream(m.nextStream) s := m.newStream(m.nextStream)

View file

@ -257,6 +257,10 @@ var _ = Describe("Streams Map (for IETF QUIC)", func() {
}) })
Context("updating stream ID limits", func() { Context("updating stream ID limits", func() {
BeforeEach(func() {
mockSender.EXPECT().queueControlFrame(gomock.Any())
})
It("processes the parameter for outgoing bidirectional streams", func() { It("processes the parameter for outgoing bidirectional streams", func() {
_, err := m.OpenStream() _, err := m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams)) Expect(err).To(MatchError(qerr.TooManyOpenStreams))
@ -281,6 +285,10 @@ var _ = Describe("Streams Map (for IETF QUIC)", func() {
}) })
Context("handling MAX_STREAM_ID frames", func() { Context("handling MAX_STREAM_ID frames", func() {
BeforeEach(func() {
mockSender.EXPECT().queueControlFrame(gomock.Any()).AnyTimes()
})
It("processes IDs for outgoing bidirectional streams", func() { It("processes IDs for outgoing bidirectional streams", func() {
_, err := m.OpenStream() _, err := m.OpenStream()
Expect(err).To(MatchError(qerr.TooManyOpenStreams)) Expect(err).To(MatchError(qerr.TooManyOpenStreams))