diff options
author | Yury Selivanov <yury@magic.io> | 2018-01-21 19:56:59 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2018-01-21 19:56:59 (GMT) |
commit | a4afcdfa55ddffa4b9ae3b0cf101628c7bff4102 (patch) | |
tree | 61649205a7b3d95fed03f643b66c568aab79be3a /Lib | |
parent | fc2f407829d9817ddacccae6944dd0879cfaca24 (diff) | |
download | cpython-a4afcdfa55ddffa4b9ae3b0cf101628c7bff4102.zip cpython-a4afcdfa55ddffa4b9ae3b0cf101628c7bff4102.tar.gz cpython-a4afcdfa55ddffa4b9ae3b0cf101628c7bff4102.tar.bz2 |
bpo-32314: Fix asyncio.run() to cancel runinng tasks on shutdown (#5262)
Diffstat (limited to 'Lib')
-rw-r--r-- | Lib/asyncio/base_events.py | 25 | ||||
-rw-r--r-- | Lib/asyncio/runners.py | 25 | ||||
-rw-r--r-- | Lib/test/test_asyncio/test_runners.py | 79 | ||||
-rw-r--r-- | Lib/test/test_asyncio/utils.py | 8 |
4 files changed, 122 insertions, 15 deletions
diff --git a/Lib/asyncio/base_events.py b/Lib/asyncio/base_events.py index a10e706..ca9eee7 100644 --- a/Lib/asyncio/base_events.py +++ b/Lib/asyncio/base_events.py @@ -228,14 +228,9 @@ class BaseEventLoop(events.AbstractEventLoop): self._coroutine_origin_tracking_enabled = False self._coroutine_origin_tracking_saved_depth = None - if hasattr(sys, 'get_asyncgen_hooks'): - # Python >= 3.6 - # A weak set of all asynchronous generators that are - # being iterated by the loop. - self._asyncgens = weakref.WeakSet() - else: - self._asyncgens = None - + # A weak set of all asynchronous generators that are + # being iterated by the loop. + self._asyncgens = weakref.WeakSet() # Set to True when `loop.shutdown_asyncgens` is called. self._asyncgens_shutdown_called = False @@ -354,7 +349,7 @@ class BaseEventLoop(events.AbstractEventLoop): """Shutdown all active asynchronous generators.""" self._asyncgens_shutdown_called = True - if self._asyncgens is None or not len(self._asyncgens): + if not len(self._asyncgens): # If Python version is <3.6 or we don't have any asynchronous # generators alive. return @@ -386,10 +381,10 @@ class BaseEventLoop(events.AbstractEventLoop): 'Cannot run the event loop while another loop is running') self._set_coroutine_origin_tracking(self._debug) self._thread_id = threading.get_ident() - if self._asyncgens is not None: - old_agen_hooks = sys.get_asyncgen_hooks() - sys.set_asyncgen_hooks(firstiter=self._asyncgen_firstiter_hook, - finalizer=self._asyncgen_finalizer_hook) + + old_agen_hooks = sys.get_asyncgen_hooks() + sys.set_asyncgen_hooks(firstiter=self._asyncgen_firstiter_hook, + finalizer=self._asyncgen_finalizer_hook) try: events._set_running_loop(self) while True: @@ -401,8 +396,7 @@ class BaseEventLoop(events.AbstractEventLoop): self._thread_id = None events._set_running_loop(None) self._set_coroutine_origin_tracking(False) - if self._asyncgens is not None: - sys.set_asyncgen_hooks(*old_agen_hooks) + sys.set_asyncgen_hooks(*old_agen_hooks) def run_until_complete(self, future): """Run until the Future is done. @@ -1374,6 +1368,7 @@ class BaseEventLoop(events.AbstractEventLoop): - 'message': Error message; - 'exception' (optional): Exception object; - 'future' (optional): Future instance; + - 'task' (optional): Task instance; - 'handle' (optional): Handle instance; - 'protocol' (optional): Protocol instance; - 'transport' (optional): Transport instance; diff --git a/Lib/asyncio/runners.py b/Lib/asyncio/runners.py index 94d9409..bb54b72 100644 --- a/Lib/asyncio/runners.py +++ b/Lib/asyncio/runners.py @@ -2,6 +2,7 @@ __all__ = 'run', from . import coroutines from . import events +from . import tasks def run(main, *, debug=False): @@ -42,7 +43,31 @@ def run(main, *, debug=False): return loop.run_until_complete(main) finally: try: + _cancel_all_tasks(loop) loop.run_until_complete(loop.shutdown_asyncgens()) finally: events.set_event_loop(None) loop.close() + + +def _cancel_all_tasks(loop): + to_cancel = [task for task in tasks.all_tasks(loop) + if not task.done()] + if not to_cancel: + return + + for task in to_cancel: + task.cancel() + + loop.run_until_complete( + tasks.gather(*to_cancel, loop=loop, return_exceptions=True)) + + for task in to_cancel: + if task.cancelled(): + continue + if task.exception() is not None: + loop.call_exception_handler({ + 'message': 'unhandled exception during asyncio.run() shutdown', + 'exception': task.exception(), + 'task': task, + }) diff --git a/Lib/test/test_asyncio/test_runners.py b/Lib/test/test_asyncio/test_runners.py index c52bd94..3b58dde 100644 --- a/Lib/test/test_asyncio/test_runners.py +++ b/Lib/test/test_asyncio/test_runners.py @@ -2,6 +2,7 @@ import asyncio import unittest from unittest import mock +from . import utils as test_utils class TestPolicy(asyncio.AbstractEventLoopPolicy): @@ -98,3 +99,81 @@ class RunTests(BaseTest): with self.assertRaisesRegex(RuntimeError, 'cannot be called from a running'): asyncio.run(main()) + + def test_asyncio_run_cancels_hanging_tasks(self): + lo_task = None + + async def leftover(): + await asyncio.sleep(0.1) + + async def main(): + nonlocal lo_task + lo_task = asyncio.create_task(leftover()) + return 123 + + self.assertEqual(asyncio.run(main()), 123) + self.assertTrue(lo_task.done()) + + def test_asyncio_run_reports_hanging_tasks_errors(self): + lo_task = None + call_exc_handler_mock = mock.Mock() + + async def leftover(): + try: + await asyncio.sleep(0.1) + except asyncio.CancelledError: + 1 / 0 + + async def main(): + loop = asyncio.get_running_loop() + loop.call_exception_handler = call_exc_handler_mock + + nonlocal lo_task + lo_task = asyncio.create_task(leftover()) + return 123 + + self.assertEqual(asyncio.run(main()), 123) + self.assertTrue(lo_task.done()) + + call_exc_handler_mock.assert_called_with({ + 'message': test_utils.MockPattern(r'asyncio.run.*shutdown'), + 'task': lo_task, + 'exception': test_utils.MockInstanceOf(ZeroDivisionError) + }) + + def test_asyncio_run_closes_gens_after_hanging_tasks_errors(self): + spinner = None + lazyboy = None + + class FancyExit(Exception): + pass + + async def fidget(): + while True: + yield 1 + await asyncio.sleep(1) + + async def spin(): + nonlocal spinner + spinner = fidget() + try: + async for the_meaning_of_life in spinner: # NoQA + pass + except asyncio.CancelledError: + 1 / 0 + + async def main(): + loop = asyncio.get_running_loop() + loop.call_exception_handler = mock.Mock() + + nonlocal lazyboy + lazyboy = asyncio.create_task(spin()) + raise FancyExit + + with self.assertRaises(FancyExit): + asyncio.run(main()) + + self.assertTrue(lazyboy.done()) + + self.assertIsNone(spinner.ag_frame) + self.assertFalse(spinner.ag_running) diff --git a/Lib/test/test_asyncio/utils.py b/Lib/test/test_asyncio/utils.py index 6c80977..f756ec9 100644 --- a/Lib/test/test_asyncio/utils.py +++ b/Lib/test/test_asyncio/utils.py @@ -485,6 +485,14 @@ class MockPattern(str): return bool(re.search(str(self), other, re.S)) +class MockInstanceOf: + def __init__(self, type): + self._type = type + + def __eq__(self, other): + return isinstance(other, self._type) + + def get_function_source(func): source = format_helpers._get_function_source(func) if source is None: |