diff options
author | Victor Stinner <victor.stinner@gmail.com> | 2015-01-15 15:29:10 (GMT) |
---|---|---|
committer | Victor Stinner <victor.stinner@gmail.com> | 2015-01-15 15:29:10 (GMT) |
commit | 922bc2ca123630aa9cff63c605a82af05408318d (patch) | |
tree | 98bdd9abdbeabc074bd223e598f52eb9f0c4844f | |
parent | ab8848bc2a64930e0e9a2e56592bb692fb31d9e9 (diff) | |
download | cpython-922bc2ca123630aa9cff63c605a82af05408318d.zip cpython-922bc2ca123630aa9cff63c605a82af05408318d.tar.gz cpython-922bc2ca123630aa9cff63c605a82af05408318d.tar.bz2 |
Closes #23219: cancelling asyncio.wait_for() now cancels the task
-rw-r--r-- | Lib/asyncio/tasks.py | 12 | ||||
-rw-r--r-- | Lib/test/test_asyncio/test_tasks.py | 27 |
2 files changed, 35 insertions, 4 deletions
diff --git a/Lib/asyncio/tasks.py b/Lib/asyncio/tasks.py index 7959a55..63412a9 100644 --- a/Lib/asyncio/tasks.py +++ b/Lib/asyncio/tasks.py @@ -347,10 +347,9 @@ def wait_for(fut, timeout, *, loop=None): it cancels the task and raises TimeoutError. To avoid the task cancellation, wrap it in shield(). - Usage: - - result = yield from asyncio.wait_for(fut, 10.0) + If the wait is cancelled, the task is also cancelled. + This function is a coroutine. """ if loop is None: loop = events.get_event_loop() @@ -367,7 +366,12 @@ def wait_for(fut, timeout, *, loop=None): try: # wait until the future completes or the timeout - yield from waiter + try: + yield from waiter + except futures.CancelledError: + fut.remove_done_callback(cb) + fut.cancel() + raise if fut.done(): return fut.result() diff --git a/Lib/test/test_asyncio/test_tasks.py b/Lib/test/test_asyncio/test_tasks.py index 7807dc0..06447d7 100644 --- a/Lib/test/test_asyncio/test_tasks.py +++ b/Lib/test/test_asyncio/test_tasks.py @@ -1705,6 +1705,33 @@ class TaskTests(test_utils.TestCase): 'test_task_source_traceback')) self.loop.run_until_complete(task) + def _test_cancel_wait_for(self, timeout): + loop = asyncio.new_event_loop() + self.addCleanup(loop.close) + + @asyncio.coroutine + def blocking_coroutine(): + fut = asyncio.Future(loop=loop) + # Block: fut result is never set + yield from fut + + task = loop.create_task(blocking_coroutine()) + + wait = loop.create_task(asyncio.wait_for(task, timeout, loop=loop)) + loop.call_soon(wait.cancel) + + self.assertRaises(asyncio.CancelledError, + loop.run_until_complete, wait) + + # Python issue #23219: cancelling the wait must also cancel the task + self.assertTrue(task.cancelled()) + + def test_cancel_blocking_wait_for(self): + self._test_cancel_wait_for(None) + + def test_cancel_wait_for(self): + self._test_cancel_wait_for(60.0) + class GatherTestsBase: |