aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/Lib/test/test_asyncio/test_tasks.py
diff options
context:
space:
mode:
authorElvis Pranskevichus <elvis@magic.io>2018-10-03 10:30:31 -0400
committerYury Selivanov <yury@magic.io>2018-10-03 10:30:31 -0400
commit0c797a6aca1c293e530e18c5e9fa02c670a9a4ed (patch)
tree0a7627593801b009925f9909715b8cacabf6c91e /Lib/test/test_asyncio/test_tasks.py
parent96c593279400693226d5a560c420ae0fcf1731b9 (diff)
downloadcpython-0c797a6aca1c293e530e18c5e9fa02c670a9a4ed.tar.gz
cpython-0c797a6aca1c293e530e18c5e9fa02c670a9a4ed.zip
bpo-34872: Fix self-cancellation in C implementation of asyncio.Task (GH-9679)
The C implementation of asyncio.Task currently fails to perform the cancellation cleanup correctly in the following scenario. async def task1(): async def task2(): await task3 # task3 is never cancelled asyncio.current_task().cancel() await asyncio.create_task(task2()) The actuall error is a hardcoded call to `future_cancel()` instead of calling the `cancel()` method of a future-like object. Thanks to Vladimir Matveev for noticing the code discrepancy and to Yury Selivanov for coming up with a pathological scenario.
Diffstat (limited to 'Lib/test/test_asyncio/test_tasks.py')
-rw-r--r--Lib/test/test_asyncio/test_tasks.py36
1 files changed, 36 insertions, 0 deletions
diff --git a/Lib/test/test_asyncio/test_tasks.py b/Lib/test/test_asyncio/test_tasks.py
index eb6f2f5ecb1..0fe767630f1 100644
--- a/Lib/test/test_asyncio/test_tasks.py
+++ b/Lib/test/test_asyncio/test_tasks.py
@@ -622,6 +622,42 @@ class BaseTaskTests:
self.assertFalse(t._must_cancel) # White-box test.
self.assertFalse(t.cancel())
+ def test_cancel_awaited_task(self):
+ # This tests for a relatively rare condition when
+ # a task cancellation is requested for a task which is not
+ # currently blocked, such as a task cancelling itself.
+ # In this situation we must ensure that whatever next future
+ # or task the cancelled task blocks on is cancelled correctly
+ # as well. See also bpo-34872.
+ loop = asyncio.new_event_loop()
+ self.addCleanup(lambda: loop.close())
+
+ task = nested_task = None
+ fut = self.new_future(loop)
+
+ async def nested():
+ await fut
+
+ async def coro():
+ nonlocal nested_task
+ # Create a sub-task and wait for it to run.
+ nested_task = self.new_task(loop, nested())
+ await asyncio.sleep(0)
+
+ # Request the current task to be cancelled.
+ task.cancel()
+ # Block on the nested task, which should be immediately
+ # cancelled.
+ await nested_task
+
+ task = self.new_task(loop, coro())
+ with self.assertRaises(asyncio.CancelledError):
+ loop.run_until_complete(task)
+
+ self.assertTrue(task.cancelled())
+ self.assertTrue(nested_task.cancelled())
+ self.assertTrue(fut.cancelled())
+
def test_stop_while_run_in_complete(self):
def gen():