Skip to content
This repository was archived by the owner on Jan 13, 2021. It is now read-only.

Commit 3e52e91

Browse files
committed
Fix: no end of the stream when (length of body) % MAX_CHUNK == MAX_CHUNK
1 parent 86642e6 commit 3e52e91

File tree

3 files changed

+73
-11
lines changed

3 files changed

+73
-11
lines changed

HISTORY.rst

+4
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,10 @@ Release History
44
dev
55
---
66

7+
*Bugfixes*
8+
9+
- Stream end flag when length of last chunk equal to MAX_CHUNK
10+
711
v0.7.0 (2016-09-27)
812
-------------------
913

hyper/http20/stream.py

+14-11
Original file line numberDiff line numberDiff line change
@@ -122,8 +122,18 @@ def file_iterator(fobj):
122122
chunks = (data[i:i+MAX_CHUNK]
123123
for i in range(0, len(data), MAX_CHUNK))
124124

125-
for chunk in chunks:
126-
self._send_chunk(chunk, final)
125+
# since we need to know when we have a last package we need to know
126+
# if there is another package in advance
127+
cur_chunk = None
128+
try:
129+
cur_chunk = next(chunks)
130+
while True:
131+
next_chunk = next(chunks)
132+
self._send_chunk(cur_chunk, False)
133+
cur_chunk = next_chunk
134+
except StopIteration:
135+
if cur_chunk is not None: # cur_chunk none when no chunks to send
136+
self._send_chunk(cur_chunk, final)
127137

128138
def _read(self, amt=None):
129139
"""
@@ -323,19 +333,12 @@ def _send_chunk(self, data, final):
323333
while len(data) > self._out_flow_control_window:
324334
self._recv_cb()
325335

326-
# If the length of the data is less than MAX_CHUNK, we're probably
327-
# at the end of the file. If this is the end of the data, mark it
328-
# as END_STREAM.
329-
end_stream = False
330-
if len(data) < MAX_CHUNK and final:
331-
end_stream = True
332-
333336
# Send the frame and decrement the flow control window.
334337
with self._conn as conn:
335338
conn.send_data(
336-
stream_id=self.stream_id, data=data, end_stream=end_stream
339+
stream_id=self.stream_id, data=data, end_stream=final
337340
)
338341
self._send_outstanding_data()
339342

340-
if end_stream:
343+
if final:
341344
self.local_closed = True

test/test_hyper.py

+55
Original file line numberDiff line numberDiff line change
@@ -211,6 +211,61 @@ def data_callback(chunk, **kwargs):
211211
assert frames[1].data == b'hello there'
212212
assert frames[1].flags == set(['END_STREAM'])
213213

214+
def test_request_correctly_sent_max_chunk(self, frame_buffer):
215+
"""
216+
Test that request correctly sent when data length multiple
217+
max chunk. We check last chunk has a end flag and correct number
218+
of chunks.
219+
"""
220+
def data_callback(chunk, **kwargs):
221+
frame_buffer.add_data(chunk)
222+
223+
# one chunk
224+
c = HTTP20Connection('www.google.com')
225+
c._sock = DummySocket()
226+
c._send_cb = data_callback
227+
c.putrequest('GET', '/')
228+
c.endheaders(message_body=b'1'*1024, final=True)
229+
230+
frames = list(frame_buffer)
231+
assert len(frames) == 2
232+
assert isinstance(frames[1], DataFrame)
233+
assert frames[1].flags == set(['END_STREAM'])
234+
235+
# two chunks
236+
c = HTTP20Connection('www.google.com')
237+
c._sock = DummySocket()
238+
c._send_cb = data_callback
239+
c.putrequest('GET', '/')
240+
c.endheaders(message_body=b'1' * 2024, final=True)
241+
242+
frames = list(frame_buffer)
243+
assert len(frames) == 3
244+
assert isinstance(frames[1], DataFrame)
245+
assert frames[2].flags == set(['END_STREAM'])
246+
247+
# two chunks with last chunk < 1024
248+
c = HTTP20Connection('www.google.com')
249+
c._sock = DummySocket()
250+
c._send_cb = data_callback
251+
c.putrequest('GET', '/')
252+
c.endheaders(message_body=b'1' * 2000, final=True)
253+
254+
frames = list(frame_buffer)
255+
assert len(frames) == 3
256+
assert isinstance(frames[1], DataFrame)
257+
assert frames[2].flags == set(['END_STREAM'])
258+
259+
# no chunks
260+
c = HTTP20Connection('www.google.com')
261+
c._sock = DummySocket()
262+
c._send_cb = data_callback
263+
c.putrequest('GET', '/')
264+
c.endheaders(message_body=b'', final=True)
265+
266+
frames = list(frame_buffer)
267+
assert len(frames) == 1
268+
214269
def test_that_we_correctly_send_over_the_socket(self):
215270
sock = DummySocket()
216271
c = HTTP20Connection('www.google.com')

0 commit comments

Comments
 (0)