From: Yury Selivanov Date: Wed, 5 Oct 2016 22:01:12 +0000 (-0400) Subject: Issue #28370: Speedup asyncio.StreamReader.readexactly X-Git-Tag: v3.6.0b2~46^2 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=3e56ff0d08a65ea1e0fe361ae22d855acc9d65bb;p=python Issue #28370: Speedup asyncio.StreamReader.readexactly Patch by Коренберг Марк. --- diff --git a/Lib/asyncio/streams.py b/Lib/asyncio/streams.py index b4adc7d9c6..a82cc79aca 100644 --- a/Lib/asyncio/streams.py +++ b/Lib/asyncio/streams.py @@ -448,6 +448,7 @@ class StreamReader: assert not self._eof, '_wait_for_data after EOF' # Waiting for data while paused will make deadlock, so prevent it. + # This is essential for readexactly(n) for case when n > self._limit. if self._paused: self._paused = False self._transport.resume_reading() @@ -658,25 +659,22 @@ class StreamReader: if n == 0: return b'' - # There used to be "optimized" code here. It created its own - # Future and waited until self._buffer had at least the n - # bytes, then called read(n). Unfortunately, this could pause - # the transport if the argument was larger than the pause - # limit (which is twice self._limit). So now we just read() - # into a local buffer. - - blocks = [] - while n > 0: - block = yield from self.read(n) - if not block: - partial = b''.join(blocks) - raise IncompleteReadError(partial, len(partial) + n) - blocks.append(block) - n -= len(block) - - assert n == 0 - - return b''.join(blocks) + while len(self._buffer) < n: + if self._eof: + incomplete = bytes(self._buffer) + self._buffer.clear() + raise IncompleteReadError(incomplete, n) + + yield from self._wait_for_data('readexactly') + + if len(self._buffer) == n: + data = bytes(self._buffer) + self._buffer.clear() + else: + data = bytes(self._buffer[:n]) + del self._buffer[:n] + self._maybe_resume_transport() + return data if compat.PY35: @coroutine diff --git a/Misc/NEWS b/Misc/NEWS index 8ea2ce2346..354dc59ca2 100644 --- a/Misc/NEWS +++ b/Misc/NEWS @@ -353,6 +353,9 @@ Library - Issue #28369: Raise RuntimeError when transport's FD is used with add_reader, add_writer, etc. +- Issue #28370: Speedup asyncio.StreamReader.readexactly. + Patch by Коренберг Марк. + IDLE ----