Merge pull request #1028 from lucas-clemente/fix-986

add the offset to the BLOCKED and STREAM_BLOCKED frames
This commit is contained in:
Marten Seemann 2017-12-14 19:21:05 +07:00 committed by GitHub
commit 448928fc63
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
18 changed files with 116 additions and 53 deletions

View file

@ -19,7 +19,7 @@ type cryptoStreamI interface {
// methods needed for flow control // methods needed for flow control
GetWindowUpdate() protocol.ByteCount GetWindowUpdate() protocol.ByteCount
HandleMaxStreamDataFrame(*wire.MaxStreamDataFrame) HandleMaxStreamDataFrame(*wire.MaxStreamDataFrame)
IsFlowControlBlocked() bool IsFlowControlBlocked() (bool, protocol.ByteCount)
} }
type cryptoStream struct { type cryptoStream struct {

View file

@ -79,11 +79,16 @@ func (c *baseFlowController) getWindowUpdate() protocol.ByteCount {
return c.receiveWindow return c.receiveWindow
} }
func (c *baseFlowController) IsBlocked() bool { // IsBlocked says if it is blocked by flow control.
// If it is blocked, the offset is returned.
func (c *baseFlowController) IsBlocked() (bool, protocol.ByteCount) {
c.mutex.RLock() c.mutex.RLock()
defer c.mutex.RUnlock() defer c.mutex.RUnlock()
return c.sendWindowSize() == 0 if c.sendWindowSize() != 0 {
return false, 0
}
return true, c.sendWindow
} }
// maybeAdjustWindowIncrement increases the receiveWindowIncrement if we're sending updates too often. // maybeAdjustWindowIncrement increases the receiveWindowIncrement if we're sending updates too often.

View file

@ -54,7 +54,9 @@ var _ = Describe("Base Flow controller", func() {
controller.UpdateSendWindow(100) controller.UpdateSendWindow(100)
Expect(controller.IsBlocked()).To(BeFalse()) Expect(controller.IsBlocked()).To(BeFalse())
controller.AddBytesSent(100) controller.AddBytesSent(100)
Expect(controller.IsBlocked()).To(BeTrue()) blocked, offset := controller.IsBlocked()
Expect(blocked).To(BeTrue())
Expect(offset).To(Equal(protocol.ByteCount(100)))
}) })
}) })

View file

@ -5,7 +5,7 @@ import "github.com/lucas-clemente/quic-go/internal/protocol"
type flowController interface { type flowController interface {
// for sending // for sending
SendWindowSize() protocol.ByteCount SendWindowSize() protocol.ByteCount
IsBlocked() bool IsBlocked() (bool, protocol.ByteCount)
UpdateSendWindow(protocol.ByteCount) UpdateSendWindow(protocol.ByteCount)
AddBytesSent(protocol.ByteCount) AddBytesSent(protocol.ByteCount)
// for receiving // for receiving

View file

@ -233,7 +233,8 @@ var _ = Describe("Stream Flow controller", func() {
controller.connection.UpdateSendWindow(50) controller.connection.UpdateSendWindow(50)
controller.UpdateSendWindow(100) controller.UpdateSendWindow(100)
controller.AddBytesSent(50) controller.AddBytesSent(50)
Expect(controller.connection.IsBlocked()).To(BeTrue()) blocked, _ := controller.connection.IsBlocked()
Expect(blocked).To(BeTrue())
Expect(controller.IsBlocked()).To(BeFalse()) Expect(controller.IsBlocked()).To(BeFalse())
}) })
}) })

View file

@ -66,10 +66,11 @@ func (_mr *MockConnectionFlowControllerMockRecorder) GetWindowUpdate() *gomock.C
} }
// IsBlocked mocks base method // IsBlocked mocks base method
func (_m *MockConnectionFlowController) IsBlocked() bool { func (_m *MockConnectionFlowController) IsBlocked() (bool, protocol.ByteCount) {
ret := _m.ctrl.Call(_m, "IsBlocked") ret := _m.ctrl.Call(_m, "IsBlocked")
ret0, _ := ret[0].(bool) ret0, _ := ret[0].(bool)
return ret0 ret1, _ := ret[1].(protocol.ByteCount)
return ret0, ret1
} }
// IsBlocked indicates an expected call of IsBlocked // IsBlocked indicates an expected call of IsBlocked

View file

@ -129,10 +129,11 @@ func (_mr *MockStreamIMockRecorder) HandleStreamFrame(arg0 interface{}) *gomock.
} }
// IsFlowControlBlocked mocks base method // IsFlowControlBlocked mocks base method
func (_m *MockStreamI) IsFlowControlBlocked() bool { func (_m *MockStreamI) IsFlowControlBlocked() (bool, protocol.ByteCount) {
ret := _m.ctrl.Call(_m, "IsFlowControlBlocked") ret := _m.ctrl.Call(_m, "IsFlowControlBlocked")
ret0, _ := ret[0].(bool) ret0, _ := ret[0].(bool)
return ret0 ret1, _ := ret[1].(protocol.ByteCount)
return ret0, ret1
} }
// IsFlowControlBlocked indicates an expected call of IsFlowControlBlocked // IsFlowControlBlocked indicates an expected call of IsFlowControlBlocked

View file

@ -66,10 +66,11 @@ func (_mr *MockStreamFlowControllerMockRecorder) GetWindowUpdate() *gomock.Call
} }
// IsBlocked mocks base method // IsBlocked mocks base method
func (_m *MockStreamFlowController) IsBlocked() bool { func (_m *MockStreamFlowController) IsBlocked() (bool, protocol.ByteCount) {
ret := _m.ctrl.Call(_m, "IsBlocked") ret := _m.ctrl.Call(_m, "IsBlocked")
ret0, _ := ret[0].(bool) ret0, _ := ret[0].(bool)
return ret0 ret1, _ := ret[1].(protocol.ByteCount)
return ret0, ret1
} }
// IsBlocked indicates an expected call of IsBlocked // IsBlocked indicates an expected call of IsBlocked

View file

@ -4,17 +4,26 @@ import (
"bytes" "bytes"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/utils"
) )
// A BlockedFrame is a BLOCKED frame // A BlockedFrame is a BLOCKED frame
type BlockedFrame struct{} type BlockedFrame struct {
Offset protocol.ByteCount
}
// ParseBlockedFrame parses a BLOCKED frame // ParseBlockedFrame parses a BLOCKED frame
func ParseBlockedFrame(r *bytes.Reader, version protocol.VersionNumber) (*BlockedFrame, error) { func ParseBlockedFrame(r *bytes.Reader, _ protocol.VersionNumber) (*BlockedFrame, error) {
if _, err := r.ReadByte(); err != nil { if _, err := r.ReadByte(); err != nil {
return nil, err return nil, err
} }
return &BlockedFrame{}, nil offset, err := utils.ReadVarInt(r)
if err != nil {
return nil, err
}
return &BlockedFrame{
Offset: protocol.ByteCount(offset),
}, nil
} }
func (f *BlockedFrame) Write(b *bytes.Buffer, version protocol.VersionNumber) error { func (f *BlockedFrame) Write(b *bytes.Buffer, version protocol.VersionNumber) error {
@ -23,13 +32,14 @@ func (f *BlockedFrame) Write(b *bytes.Buffer, version protocol.VersionNumber) er
} }
typeByte := uint8(0x08) typeByte := uint8(0x08)
b.WriteByte(typeByte) b.WriteByte(typeByte)
utils.WriteVarInt(b, uint64(f.Offset))
return nil return nil
} }
// MinLength of a written frame // MinLength of a written frame
func (f *BlockedFrame) MinLength(version protocol.VersionNumber) protocol.ByteCount { func (f *BlockedFrame) MinLength(version protocol.VersionNumber) protocol.ByteCount {
if !version.UsesIETFFrameFormat() { // writing this frame would result in a legacy BLOCKED being written, which is longer if !version.UsesIETFFrameFormat() {
return 1 + 4 return 1 + 4
} }
return 1 return 1 + utils.VarIntLen(uint64(f.Offset))
} }

View file

@ -2,8 +2,10 @@ package wire
import ( import (
"bytes" "bytes"
"io"
"github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/protocol"
"github.com/lucas-clemente/quic-go/internal/utils"
. "github.com/onsi/ginkgo" . "github.com/onsi/ginkgo"
. "github.com/onsi/gomega" . "github.com/onsi/gomega"
@ -12,30 +14,41 @@ import (
var _ = Describe("BLOCKED frame", func() { var _ = Describe("BLOCKED frame", func() {
Context("when parsing", func() { Context("when parsing", func() {
It("accepts sample frame", func() { It("accepts sample frame", func() {
b := bytes.NewReader([]byte{0x08}) data := []byte{0x08}
_, err := ParseBlockedFrame(b, protocol.VersionWhatever) data = append(data, encodeVarInt(0x12345678)...)
b := bytes.NewReader(data)
frame, err := ParseBlockedFrame(b, versionIETFFrames)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
Expect(frame.Offset).To(Equal(protocol.ByteCount(0x12345678)))
Expect(b.Len()).To(BeZero()) Expect(b.Len()).To(BeZero())
}) })
It("errors on EOFs", func() { It("errors on EOFs", func() {
_, err := ParseBlockedFrame(bytes.NewReader(nil), protocol.VersionWhatever) data := []byte{0x08}
Expect(err).To(HaveOccurred()) data = append(data, encodeVarInt(0x12345678)...)
_, err := ParseBlockedFrame(bytes.NewReader(data), versionIETFFrames)
Expect(err).ToNot(HaveOccurred())
for i := range data {
_, err := ParseBlockedFrame(bytes.NewReader(data[:i]), versionIETFFrames)
Expect(err).To(MatchError(io.EOF))
}
}) })
}) })
Context("when writing", func() { Context("when writing", func() {
It("writes a sample frame", func() { It("writes a sample frame", func() {
b := &bytes.Buffer{} b := &bytes.Buffer{}
frame := BlockedFrame{} frame := BlockedFrame{Offset: 0xdeadbeef}
err := frame.Write(b, protocol.VersionWhatever) err := frame.Write(b, protocol.VersionWhatever)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
Expect(b.Bytes()).To(Equal([]byte{0x08})) expected := []byte{0x08}
expected = append(expected, encodeVarInt(0xdeadbeef)...)
Expect(b.Bytes()).To(Equal(expected))
}) })
It("has the correct min length", func() { It("has the correct min length", func() {
frame := BlockedFrame{} frame := BlockedFrame{Offset: 0x12345}
Expect(frame.MinLength(versionIETFFrames)).To(Equal(protocol.ByteCount(1))) Expect(frame.MinLength(versionIETFFrames)).To(Equal(1 + utils.VarIntLen(0x12345)))
}) })
}) })
}) })

View file

@ -10,10 +10,11 @@ import (
// A StreamBlockedFrame in QUIC // A StreamBlockedFrame in QUIC
type StreamBlockedFrame struct { type StreamBlockedFrame struct {
StreamID protocol.StreamID StreamID protocol.StreamID
Offset protocol.ByteCount
} }
// ParseStreamBlockedFrame parses a STREAM_BLOCKED frame // ParseStreamBlockedFrame parses a STREAM_BLOCKED frame
func ParseStreamBlockedFrame(r *bytes.Reader, version protocol.VersionNumber) (*StreamBlockedFrame, error) { func ParseStreamBlockedFrame(r *bytes.Reader, _ protocol.VersionNumber) (*StreamBlockedFrame, error) {
if _, err := r.ReadByte(); err != nil { // read the TypeByte if _, err := r.ReadByte(); err != nil { // read the TypeByte
return nil, err return nil, err
} }
@ -21,7 +22,14 @@ func ParseStreamBlockedFrame(r *bytes.Reader, version protocol.VersionNumber) (*
if err != nil { if err != nil {
return nil, err return nil, err
} }
return &StreamBlockedFrame{StreamID: protocol.StreamID(sid)}, nil offset, err := utils.ReadVarInt(r)
if err != nil {
return nil, err
}
return &StreamBlockedFrame{
StreamID: protocol.StreamID(sid),
Offset: protocol.ByteCount(offset),
}, nil
} }
// Write writes a STREAM_BLOCKED frame // Write writes a STREAM_BLOCKED frame
@ -31,6 +39,7 @@ func (f *StreamBlockedFrame) Write(b *bytes.Buffer, version protocol.VersionNumb
} }
b.WriteByte(0x09) b.WriteByte(0x09)
utils.WriteVarInt(b, uint64(f.StreamID)) utils.WriteVarInt(b, uint64(f.StreamID))
utils.WriteVarInt(b, uint64(f.Offset))
return nil return nil
} }
@ -39,5 +48,5 @@ func (f *StreamBlockedFrame) MinLength(version protocol.VersionNumber) protocol.
if !version.UsesIETFFrameFormat() { if !version.UsesIETFFrameFormat() {
return 1 + 4 return 1 + 4
} }
return 1 + utils.VarIntLen(uint64(f.StreamID)) return 1 + utils.VarIntLen(uint64(f.StreamID)) + utils.VarIntLen(uint64(f.Offset))
} }

View file

@ -14,17 +14,20 @@ var _ = Describe("STREAM_BLOCKED frame", func() {
Context("parsing", func() { Context("parsing", func() {
It("accepts sample frame", func() { It("accepts sample frame", func() {
data := []byte{0x9} data := []byte{0x9}
data = append(data, encodeVarInt(0xdeadbeef)...) data = append(data, encodeVarInt(0xdeadbeef)...) // stream ID
data = append(data, encodeVarInt(0xdecafbad)...) // offset
b := bytes.NewReader(data) b := bytes.NewReader(data)
frame, err := ParseStreamBlockedFrame(b, versionIETFFrames) frame, err := ParseStreamBlockedFrame(b, versionIETFFrames)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
Expect(frame.StreamID).To(Equal(protocol.StreamID(0xdeadbeef))) Expect(frame.StreamID).To(Equal(protocol.StreamID(0xdeadbeef)))
Expect(frame.Offset).To(Equal(protocol.ByteCount(0xdecafbad)))
Expect(b.Len()).To(BeZero()) Expect(b.Len()).To(BeZero())
}) })
It("errors on EOFs", func() { It("errors on EOFs", func() {
data := []byte{0x9} data := []byte{0x9}
data = append(data, encodeVarInt(0xdeadbeef)...) data = append(data, encodeVarInt(0xdeadbeef)...)
data = append(data, encodeVarInt(0xc0010ff)...)
_, err := ParseStreamBlockedFrame(bytes.NewReader(data), versionIETFFrames) _, err := ParseStreamBlockedFrame(bytes.NewReader(data), versionIETFFrames)
Expect(err).NotTo(HaveOccurred()) Expect(err).NotTo(HaveOccurred())
for i := range data { for i := range data {
@ -38,19 +41,22 @@ var _ = Describe("STREAM_BLOCKED frame", func() {
It("has proper min length", func() { It("has proper min length", func() {
f := &StreamBlockedFrame{ f := &StreamBlockedFrame{
StreamID: 0x1337, StreamID: 0x1337,
Offset: 0xdeadbeef,
} }
Expect(f.MinLength(0)).To(Equal(1 + utils.VarIntLen(0x1337))) Expect(f.MinLength(0)).To(Equal(1 + utils.VarIntLen(0x1337) + utils.VarIntLen(0xdeadbeef)))
}) })
It("writes a sample frame", func() { It("writes a sample frame", func() {
b := &bytes.Buffer{} b := &bytes.Buffer{}
f := &StreamBlockedFrame{ f := &StreamBlockedFrame{
StreamID: 0xdecafbad, StreamID: 0xdecafbad,
Offset: 0x1337,
} }
err := f.Write(b, versionIETFFrames) err := f.Write(b, versionIETFFrames)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
expected := []byte{0x9} expected := []byte{0x9}
expected = append(expected, encodeVarInt(uint64(f.StreamID))...) expected = append(expected, encodeVarInt(uint64(f.StreamID))...)
expected = append(expected, encodeVarInt(uint64(f.Offset))...)
Expect(b.Bytes()).To(Equal(expected)) Expect(b.Bytes()).To(Equal(expected))
}) })
}) })

View file

@ -117,6 +117,9 @@ func (u *packetUnpacker) parseIETFFrame(r *bytes.Reader, typeByte byte, hdr *wir
frame, err = wire.ParsePingFrame(r, u.version) frame, err = wire.ParsePingFrame(r, u.version)
case 0x8: case 0x8:
frame, err = wire.ParseBlockedFrame(r, u.version) frame, err = wire.ParseBlockedFrame(r, u.version)
if err != nil {
err = qerr.Error(qerr.InvalidBlockedData, err.Error())
}
case 0x9: case 0x9:
frame, err = wire.ParseStreamBlockedFrame(r, u.version) frame, err = wire.ParseStreamBlockedFrame(r, u.version)
if err != nil { if err != nil {

View file

@ -343,7 +343,7 @@ var _ = Describe("Packet unpacker", func() {
}) })
It("unpacks connection-level BLOCKED frames", func() { It("unpacks connection-level BLOCKED frames", func() {
f := &wire.BlockedFrame{} f := &wire.BlockedFrame{Offset: 0x1234}
buf := &bytes.Buffer{} buf := &bytes.Buffer{}
err := f.Write(buf, versionIETFFrames) err := f.Write(buf, versionIETFFrames)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
@ -354,7 +354,10 @@ var _ = Describe("Packet unpacker", func() {
}) })
It("unpacks stream-level BLOCKED frames", func() { It("unpacks stream-level BLOCKED frames", func() {
f := &wire.StreamBlockedFrame{StreamID: 0xdeadbeef} f := &wire.StreamBlockedFrame{
StreamID: 0xdeadbeef,
Offset: 0xdead,
}
buf := &bytes.Buffer{} buf := &bytes.Buffer{}
err := f.Write(buf, versionIETFFrames) err := f.Write(buf, versionIETFFrames)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
@ -403,6 +406,7 @@ var _ = Describe("Packet unpacker", func() {
0x02: qerr.InvalidConnectionCloseData, 0x02: qerr.InvalidConnectionCloseData,
0x04: qerr.InvalidWindowUpdateData, 0x04: qerr.InvalidWindowUpdateData,
0x05: qerr.InvalidWindowUpdateData, 0x05: qerr.InvalidWindowUpdateData,
0x08: qerr.InvalidBlockedData,
0x09: qerr.InvalidBlockedData, 0x09: qerr.InvalidBlockedData,
0x0c: qerr.InvalidFrameData, 0x0c: qerr.InvalidFrameData,
0x0e: qerr.InvalidAckData, 0x0e: qerr.InvalidAckData,

View file

@ -25,7 +25,7 @@ type streamI interface {
// methods needed for flow control // methods needed for flow control
GetWindowUpdate() protocol.ByteCount GetWindowUpdate() protocol.ByteCount
HandleMaxStreamDataFrame(*wire.MaxStreamDataFrame) HandleMaxStreamDataFrame(*wire.MaxStreamDataFrame)
IsFlowControlBlocked() bool IsFlowControlBlocked() (bool, protocol.ByteCount)
} }
// A Stream assembles the data from StreamFrames and provides a super-convenient Read-Interface // A Stream assembles the data from StreamFrames and provides a super-convenient Read-Interface
@ -485,7 +485,7 @@ func (s *stream) HandleMaxStreamDataFrame(frame *wire.MaxStreamDataFrame) {
s.flowController.UpdateSendWindow(frame.ByteOffset) s.flowController.UpdateSendWindow(frame.ByteOffset)
} }
func (s *stream) IsFlowControlBlocked() bool { func (s *stream) IsFlowControlBlocked() (bool, protocol.ByteCount) {
return s.flowController.IsBlocked() return s.flowController.IsBlocked()
} }

View file

@ -105,11 +105,16 @@ func (f *streamFramer) maybePopNormalFrames(maxTotalLen protocol.ByteCount) (res
} }
// Finally, check if we are now FC blocked and should queue a BLOCKED frame // Finally, check if we are now FC blocked and should queue a BLOCKED frame
if !frame.FinBit && s.IsFlowControlBlocked() { if !frame.FinBit {
f.blockedFrameQueue = append(f.blockedFrameQueue, &wire.StreamBlockedFrame{StreamID: s.StreamID()}) if blocked, offset := s.IsFlowControlBlocked(); blocked {
f.blockedFrameQueue = append(f.blockedFrameQueue, &wire.StreamBlockedFrame{
StreamID: s.StreamID(),
Offset: offset,
})
}
} }
if f.connFlowController.IsBlocked() { if blocked, offset := f.connFlowController.IsBlocked(); blocked {
f.blockedFrameQueue = append(f.blockedFrameQueue, &wire.BlockedFrame{}) f.blockedFrameQueue = append(f.blockedFrameQueue, &wire.BlockedFrame{Offset: offset})
} }
res = append(res, frame) res = append(res, frame)

View file

@ -72,8 +72,8 @@ var _ = Describe("Stream Framer", func() {
BeforeEach(func() { BeforeEach(func() {
// nothing is blocked here // nothing is blocked here
connFC.EXPECT().IsBlocked().AnyTimes() connFC.EXPECT().IsBlocked().AnyTimes()
stream1.EXPECT().IsFlowControlBlocked().Return(false).AnyTimes() stream1.EXPECT().IsFlowControlBlocked().Return(false, protocol.ByteCount(0)).AnyTimes()
stream2.EXPECT().IsFlowControlBlocked().Return(false).AnyTimes() stream2.EXPECT().IsFlowControlBlocked().Return(false, protocol.ByteCount(0)).AnyTimes()
}) })
It("returns nil when popping an empty framer", func() { It("returns nil when popping an empty framer", func() {
@ -272,13 +272,13 @@ var _ = Describe("Stream Framer", func() {
StreamID: id1, StreamID: id1,
Data: []byte("foobar"), Data: []byte("foobar"),
}) })
stream1.EXPECT().IsFlowControlBlocked().Return(true) stream1.EXPECT().IsFlowControlBlocked().Return(true, protocol.ByteCount(0x1234))
frames := framer.PopStreamFrames(1000) frames := framer.PopStreamFrames(1000)
Expect(frames).To(HaveLen(1)) Expect(frames).To(HaveLen(1))
f := framer.PopBlockedFrame() Expect(framer.PopBlockedFrame()).To(Equal(&wire.StreamBlockedFrame{
Expect(f).To(BeAssignableToTypeOf(&wire.StreamBlockedFrame{})) StreamID: stream1.StreamID(),
bf := f.(*wire.StreamBlockedFrame) Offset: 0x1234,
Expect(bf.StreamID).To(Equal(stream1.StreamID())) }))
Expect(framer.PopBlockedFrame()).To(BeNil()) Expect(framer.PopBlockedFrame()).To(BeNil())
}) })
@ -300,15 +300,14 @@ var _ = Describe("Stream Framer", func() {
It("queues and pops BLOCKED frames for connection blocked streams", func() { It("queues and pops BLOCKED frames for connection blocked streams", func() {
setNoData(stream2) setNoData(stream2)
connFC.EXPECT().IsBlocked().Return(true) connFC.EXPECT().IsBlocked().Return(true, protocol.ByteCount(0x4321))
stream1.EXPECT().PopStreamFrame(gomock.Any()).Return(&wire.StreamFrame{ stream1.EXPECT().PopStreamFrame(gomock.Any()).Return(&wire.StreamFrame{
StreamID: id1, StreamID: id1,
Data: []byte("foo"), Data: []byte("foo"),
}) })
stream1.EXPECT().IsFlowControlBlocked().Return(false) stream1.EXPECT().IsFlowControlBlocked().Return(false, protocol.ByteCount(0))
framer.PopStreamFrames(1000) framer.PopStreamFrames(1000)
f := framer.PopBlockedFrame() Expect(framer.PopBlockedFrame()).To(Equal(&wire.BlockedFrame{Offset: 0x4321}))
Expect(f).To(BeAssignableToTypeOf(&wire.BlockedFrame{}))
Expect(framer.PopBlockedFrame()).To(BeNil()) Expect(framer.PopBlockedFrame()).To(BeNil())
}) })
}) })

View file

@ -1146,10 +1146,13 @@ var _ = Describe("Stream", func() {
Context("flow control", func() { Context("flow control", func() {
It("says when it's flow control blocked", func() { It("says when it's flow control blocked", func() {
mockFC.EXPECT().IsBlocked().Return(false) mockFC.EXPECT().IsBlocked().Return(false, protocol.ByteCount(0))
Expect(str.IsFlowControlBlocked()).To(BeFalse()) blocked, _ := str.IsFlowControlBlocked()
mockFC.EXPECT().IsBlocked().Return(true) Expect(blocked).To(BeFalse())
Expect(str.IsFlowControlBlocked()).To(BeTrue()) mockFC.EXPECT().IsBlocked().Return(true, protocol.ByteCount(0x1337))
blocked, offset := str.IsFlowControlBlocked()
Expect(blocked).To(BeTrue())
Expect(offset).To(Equal(protocol.ByteCount(0x1337)))
}) })
It("updates the flow control window", func() { It("updates the flow control window", func() {