diff options
author | Yury Selivanov <yury@magic.io> | 2016-10-21 21:23:35 (GMT) |
---|---|---|
committer | Yury Selivanov <yury@magic.io> | 2016-10-21 21:23:35 (GMT) |
commit | e145efcd7a6e72f3190ea0934a1101cd45ee3c08 (patch) | |
tree | 1cd7c5cf249d4ffb02cbbf8e210b323cd7fdfc9b | |
parent | ed0540698ef2ea66dfd662ac6e98a15e8eabf365 (diff) | |
parent | 3d67615a485f4769eec5927e17989b31d6917e1c (diff) | |
download | cpython-e145efcd7a6e72f3190ea0934a1101cd45ee3c08.zip cpython-e145efcd7a6e72f3190ea0934a1101cd45ee3c08.tar.gz cpython-e145efcd7a6e72f3190ea0934a1101cd45ee3c08.tar.bz2 |
Merge 3.5 (issue #26923)
-rw-r--r-- | Lib/asyncio/tasks.py | 6 | ||||
-rw-r--r-- | Lib/test/test_asyncio/test_tasks.py | 30 | ||||
-rw-r--r-- | Misc/NEWS | 4 |
3 files changed, 38 insertions, 2 deletions
diff --git a/Lib/asyncio/tasks.py b/Lib/asyncio/tasks.py index 14949d1..8852aa5 100644 --- a/Lib/asyncio/tasks.py +++ b/Lib/asyncio/tasks.py @@ -592,9 +592,11 @@ class _GatheringFuture(futures.Future): def cancel(self): if self.done(): return False + ret = False for child in self._children: - child.cancel() - return True + if child.cancel(): + ret = True + return ret def gather(*coros_or_futures, loop=None, return_exceptions=False): diff --git a/Lib/test/test_asyncio/test_tasks.py b/Lib/test/test_asyncio/test_tasks.py index a5af7d1..1ceb9b2 100644 --- a/Lib/test/test_asyncio/test_tasks.py +++ b/Lib/test/test_asyncio/test_tasks.py @@ -1899,6 +1899,36 @@ class TaskTests(test_utils.TestCase): def test_cancel_wait_for(self): self._test_cancel_wait_for(60.0) + def test_cancel_gather(self): + """Ensure that a gathering future refuses to be cancelled once all + children are done""" + loop = asyncio.new_event_loop() + self.addCleanup(loop.close) + + fut = asyncio.Future(loop=loop) + # The indirection fut->child_coro is needed since otherwise the + # gathering task is done at the same time as the child future + def child_coro(): + return (yield from fut) + gather_future = asyncio.gather(child_coro(), loop=loop) + gather_task = asyncio.ensure_future(gather_future, loop=loop) + + cancel_result = None + def cancelling_callback(_): + nonlocal cancel_result + cancel_result = gather_task.cancel() + fut.add_done_callback(cancelling_callback) + + fut.set_result(42) # calls the cancelling_callback after fut is done() + + # At this point the task should complete. + loop.run_until_complete(gather_task) + + # Python issue #26923: asyncio.gather drops cancellation + self.assertEqual(cancel_result, False) + self.assertFalse(gather_task.cancelled()) + self.assertEqual(gather_task.result(), [42]) + class GatherTestsBase: @@ -40,6 +40,10 @@ Library - Issue #28500: Fix asyncio to handle async gens GC from another thread. +- Issue #26923: Fix asyncio.Gather to refuse being cancelled once all + children are done. + Patch by Johannes Ebke. + Build ----- |