mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 12:47:36 +03:00
implement sending of STREAM_ID_BLOCKED frames
This commit is contained in:
parent
4f364a7c24
commit
1ec720f2f2
8 changed files with 120 additions and 26 deletions
|
@ -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:
|
||||||
|
|
|
@ -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"))
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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))
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue