Commit 0bd16bc4 authored by Guido van Rossum's avatar Guido van Rossum

Fix asyncio issue 235: Queue subclass bug caused by JoinableQueue merge.

parent 77e8311d
...@@ -54,6 +54,8 @@ class Queue: ...@@ -54,6 +54,8 @@ class Queue:
self._finished.set() self._finished.set()
self._init(maxsize) self._init(maxsize)
# These three are overridable in subclasses.
def _init(self, maxsize): def _init(self, maxsize):
self._queue = collections.deque() self._queue = collections.deque()
...@@ -62,6 +64,11 @@ class Queue: ...@@ -62,6 +64,11 @@ class Queue:
def _put(self, item): def _put(self, item):
self._queue.append(item) self._queue.append(item)
# End of the overridable methods.
def __put_internal(self, item):
self._put(item)
self._unfinished_tasks += 1 self._unfinished_tasks += 1
self._finished.clear() self._finished.clear()
...@@ -133,7 +140,7 @@ class Queue: ...@@ -133,7 +140,7 @@ class Queue:
'queue non-empty, why are getters waiting?') 'queue non-empty, why are getters waiting?')
getter = self._getters.popleft() getter = self._getters.popleft()
self._put(item) self.__put_internal(item)
# getter cannot be cancelled, we just removed done getters # getter cannot be cancelled, we just removed done getters
getter.set_result(self._get()) getter.set_result(self._get())
...@@ -145,7 +152,7 @@ class Queue: ...@@ -145,7 +152,7 @@ class Queue:
yield from waiter yield from waiter
else: else:
self._put(item) self.__put_internal(item)
def put_nowait(self, item): def put_nowait(self, item):
"""Put an item into the queue without blocking. """Put an item into the queue without blocking.
...@@ -158,7 +165,7 @@ class Queue: ...@@ -158,7 +165,7 @@ class Queue:
'queue non-empty, why are getters waiting?') 'queue non-empty, why are getters waiting?')
getter = self._getters.popleft() getter = self._getters.popleft()
self._put(item) self.__put_internal(item)
# getter cannot be cancelled, we just removed done getters # getter cannot be cancelled, we just removed done getters
getter.set_result(self._get()) getter.set_result(self._get())
...@@ -166,7 +173,7 @@ class Queue: ...@@ -166,7 +173,7 @@ class Queue:
elif self._maxsize > 0 and self._maxsize <= self.qsize(): elif self._maxsize > 0 and self._maxsize <= self.qsize():
raise QueueFull raise QueueFull
else: else:
self._put(item) self.__put_internal(item)
@coroutine @coroutine
def get(self): def get(self):
...@@ -180,7 +187,7 @@ class Queue: ...@@ -180,7 +187,7 @@ class Queue:
if self._putters: if self._putters:
assert self.full(), 'queue not full, why are putters waiting?' assert self.full(), 'queue not full, why are putters waiting?'
item, putter = self._putters.popleft() item, putter = self._putters.popleft()
self._put(item) self.__put_internal(item)
# When a getter runs and frees up a slot so this putter can # When a getter runs and frees up a slot so this putter can
# run, we need to defer the put for a tick to ensure that # run, we need to defer the put for a tick to ensure that
...@@ -207,7 +214,7 @@ class Queue: ...@@ -207,7 +214,7 @@ class Queue:
if self._putters: if self._putters:
assert self.full(), 'queue not full, why are putters waiting?' assert self.full(), 'queue not full, why are putters waiting?'
item, putter = self._putters.popleft() item, putter = self._putters.popleft()
self._put(item) self.__put_internal(item)
# Wake putter on next tick. # Wake putter on next tick.
# getter cannot be cancelled, we just removed done putters # getter cannot be cancelled, we just removed done putters
......
...@@ -408,14 +408,16 @@ class PriorityQueueTests(_QueueTestBase): ...@@ -408,14 +408,16 @@ class PriorityQueueTests(_QueueTestBase):
self.assertEqual([1, 2, 3], items) self.assertEqual([1, 2, 3], items)
class QueueJoinTests(_QueueTestBase): class _QueueJoinTestMixin:
q_class = None
def test_task_done_underflow(self): def test_task_done_underflow(self):
q = asyncio.Queue(loop=self.loop) q = self.q_class(loop=self.loop)
self.assertRaises(ValueError, q.task_done) self.assertRaises(ValueError, q.task_done)
def test_task_done(self): def test_task_done(self):
q = asyncio.Queue(loop=self.loop) q = self.q_class(loop=self.loop)
for i in range(100): for i in range(100):
q.put_nowait(i) q.put_nowait(i)
...@@ -452,7 +454,7 @@ class QueueJoinTests(_QueueTestBase): ...@@ -452,7 +454,7 @@ class QueueJoinTests(_QueueTestBase):
self.loop.run_until_complete(asyncio.wait(tasks, loop=self.loop)) self.loop.run_until_complete(asyncio.wait(tasks, loop=self.loop))
def test_join_empty_queue(self): def test_join_empty_queue(self):
q = asyncio.Queue(loop=self.loop) q = self.q_class(loop=self.loop)
# Test that a queue join()s successfully, and before anything else # Test that a queue join()s successfully, and before anything else
# (done twice for insurance). # (done twice for insurance).
...@@ -465,12 +467,24 @@ class QueueJoinTests(_QueueTestBase): ...@@ -465,12 +467,24 @@ class QueueJoinTests(_QueueTestBase):
self.loop.run_until_complete(join()) self.loop.run_until_complete(join())
def test_format(self): def test_format(self):
q = asyncio.Queue(loop=self.loop) q = self.q_class(loop=self.loop)
self.assertEqual(q._format(), 'maxsize=0') self.assertEqual(q._format(), 'maxsize=0')
q._unfinished_tasks = 2 q._unfinished_tasks = 2
self.assertEqual(q._format(), 'maxsize=0 tasks=2') self.assertEqual(q._format(), 'maxsize=0 tasks=2')
class QueueJoinTests(_QueueJoinTestMixin, _QueueTestBase):
q_class = asyncio.Queue
class LifoQueueJoinTests(_QueueJoinTestMixin, _QueueTestBase):
q_class = asyncio.LifoQueue
class PriorityQueueJoinTests(_QueueJoinTestMixin, _QueueTestBase):
q_class = asyncio.PriorityQueue
if __name__ == '__main__': if __name__ == '__main__':
unittest.main() unittest.main()
...@@ -29,6 +29,10 @@ Core and Builtins ...@@ -29,6 +29,10 @@ Core and Builtins
Library Library
------- -------
- Fix asyncio issue 235: LifoQueue and PriorityQueue's put didn't
increment unfinished tasks (this bug was introduced in 3.4.3 when
JoinableQueue was merged with Queue).
- Issue #23908: os functions now reject paths with embedded null character - Issue #23908: os functions now reject paths with embedded null character
on Windows instead of silently truncate them. on Windows instead of silently truncate them.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment