| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | """Tests for tasks.py.""" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | import collections | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  | import contextlib | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  | import contextvars | 
					
						
							| 
									
										
										
										
											2017-08-01 23:31:07 -07:00
										 |  |  | import gc | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  | import io | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2014-06-27 12:28:41 +02:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2014-06-16 17:11:05 +02:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  | import traceback | 
					
						
							| 
									
										
										
										
											2023-03-16 09:20:43 +05:30
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | import unittest | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  | from unittest import mock | 
					
						
							| 
									
										
										
										
											2022-01-22 16:58:53 +05:30
										 |  |  | from types import GenericAlias | 
					
						
							| 
									
										
										
										
											2014-12-26 21:16:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import asyncio | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | from asyncio import futures | 
					
						
							|  |  |  | from asyncio import tasks | 
					
						
							| 
									
										
										
										
											2017-12-11 10:04:40 -05:00
										 |  |  | from test.test_asyncio import utils as test_utils | 
					
						
							|  |  |  | from test import support | 
					
						
							|  |  |  | from test.support.script_helper import assert_python_ok | 
					
						
							| 
									
										
										
										
											2024-08-12 00:35:51 +08:00
										 |  |  | from test.support.warnings_helper import ignore_warnings | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-01 20:34:09 -07:00
										 |  |  | def tearDownModule(): | 
					
						
							|  |  |  |     asyncio.set_event_loop_policy(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  | async def coroutine_function(): | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-11 00:21:27 +02:00
										 |  |  | def format_coroutine(qualname, state, src, source_traceback, generator=False): | 
					
						
							|  |  |  |     if generator: | 
					
						
							|  |  |  |         state = '%s' % state | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         state = '%s, defined' % state | 
					
						
							|  |  |  |     if source_traceback is not None: | 
					
						
							|  |  |  |         frame = source_traceback[-1] | 
					
						
							|  |  |  |         return ('coro=<%s() %s at %s> created at %s:%s' | 
					
						
							|  |  |  |                 % (qualname, state, src, frame[0], frame[1])) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return 'coro=<%s() %s at %s>' % (qualname, state, src) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  | def get_innermost_context(exc): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Return information about the innermost exception context in the chain. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     depth = 0 | 
					
						
							|  |  |  |     while True: | 
					
						
							|  |  |  |         context = exc.__context__ | 
					
						
							|  |  |  |         if context is None: | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exc = context | 
					
						
							|  |  |  |         depth += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return (type(exc), exc.args, depth) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | class Dummy: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         return '<Dummy>' | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __call__(self, *args): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-19 07:18:45 -05:00
										 |  |  | class CoroLikeObject: | 
					
						
							|  |  |  |     def send(self, v): | 
					
						
							|  |  |  |         raise StopIteration(42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def throw(self, *exc): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def close(self): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __await__(self): | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | class BaseTaskTests: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Task = None | 
					
						
							|  |  |  |     Future = None | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = None | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:54:13 +02:00
										 |  |  |     def new_task(self, loop, coro, name='TestTask', context=None): | 
					
						
							|  |  |  |         return self.__class__.Task(coro, loop=loop, name=name, context=context) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def new_future(self, loop): | 
					
						
							|  |  |  |         return self.__class__.Future(loop=loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2016-11-04 14:29:28 -04:00
										 |  |  |         super().setUp() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.loop = self.new_test_loop() | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         self.loop.set_task_factory(self.new_task) | 
					
						
							|  |  |  |         self.loop.create_future = lambda: self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 16:58:53 +05:30
										 |  |  |     def test_generic_alias(self): | 
					
						
							|  |  |  |         task = self.__class__.Task[str] | 
					
						
							|  |  |  |         self.assertEqual(task.__args__, (str,)) | 
					
						
							|  |  |  |         self.assertIsInstance(task, GenericAlias) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |     def test_task_cancel_message_getter(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         t = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         self.assertTrue(hasattr(t, '_cancel_message')) | 
					
						
							|  |  |  |         self.assertEqual(t._cancel_message, None) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |         t.cancel('my message') | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |         self.assertEqual(t._cancel_message, 'my message') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError) as cm: | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             self.loop.run_until_complete(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         self.assertEqual('my message', cm.exception.args[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |     def test_task_cancel_message_setter(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         t = self.new_task(self.loop, coro()) | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |         t.cancel('my message') | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |         t._cancel_message = 'my new message' | 
					
						
							|  |  |  |         self.assertEqual(t._cancel_message, 'my new message') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError) as cm: | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             self.loop.run_until_complete(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         self.assertEqual('my new message', cm.exception.args[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-01 23:31:07 -07:00
										 |  |  |     def test_task_del_collect(self): | 
					
						
							|  |  |  |         class Evil: | 
					
						
							|  |  |  |             def __del__(self): | 
					
						
							|  |  |  |                 gc.collect() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def run(): | 
					
						
							| 
									
										
										
										
											2017-08-01 23:31:07 -07:00
										 |  |  |             return Evil() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete( | 
					
						
							|  |  |  |             asyncio.gather(*[ | 
					
						
							|  |  |  |                 self.new_task(self.loop, run()) for _ in range(100) | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             ])) | 
					
						
							| 
									
										
										
										
											2017-08-01 23:31:07 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-11 11:33:59 -05:00
										 |  |  |     def test_other_loop_future(self): | 
					
						
							|  |  |  |         other_loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(other_loop) | 
					
						
							| 
									
										
										
										
											2015-12-11 11:33:59 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def run(fut): | 
					
						
							|  |  |  |             await fut | 
					
						
							| 
									
										
										
										
											2015-12-11 11:33:59 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             with self.assertRaisesRegex(RuntimeError, | 
					
						
							|  |  |  |                                         r'Task .* got Future .* attached'): | 
					
						
							|  |  |  |                 self.loop.run_until_complete(run(fut)) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             other_loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-09 12:19:12 -04:00
										 |  |  |     def test_task_awaits_on_itself(self): | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         async def test(): | 
					
						
							|  |  |  |             await task | 
					
						
							| 
									
										
										
										
											2016-10-09 12:19:12 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         task = asyncio.ensure_future(test(), loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaisesRegex(RuntimeError, | 
					
						
							|  |  |  |                                     'Task cannot await on itself'): | 
					
						
							|  |  |  |             self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_task_class(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 'ok' | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, notmuch()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'ok') | 
					
						
							|  |  |  |         self.assertIs(t._loop, self.loop) | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |         self.assertIs(t.get_loop(), self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.set_event_loop(loop) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(loop, notmuch()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(t._loop, loop) | 
					
						
							| 
									
										
										
										
											2014-06-25 23:32:25 +02:00
										 |  |  |         loop.run_until_complete(t) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |     def test_ensure_future_coroutine(self): | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							|  |  |  |             return 'ok' | 
					
						
							|  |  |  |         t = asyncio.ensure_future(notmuch(), loop=self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._loop, self.loop) | 
					
						
							|  |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'ok') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         a = notmuch() | 
					
						
							|  |  |  |         self.addCleanup(a.close) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no current event loop'): | 
					
						
							|  |  |  |             asyncio.ensure_future(a) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         async def test(): | 
					
						
							|  |  |  |             return asyncio.ensure_future(notmuch()) | 
					
						
							|  |  |  |         t = self.loop.run_until_complete(test()) | 
					
						
							|  |  |  |         self.assertIs(t._loop, self.loop) | 
					
						
							|  |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'ok') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         # Deprecated in 3.10, undeprecated in 3.12 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         asyncio.set_event_loop(self.loop) | 
					
						
							|  |  |  |         self.addCleanup(asyncio.set_event_loop, None) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         t = asyncio.ensure_future(notmuch()) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.assertIs(t._loop, self.loop) | 
					
						
							|  |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'ok') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |     def test_ensure_future_future(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         f_orig = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         f_orig.set_result('ko') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(f_orig) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.loop.run_until_complete(f) | 
					
						
							|  |  |  |         self.assertTrue(f.done()) | 
					
						
							|  |  |  |         self.assertEqual(f.result(), 'ko') | 
					
						
							|  |  |  |         self.assertIs(f, f_orig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.set_event_loop(loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |             f = asyncio.ensure_future(f_orig, loop=loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(f_orig, loop=self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(f, f_orig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |     def test_ensure_future_task(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 'ok' | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t_orig = self.new_task(self.loop, notmuch()) | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         t = asyncio.ensure_future(t_orig) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'ok') | 
					
						
							|  |  |  |         self.assertIs(t, t_orig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.set_event_loop(loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |             t = asyncio.ensure_future(t_orig, loop=loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         t = asyncio.ensure_future(t_orig, loop=self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(t, t_orig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-02 15:00:19 -04:00
										 |  |  |     def test_ensure_future_awaitable(self): | 
					
						
							|  |  |  |         class Aw: | 
					
						
							|  |  |  |             def __init__(self, coro): | 
					
						
							|  |  |  |                 self.coro = coro | 
					
						
							|  |  |  |             def __await__(self): | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |                 return self.coro.__await__() | 
					
						
							| 
									
										
										
										
											2015-10-02 15:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 'ok' | 
					
						
							| 
									
										
										
										
											2015-10-02 15:00:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  |         fut = asyncio.ensure_future(Aw(coro()), loop=loop) | 
					
						
							|  |  |  |         loop.run_until_complete(fut) | 
					
						
							| 
									
										
										
										
											2021-12-20 12:23:05 +02:00
										 |  |  |         self.assertEqual(fut.result(), 'ok') | 
					
						
							| 
									
										
										
										
											2015-10-02 15:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-16 09:20:43 +05:30
										 |  |  |     def test_ensure_future_task_awaitable(self): | 
					
						
							|  |  |  |         class Aw: | 
					
						
							|  |  |  |             def __await__(self): | 
					
						
							|  |  |  |                 return asyncio.sleep(0, result='ok').__await__() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  |         task = asyncio.ensure_future(Aw(), loop=loop) | 
					
						
							|  |  |  |         loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertTrue(task.done()) | 
					
						
							|  |  |  |         self.assertEqual(task.result(), 'ok') | 
					
						
							|  |  |  |         self.assertIsInstance(task.get_coro(), types.CoroutineType) | 
					
						
							|  |  |  |         loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |     def test_ensure_future_neither(self): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         with self.assertRaises(TypeError): | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |             asyncio.ensure_future('ok') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-03 09:35:26 -06:00
										 |  |  |     def test_ensure_future_error_msg(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         f = self.new_future(self.loop) | 
					
						
							|  |  |  |         with self.assertRaisesRegex(ValueError, 'The future belongs to a ' | 
					
						
							|  |  |  |                                     'different loop than the one specified as ' | 
					
						
							|  |  |  |                                     'the loop argument'): | 
					
						
							|  |  |  |             asyncio.ensure_future(f, loop=loop) | 
					
						
							|  |  |  |         loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  |     def test_get_stack(self): | 
					
						
							|  |  |  |         T = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def foo(): | 
					
						
							|  |  |  |             await bar() | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def bar(): | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  |             # test get_stack() | 
					
						
							|  |  |  |             f = T.get_stack(limit=1) | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 self.assertEqual(f[0].f_code.co_name, 'foo') | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 f = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # test print_stack() | 
					
						
							|  |  |  |             file = io.StringIO() | 
					
						
							|  |  |  |             T.print_stack(limit=1, file=file) | 
					
						
							|  |  |  |             file.seek(0) | 
					
						
							|  |  |  |             tb = file.read() | 
					
						
							|  |  |  |             self.assertRegex(tb, r'foo\(\) running') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def runner(): | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  |             nonlocal T | 
					
						
							|  |  |  |             T = asyncio.ensure_future(foo(), loop=self.loop) | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             await T | 
					
						
							| 
									
										
										
										
											2015-08-14 15:30:59 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete(runner()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_task_repr(self): | 
					
						
							| 
									
										
										
										
											2014-07-29 12:58:23 +02:00
										 |  |  |         self.loop.set_debug(False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 'abc' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         # test coroutine function | 
					
						
							| 
									
										
										
										
											2014-06-18 01:14:59 +02:00
										 |  |  |         self.assertEqual(notmuch.__name__, 'notmuch') | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         self.assertRegex(notmuch.__qualname__, | 
					
						
							|  |  |  |                          r'\w+.test_task_repr.<locals>.notmuch') | 
					
						
							| 
									
										
										
										
											2014-06-18 01:14:59 +02:00
										 |  |  |         self.assertEqual(notmuch.__module__, __name__) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  |         filename, lineno = test_utils.get_function_source(notmuch) | 
					
						
							|  |  |  |         src = "%s:%s" % (filename, lineno) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         # test coroutine object | 
					
						
							| 
									
										
										
										
											2014-06-18 01:14:59 +02:00
										 |  |  |         gen = notmuch() | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         coro_qualname = 'BaseTaskTests.test_task_repr.<locals>.notmuch' | 
					
						
							| 
									
										
										
										
											2014-06-18 01:14:59 +02:00
										 |  |  |         self.assertEqual(gen.__name__, 'notmuch') | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         self.assertEqual(gen.__qualname__, coro_qualname) | 
					
						
							| 
									
										
										
										
											2014-06-18 01:14:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         # test pending Task | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         t.add_done_callback(Dummy()) | 
					
						
							| 
									
										
										
										
											2014-07-11 00:21:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         coro = format_coroutine(coro_qualname, 'running', src, | 
					
						
							|  |  |  |                                 t._source_traceback, generator=True) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  |         self.assertEqual(repr(t), | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |                          "<Task pending name='TestTask' %s cb=[<Dummy>()]>" % coro) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         # test cancelling Task | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         t.cancel()  # Does not take immediate effect! | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  |         self.assertEqual(repr(t), | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |                          "<Task cancelling name='TestTask' %s cb=[<Dummy>()]>" % coro) | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # test cancelled Task | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertRaises(asyncio.CancelledError, | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                           self.loop.run_until_complete, t) | 
					
						
							| 
									
										
										
										
											2014-07-11 00:21:27 +02:00
										 |  |  |         coro = format_coroutine(coro_qualname, 'done', src, | 
					
						
							|  |  |  |                                 t._source_traceback) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  |         self.assertEqual(repr(t), | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |                          "<Task cancelled name='TestTask' %s>" % coro) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 21:41:58 +02:00
										 |  |  |         # test finished Task | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, notmuch()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.loop.run_until_complete(t) | 
					
						
							| 
									
										
										
										
											2014-07-11 00:21:27 +02:00
										 |  |  |         coro = format_coroutine(coro_qualname, 'done', src, | 
					
						
							|  |  |  |                                 t._source_traceback) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:39:26 +02:00
										 |  |  |         self.assertEqual(repr(t), | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |                          "<Task finished name='TestTask' %s result='abc'>" % coro) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_task_repr_autogenerated(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |             return 123 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t1 = self.new_task(self.loop, notmuch(), None) | 
					
						
							|  |  |  |         t2 = self.new_task(self.loop, notmuch(), None) | 
					
						
							|  |  |  |         self.assertNotEqual(repr(t1), repr(t2)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 21:32:30 -07:00
										 |  |  |         match1 = re.match(r"^<Task pending name='Task-(\d+)'", repr(t1)) | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |         self.assertIsNotNone(match1) | 
					
						
							| 
									
										
										
										
											2018-08-13 21:32:30 -07:00
										 |  |  |         match2 = re.match(r"^<Task pending name='Task-(\d+)'", repr(t2)) | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |         self.assertIsNotNone(match2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Autogenerated task names should have monotonically increasing numbers | 
					
						
							|  |  |  |         self.assertLess(int(match1.group(1)), int(match2.group(1))) | 
					
						
							|  |  |  |         self.loop.run_until_complete(t1) | 
					
						
							|  |  |  |         self.loop.run_until_complete(t2) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-29 08:20:09 -07:00
										 |  |  |     def test_task_set_name_pylong(self): | 
					
						
							|  |  |  |         # test that setting the task name to a PyLong explicitly doesn't | 
					
						
							|  |  |  |         # incorrectly trigger the deferred name formatting logic | 
					
						
							|  |  |  |         async def notmuch(): | 
					
						
							|  |  |  |             return 123 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = self.new_task(self.loop, notmuch(), name=987654321) | 
					
						
							|  |  |  |         self.assertEqual(t.get_name(), '987654321') | 
					
						
							|  |  |  |         t.set_name(123456789) | 
					
						
							|  |  |  |         self.assertEqual(t.get_name(), '123456789') | 
					
						
							|  |  |  |         self.loop.run_until_complete(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |     def test_task_repr_name_not_str(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |             return 123 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = self.new_task(self.loop, notmuch()) | 
					
						
							|  |  |  |         t.set_name({6}) | 
					
						
							|  |  |  |         self.assertEqual(t.get_name(), '{6}') | 
					
						
							|  |  |  |         self.loop.run_until_complete(t) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-03 00:59:00 +02:00
										 |  |  |     def test_task_repr_wait_for(self): | 
					
						
							| 
									
										
										
										
											2014-07-29 12:58:23 +02:00
										 |  |  |         self.loop.set_debug(False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def wait_for(fut): | 
					
						
							|  |  |  |             return await fut | 
					
						
							| 
									
										
										
										
											2014-07-03 00:59:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							|  |  |  |         task = self.new_task(self.loop, wait_for(fut)) | 
					
						
							| 
									
										
										
										
											2014-07-03 00:59:00 +02:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertRegex(repr(task), | 
					
						
							|  |  |  |                          '<Task .* wait_for=%s>' % re.escape(repr(fut))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-08 11:29:25 +02:00
										 |  |  |         fut.set_result(None) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_task_basics(self): | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         async def outer(): | 
					
						
							|  |  |  |             a = await inner1() | 
					
						
							|  |  |  |             b = await inner2() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return a+b | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def inner1(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def inner2(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 1000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = outer() | 
					
						
							|  |  |  |         self.assertEqual(self.loop.run_until_complete(t), 1042) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-13 16:18:27 -07:00
										 |  |  |     def test_exception_chaining_after_await(self): | 
					
						
							|  |  |  |         # Test that when awaiting on a task when an exception is already | 
					
						
							|  |  |  |         # active, if the task raises an exception it will be chained | 
					
						
							|  |  |  |         # with the original. | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def raise_error(): | 
					
						
							|  |  |  |             raise ValueError | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def run(): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 raise KeyError(3) | 
					
						
							|  |  |  |             except Exception as exc: | 
					
						
							|  |  |  |                 task = self.new_task(loop, raise_error()) | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     await task | 
					
						
							|  |  |  |                 except Exception as exc: | 
					
						
							|  |  |  |                     self.assertEqual(type(exc), ValueError) | 
					
						
							|  |  |  |                     chained = exc.__context__ | 
					
						
							|  |  |  |                     self.assertEqual((type(chained), chained.args), | 
					
						
							|  |  |  |                         (KeyError, (3,))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-22 13:33:27 -07:00
										 |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, run()) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_exception_chaining_after_await_with_context_cycle(self): | 
					
						
							|  |  |  |         # Check trying to create an exception context cycle: | 
					
						
							|  |  |  |         # https://bugs.python.org/issue40696 | 
					
						
							|  |  |  |         has_cycle = None | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def process_exc(exc): | 
					
						
							|  |  |  |             raise exc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def run(): | 
					
						
							|  |  |  |             nonlocal has_cycle | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 raise KeyError('a') | 
					
						
							|  |  |  |             except Exception as exc: | 
					
						
							|  |  |  |                 task = self.new_task(loop, process_exc(exc)) | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     await task | 
					
						
							|  |  |  |                 except BaseException as exc: | 
					
						
							|  |  |  |                     has_cycle = (exc is exc.__context__) | 
					
						
							|  |  |  |                     # Prevent a hang if has_cycle is True. | 
					
						
							|  |  |  |                     exc.__context__ = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, run()) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  |         # This also distinguishes from the initial has_cycle=None. | 
					
						
							|  |  |  |         self.assertEqual(has_cycle, False) | 
					
						
							| 
									
										
										
										
											2020-05-13 16:18:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_cancelling(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def task(): | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             t = self.new_task(loop, task()) | 
					
						
							|  |  |  |             self.assertFalse(t.cancelling()) | 
					
						
							|  |  |  |             self.assertNotIn(" cancelling ", repr(t)) | 
					
						
							|  |  |  |             self.assertTrue(t.cancel()) | 
					
						
							|  |  |  |             self.assertTrue(t.cancelling()) | 
					
						
							|  |  |  |             self.assertIn(" cancelling ", repr(t)) | 
					
						
							| 
									
										
										
										
											2022-02-28 15:15:56 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Since we commented out two lines from Task.cancel(), | 
					
						
							|  |  |  |             # this t.cancel() call now returns True. | 
					
						
							|  |  |  |             # self.assertFalse(t.cancel()) | 
					
						
							|  |  |  |             self.assertTrue(t.cancel()) | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             with self.assertRaises(asyncio.CancelledError): | 
					
						
							|  |  |  |                 loop.run_until_complete(t) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |     def test_uncancel_basic(self): | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def task(): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 await asyncio.sleep(10) | 
					
						
							|  |  |  |             except asyncio.CancelledError: | 
					
						
							|  |  |  |                 asyncio.current_task().uncancel() | 
					
						
							|  |  |  |                 await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             t = self.new_task(loop, task()) | 
					
						
							|  |  |  |             loop.run_until_complete(asyncio.sleep(0.01)) | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Cancel first sleep | 
					
						
							|  |  |  |             self.assertTrue(t.cancel()) | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  |             self.assertIn(" cancelling ", repr(t)) | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |             self.assertEqual(t.cancelling(), 1) | 
					
						
							|  |  |  |             self.assertFalse(t.cancelled())  # Task is still not complete | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  |             loop.run_until_complete(asyncio.sleep(0.01)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |             # after .uncancel() | 
					
						
							|  |  |  |             self.assertNotIn(" cancelling ", repr(t)) | 
					
						
							|  |  |  |             self.assertEqual(t.cancelling(), 0) | 
					
						
							|  |  |  |             self.assertFalse(t.cancelled())  # Task is still not complete | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Cancel second sleep | 
					
						
							|  |  |  |             self.assertTrue(t.cancel()) | 
					
						
							|  |  |  |             self.assertEqual(t.cancelling(), 1) | 
					
						
							|  |  |  |             self.assertFalse(t.cancelled())  # Task is still not complete | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  |             with self.assertRaises(asyncio.CancelledError): | 
					
						
							|  |  |  |                 loop.run_until_complete(t) | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |             self.assertTrue(t.cancelled())  # Finally, task complete | 
					
						
							|  |  |  |             self.assertTrue(t.done()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # uncancel is no longer effective after the task is complete | 
					
						
							|  |  |  |             t.uncancel() | 
					
						
							|  |  |  |             self.assertTrue(t.cancelled()) | 
					
						
							|  |  |  |             self.assertTrue(t.done()) | 
					
						
							| 
									
										
										
										
											2022-02-15 15:42:04 -08:00
										 |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |     def test_uncancel_structured_blocks(self): | 
					
						
							|  |  |  |         # This test recreates the following high-level structure using uncancel():: | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         #     async def make_request_with_timeout(): | 
					
						
							|  |  |  |         #         try: | 
					
						
							|  |  |  |         #             async with asyncio.timeout(1): | 
					
						
							|  |  |  |         #                 # Structured block affected by the timeout: | 
					
						
							|  |  |  |         #                 await make_request() | 
					
						
							|  |  |  |         #                 await make_another_request() | 
					
						
							|  |  |  |         #         except TimeoutError: | 
					
						
							|  |  |  |         #             pass  # There was a timeout | 
					
						
							|  |  |  |         #         # Outer code not affected by the timeout: | 
					
						
							|  |  |  |         #         await unrelated_code() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def make_request_with_timeout(*, sleep: float, timeout: float): | 
					
						
							|  |  |  |             task = asyncio.current_task() | 
					
						
							|  |  |  |             loop = task.get_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             timed_out = False | 
					
						
							|  |  |  |             structured_block_finished = False | 
					
						
							|  |  |  |             outer_code_reached = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def on_timeout(): | 
					
						
							|  |  |  |                 nonlocal timed_out | 
					
						
							|  |  |  |                 timed_out = True | 
					
						
							|  |  |  |                 task.cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             timeout_handle = loop.call_later(timeout, on_timeout) | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     # Structured block affected by the timeout | 
					
						
							|  |  |  |                     await asyncio.sleep(sleep) | 
					
						
							|  |  |  |                     structured_block_finished = True | 
					
						
							|  |  |  |                 finally: | 
					
						
							|  |  |  |                     timeout_handle.cancel() | 
					
						
							|  |  |  |                     if ( | 
					
						
							|  |  |  |                         timed_out | 
					
						
							|  |  |  |                         and task.uncancel() == 0 | 
					
						
							| 
									
										
										
										
											2023-04-06 11:08:25 +01:00
										 |  |  |                         and type(sys.exception()) is asyncio.CancelledError | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  |                     ): | 
					
						
							|  |  |  |                         # Note the five rules that are needed here to satisfy proper | 
					
						
							|  |  |  |                         # uncancellation: | 
					
						
							|  |  |  |                         # | 
					
						
							|  |  |  |                         # 1. handle uncancellation in a `finally:` block to allow for | 
					
						
							|  |  |  |                         #    plain returns; | 
					
						
							|  |  |  |                         # 2. our `timed_out` flag is set, meaning that it was our event | 
					
						
							|  |  |  |                         #    that triggered the need to uncancel the task, regardless of | 
					
						
							|  |  |  |                         #    what exception is raised; | 
					
						
							|  |  |  |                         # 3. we can call `uncancel()` because *we* called `cancel()` | 
					
						
							|  |  |  |                         #    before; | 
					
						
							|  |  |  |                         # 4. we call `uncancel()` but we only continue converting the | 
					
						
							|  |  |  |                         #    CancelledError to TimeoutError if `uncancel()` caused the | 
					
						
							|  |  |  |                         #    cancellation request count go down to 0.  We need to look | 
					
						
							|  |  |  |                         #    at the counter vs having a simple boolean flag because our | 
					
						
							|  |  |  |                         #    code might have been nested (think multiple timeouts). See | 
					
						
							|  |  |  |                         #    commit 7fce1063b6e5a366f8504e039a8ccdd6944625cd for | 
					
						
							|  |  |  |                         #    details. | 
					
						
							|  |  |  |                         # 5. we only convert CancelledError to TimeoutError; for other | 
					
						
							|  |  |  |                         #    exceptions raised due to the cancellation (like | 
					
						
							|  |  |  |                         #    a ConnectionLostError from a database client), simply | 
					
						
							|  |  |  |                         #    propagate them. | 
					
						
							|  |  |  |                         # | 
					
						
							|  |  |  |                         # Those checks need to take place in this exact order to make | 
					
						
							|  |  |  |                         # sure the `cancelling()` counter always stays in sync. | 
					
						
							|  |  |  |                         # | 
					
						
							|  |  |  |                         # Additionally, the original stimulus to `cancel()` the task | 
					
						
							|  |  |  |                         # needs to be unscheduled to avoid re-cancelling the task later. | 
					
						
							|  |  |  |                         # Here we do it by cancelling `timeout_handle` in the `finally:` | 
					
						
							|  |  |  |                         # block. | 
					
						
							|  |  |  |                         raise TimeoutError | 
					
						
							|  |  |  |             except TimeoutError: | 
					
						
							|  |  |  |                 self.assertTrue(timed_out) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Outer code not affected by the timeout: | 
					
						
							|  |  |  |             outer_code_reached = True | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             return timed_out, structured_block_finished, outer_code_reached | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-13 04:16:44 +08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             # Test which timed out. | 
					
						
							|  |  |  |             t1 = self.new_task(loop, make_request_with_timeout(sleep=10.0, timeout=0.1)) | 
					
						
							|  |  |  |             timed_out, structured_block_finished, outer_code_reached = ( | 
					
						
							|  |  |  |                 loop.run_until_complete(t1) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             self.assertTrue(timed_out) | 
					
						
							|  |  |  |             self.assertFalse(structured_block_finished)  # it was cancelled | 
					
						
							|  |  |  |             self.assertTrue(outer_code_reached)  # task got uncancelled after leaving | 
					
						
							|  |  |  |                                                  # the structured block and continued until | 
					
						
							|  |  |  |                                                  # completion | 
					
						
							|  |  |  |             self.assertEqual(t1.cancelling(), 0) # no pending cancellation of the outer task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Test which did not time out. | 
					
						
							|  |  |  |             t2 = self.new_task(loop, make_request_with_timeout(sleep=0, timeout=10.0)) | 
					
						
							|  |  |  |             timed_out, structured_block_finished, outer_code_reached = ( | 
					
						
							|  |  |  |                 loop.run_until_complete(t2) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             self.assertFalse(timed_out) | 
					
						
							|  |  |  |             self.assertTrue(structured_block_finished) | 
					
						
							|  |  |  |             self.assertTrue(outer_code_reached) | 
					
						
							|  |  |  |             self.assertEqual(t2.cancelling(), 0) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							| 
									
										
										
										
											2022-10-01 19:42:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 08:17:28 -07:00
										 |  |  |     def test_uncancel_resets_must_cancel(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             await fut | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         fut = asyncio.Future(loop=loop) | 
					
						
							|  |  |  |         task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |         loop.run_until_complete(asyncio.sleep(0))  # Get task waiting for fut | 
					
						
							|  |  |  |         fut.set_result(None)  # Make task runnable | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task.cancel()  # Enter cancelled state | 
					
						
							|  |  |  |             self.assertEqual(task.cancelling(), 1) | 
					
						
							|  |  |  |             self.assertTrue(task._must_cancel) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             task.uncancel()  # Undo cancellation | 
					
						
							|  |  |  |             self.assertEqual(task.cancelling(), 0) | 
					
						
							|  |  |  |             self.assertFalse(task._must_cancel) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             res = loop.run_until_complete(task) | 
					
						
							|  |  |  |             self.assertEqual(res, 42) | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_cancel(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(10.0) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 12 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         loop.call_soon(t.cancel) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							|  |  |  |         self.assertFalse(t.cancel()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |     def test_cancel_with_message_then_future_result(self): | 
					
						
							|  |  |  |         # Test Future.result() after calling cancel() with a message. | 
					
						
							|  |  |  |         cases = [ | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |             ((), ()), | 
					
						
							|  |  |  |             ((None,), ()), | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             (('my message',), ('my message',)), | 
					
						
							|  |  |  |             # Non-string values should roundtrip. | 
					
						
							|  |  |  |             ((5,), (5,)), | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         for cancel_args, expected_args in cases: | 
					
						
							|  |  |  |             with self.subTest(cancel_args=cancel_args): | 
					
						
							|  |  |  |                 loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |                 self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def sleep(): | 
					
						
							|  |  |  |                     await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def coro(): | 
					
						
							|  |  |  |                     task = self.new_task(loop, sleep()) | 
					
						
							|  |  |  |                     await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |                     task.cancel(*cancel_args) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |                     done, pending = await asyncio.wait([task]) | 
					
						
							|  |  |  |                     task.result() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |                 with self.assertRaises(asyncio.CancelledError) as cm: | 
					
						
							|  |  |  |                     loop.run_until_complete(task) | 
					
						
							|  |  |  |                 exc = cm.exception | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                 self.assertEqual(exc.args, expected_args) | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 actual = get_innermost_context(exc) | 
					
						
							|  |  |  |                 self.assertEqual(actual, | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                     (asyncio.CancelledError, expected_args, 0)) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_with_message_then_future_exception(self): | 
					
						
							|  |  |  |         # Test Future.exception() after calling cancel() with a message. | 
					
						
							|  |  |  |         cases = [ | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |             ((), ()), | 
					
						
							|  |  |  |             ((None,), ()), | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             (('my message',), ('my message',)), | 
					
						
							|  |  |  |             # Non-string values should roundtrip. | 
					
						
							|  |  |  |             ((5,), (5,)), | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         for cancel_args, expected_args in cases: | 
					
						
							|  |  |  |             with self.subTest(cancel_args=cancel_args): | 
					
						
							|  |  |  |                 loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |                 self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def sleep(): | 
					
						
							|  |  |  |                     await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def coro(): | 
					
						
							|  |  |  |                     task = self.new_task(loop, sleep()) | 
					
						
							|  |  |  |                     await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |                     task.cancel(*cancel_args) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |                     done, pending = await asyncio.wait([task]) | 
					
						
							|  |  |  |                     task.exception() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |                 with self.assertRaises(asyncio.CancelledError) as cm: | 
					
						
							|  |  |  |                     loop.run_until_complete(task) | 
					
						
							|  |  |  |                 exc = cm.exception | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                 self.assertEqual(exc.args, expected_args) | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 actual = get_innermost_context(exc) | 
					
						
							|  |  |  |                 self.assertEqual(actual, | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                     (asyncio.CancelledError, expected_args, 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancellation_exception_context(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  |         fut = loop.create_future() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def sleep(): | 
					
						
							|  |  |  |             fut.set_result(None) | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             inner_task = self.new_task(loop, sleep()) | 
					
						
							|  |  |  |             await fut | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |             loop.call_soon(inner_task.cancel, 'msg') | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 await inner_task | 
					
						
							|  |  |  |             except asyncio.CancelledError as ex: | 
					
						
							|  |  |  |                 raise ValueError("cancelled") from ex | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |         with self.assertRaises(ValueError) as cm: | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         exc = cm.exception | 
					
						
							|  |  |  |         self.assertEqual(exc.args, ('cancelled',)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         actual = get_innermost_context(exc) | 
					
						
							|  |  |  |         self.assertEqual(actual, | 
					
						
							|  |  |  |             (asyncio.CancelledError, ('msg',), 1)) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_with_message_before_starting_task(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def sleep(): | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             task = self.new_task(loop, sleep()) | 
					
						
							|  |  |  |             # We deliberately leave out the sleep here. | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |             task.cancel('my message') | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             done, pending = await asyncio.wait([task]) | 
					
						
							|  |  |  |             task.exception() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |         with self.assertRaises(asyncio.CancelledError) as cm: | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         exc = cm.exception | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |         self.assertEqual(exc.args, ('my message',)) | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         actual = get_innermost_context(exc) | 
					
						
							|  |  |  |         self.assertEqual(actual, | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |             (asyncio.CancelledError, ('my message',), 0)) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_cancel_yield(self): | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def task(): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             return 12 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop)  # start coro | 
					
						
							|  |  |  |         t.cancel() | 
					
						
							|  |  |  |         self.assertRaises( | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             asyncio.CancelledError, self.loop.run_until_complete, t) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							|  |  |  |         self.assertFalse(t.cancel()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_inner_future(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         f = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							|  |  |  |             await f | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 12 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop)  # start task | 
					
						
							|  |  |  |         f.cancel() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(f.cancelled()) | 
					
						
							|  |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_both_task_and_inner_future(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         f = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							|  |  |  |             await f | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 12 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         f.cancel() | 
					
						
							|  |  |  |         t.cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.loop.run_until_complete(t) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertTrue(f.cancelled()) | 
					
						
							|  |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_task_catching(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut1 = self.new_future(self.loop) | 
					
						
							|  |  |  |         fut2 = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							|  |  |  |             await fut1 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                 await fut2 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._fut_waiter, fut1)  # White-box test. | 
					
						
							|  |  |  |         fut1.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._fut_waiter, fut2)  # White-box test. | 
					
						
							|  |  |  |         t.cancel() | 
					
						
							|  |  |  |         self.assertTrue(fut2.cancelled()) | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  |         self.assertFalse(t.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_task_ignoring(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut1 = self.new_future(self.loop) | 
					
						
							|  |  |  |         fut2 = self.new_future(self.loop) | 
					
						
							|  |  |  |         fut3 = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							|  |  |  |             await fut1 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                 await fut2 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 pass | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             res = await fut3 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, task()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._fut_waiter, fut1)  # White-box test. | 
					
						
							|  |  |  |         fut1.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._fut_waiter, fut2)  # White-box test. | 
					
						
							|  |  |  |         t.cancel() | 
					
						
							|  |  |  |         self.assertTrue(fut2.cancelled()) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(t._fut_waiter, fut3)  # White-box test. | 
					
						
							|  |  |  |         fut3.set_result(42) | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  |         self.assertFalse(fut3.cancelled()) | 
					
						
							|  |  |  |         self.assertFalse(t.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_current_task(self): | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.set_event_loop(loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             t.cancel() | 
					
						
							|  |  |  |             self.assertTrue(t._must_cancel)  # White-box test. | 
					
						
							|  |  |  |             # The sleep should be cancelled immediately. | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(100) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 12 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(loop, task()) | 
					
						
							| 
									
										
										
										
											2019-09-25 03:32:08 -07:00
										 |  |  |         self.assertFalse(t.cancelled()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertRaises( | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             asyncio.CancelledError, loop.run_until_complete, t) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertTrue(t.done()) | 
					
						
							| 
									
										
										
										
											2019-09-25 03:32:08 -07:00
										 |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertFalse(t._must_cancel)  # White-box test. | 
					
						
							| 
									
										
										
										
											2017-05-11 21:18:38 +09:00
										 |  |  |         self.assertFalse(t.cancel()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_at_end(self): | 
					
						
							|  |  |  |         """coroutine end right after task is cancelled""" | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def task(): | 
					
						
							| 
									
										
										
										
											2017-05-11 21:18:38 +09:00
										 |  |  |             t.cancel() | 
					
						
							|  |  |  |             self.assertTrue(t._must_cancel)  # White-box test. | 
					
						
							|  |  |  |             return 12 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = self.new_task(loop, task()) | 
					
						
							| 
									
										
										
										
											2019-09-25 03:32:08 -07:00
										 |  |  |         self.assertFalse(t.cancelled()) | 
					
						
							| 
									
										
										
										
											2017-05-11 21:18:38 +09:00
										 |  |  |         self.assertRaises( | 
					
						
							|  |  |  |             asyncio.CancelledError, loop.run_until_complete, t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							| 
									
										
										
										
											2019-09-25 03:32:08 -07:00
										 |  |  |         self.assertTrue(t.cancelled()) | 
					
						
							| 
									
										
										
										
											2017-05-11 21:18:38 +09:00
										 |  |  |         self.assertFalse(t._must_cancel)  # White-box test. | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertFalse(t.cancel()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-03 10:30:31 -04:00
										 |  |  |     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()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |     def assert_text_contains(self, text, substr): | 
					
						
							|  |  |  |         if substr not in text: | 
					
						
							|  |  |  |             raise RuntimeError(f'text {substr!r} not found in:\n>>>{text}<<<') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_traceback_for_future_result(self): | 
					
						
							|  |  |  |         # When calling Future.result() on a cancelled task, check that the | 
					
						
							|  |  |  |         # line of code that was interrupted is included in the traceback. | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def nested(): | 
					
						
							|  |  |  |             # This will get cancelled immediately. | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             task = self.new_task(loop, nested()) | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             task.cancel() | 
					
						
							|  |  |  |             await task  # search target | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         except asyncio.CancelledError: | 
					
						
							|  |  |  |             tb = traceback.format_exc() | 
					
						
							|  |  |  |             self.assert_text_contains(tb, "await asyncio.sleep(10)") | 
					
						
							|  |  |  |             # The intermediate await should also be included. | 
					
						
							|  |  |  |             self.assert_text_contains(tb, "await task  # search target") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.fail('CancelledError did not occur') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cancel_traceback_for_future_exception(self): | 
					
						
							|  |  |  |         # When calling Future.exception() on a cancelled task, check that the | 
					
						
							|  |  |  |         # line of code that was interrupted is included in the traceback. | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def nested(): | 
					
						
							|  |  |  |             # This will get cancelled immediately. | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             task = self.new_task(loop, nested()) | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             task.cancel() | 
					
						
							|  |  |  |             done, pending = await asyncio.wait([task]) | 
					
						
							|  |  |  |             task.exception()  # search target | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(loop, coro()) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         except asyncio.CancelledError: | 
					
						
							|  |  |  |             tb = traceback.format_exc() | 
					
						
							|  |  |  |             self.assert_text_contains(tb, "await asyncio.sleep(10)") | 
					
						
							|  |  |  |             # The intermediate await should also be included. | 
					
						
							|  |  |  |             self.assert_text_contains(tb, | 
					
						
							|  |  |  |                 "task.exception()  # search target") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.fail('CancelledError did not occur') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_stop_while_run_in_complete(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0.1 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.2, when) | 
					
						
							|  |  |  |             when = yield 0.1 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.3, when) | 
					
						
							|  |  |  |             yield 0.1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         x = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def task(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal x | 
					
						
							|  |  |  |             while x < 10: | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |                 await asyncio.sleep(0.1) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 x += 1 | 
					
						
							|  |  |  |                 if x == 2: | 
					
						
							|  |  |  |                     loop.stop() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(loop, task()) | 
					
						
							| 
									
										
										
										
											2014-06-25 23:57:50 +02:00
										 |  |  |         with self.assertRaises(RuntimeError) as cm: | 
					
						
							|  |  |  |             loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertEqual(str(cm.exception), | 
					
						
							|  |  |  |                          'Event loop stopped before Future completed.') | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertFalse(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(x, 2) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.3, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-25 23:57:50 +02:00
										 |  |  |         t.cancel() | 
					
						
							|  |  |  |         self.assertRaises(asyncio.CancelledError, loop.run_until_complete, t) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 16:16:10 -05:00
										 |  |  |     def test_log_traceback(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         with self.assertRaisesRegex(ValueError, 'can only be set to False'): | 
					
						
							|  |  |  |             task._log_traceback = True | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_wait(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.15, when) | 
					
						
							|  |  |  |             yield 0.15 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							|  |  |  |         b = self.new_task(loop, asyncio.sleep(0.15)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             done, pending = await asyncio.wait([b, a]) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.assertEqual(done, set([a, b])) | 
					
						
							|  |  |  |             self.assertEqual(pending, set()) | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         res = loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Doing it again should take no time and exercise a different path. | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         res = loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |     def test_wait_duplicate_coroutines(self): | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def coro(s): | 
					
						
							|  |  |  |             return s | 
					
						
							| 
									
										
										
										
											2022-03-17 22:51:40 +02:00
										 |  |  |         c = self.loop.create_task(coro('test')) | 
					
						
							| 
									
										
										
										
											2019-12-30 06:50:19 -05:00
										 |  |  |         task = self.new_task( | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |             self.loop, | 
					
						
							| 
									
										
										
										
											2022-03-17 22:51:40 +02:00
										 |  |  |             asyncio.wait([c, c, self.loop.create_task(coro('spam'))])) | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-17 22:51:40 +02:00
										 |  |  |         done, pending = self.loop.run_until_complete(task) | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertFalse(pending) | 
					
						
							|  |  |  |         self.assertEqual(set(f.result() for f in done), {'test', 'spam'}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_wait_errors(self): | 
					
						
							|  |  |  |         self.assertRaises( | 
					
						
							|  |  |  |             ValueError, self.loop.run_until_complete, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait(set())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-16 18:50:39 +02:00
										 |  |  |         # -1 is an invalid return_when value | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         sleep_coro = asyncio.sleep(10.0) | 
					
						
							|  |  |  |         wait_coro = asyncio.wait([sleep_coro], return_when=-1) | 
					
						
							| 
									
										
										
										
											2014-07-16 18:50:39 +02:00
										 |  |  |         self.assertRaises(ValueError, | 
					
						
							|  |  |  |                           self.loop.run_until_complete, wait_coro) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sleep_coro.close() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_first_completed(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             yield 0.1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(10.0)) | 
					
						
							|  |  |  |         b = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task = self.new_task( | 
					
						
							|  |  |  |             loop, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait([b, a], return_when=asyncio.FIRST_COMPLETED)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         done, pending = loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual({b}, done) | 
					
						
							|  |  |  |         self.assertEqual({a}, pending) | 
					
						
							|  |  |  |         self.assertFalse(a.done()) | 
					
						
							|  |  |  |         self.assertTrue(b.done()) | 
					
						
							|  |  |  |         self.assertIsNone(b.result()) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.1, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move forward to close generator | 
					
						
							|  |  |  |         loop.advance_time(10) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_really_done(self): | 
					
						
							|  |  |  |         # there is possibility that some tasks in the pending list | 
					
						
							|  |  |  |         # became done but their callbacks haven't all been called yet | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro1(): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro2(): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         a = self.new_task(self.loop, coro1()) | 
					
						
							|  |  |  |         b = self.new_task(self.loop, coro2()) | 
					
						
							|  |  |  |         task = self.new_task( | 
					
						
							|  |  |  |             self.loop, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait([b, a], return_when=asyncio.FIRST_COMPLETED)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         done, pending = self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual({a, b}, done) | 
					
						
							|  |  |  |         self.assertTrue(a.done()) | 
					
						
							|  |  |  |         self.assertIsNone(a.result()) | 
					
						
							|  |  |  |         self.assertTrue(b.done()) | 
					
						
							|  |  |  |         self.assertIsNone(b.result()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_first_exception(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # first_exception, task already has exception | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(10.0)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def exc(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             raise ZeroDivisionError('err') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         b = self.new_task(loop, exc()) | 
					
						
							|  |  |  |         task = self.new_task( | 
					
						
							|  |  |  |             loop, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait([b, a], return_when=asyncio.FIRST_EXCEPTION)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         done, pending = loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual({b}, done) | 
					
						
							|  |  |  |         self.assertEqual({a}, pending) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move forward to close generator | 
					
						
							|  |  |  |         loop.advance_time(10) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_first_exception_in_wait(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.01, when) | 
					
						
							|  |  |  |             yield 0.01 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # first_exception, exception during waiting | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(10.0)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def exc(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(0.01) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             raise ZeroDivisionError('err') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         b = self.new_task(loop, exc()) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         task = asyncio.wait([b, a], return_when=asyncio.FIRST_EXCEPTION) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         done, pending = loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertEqual({b}, done) | 
					
						
							|  |  |  |         self.assertEqual({a}, pending) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.01, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move forward to close generator | 
					
						
							|  |  |  |         loop.advance_time(10) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_with_exception(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.15, when) | 
					
						
							|  |  |  |             yield 0.15 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def sleeper(): | 
					
						
							|  |  |  |             await asyncio.sleep(0.15) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             raise ZeroDivisionError('really') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         b = self.new_task(loop, sleeper()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             done, pending = await asyncio.wait([b, a]) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.assertEqual(len(done), 2) | 
					
						
							|  |  |  |             self.assertEqual(pending, set()) | 
					
						
							|  |  |  |             errors = set(f for f in done if f.exception() is not None) | 
					
						
							|  |  |  |             self.assertEqual(len(errors), 1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_with_timeout(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.15, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.11, when) | 
					
						
							|  |  |  |             yield 0.11 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							|  |  |  |         b = self.new_task(loop, asyncio.sleep(0.15)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             done, pending = await asyncio.wait([b, a], timeout=0.11) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.assertEqual(done, set([a])) | 
					
						
							|  |  |  |             self.assertEqual(pending, set([b])) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertAlmostEqual(0.11, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move forward to close generator | 
					
						
							|  |  |  |         loop.advance_time(10) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_concurrent_complete(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.15, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             yield 0.1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							|  |  |  |         b = self.new_task(loop, asyncio.sleep(0.15)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         done, pending = loop.run_until_complete( | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait([b, a], timeout=0.1)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(done, set([a])) | 
					
						
							|  |  |  |         self.assertEqual(pending, set([b])) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.1, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move forward to close generator | 
					
						
							|  |  |  |         loop.advance_time(10) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 19:12:52 -03:00
										 |  |  |     def test_wait_with_iterator_of_tasks(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.15, when) | 
					
						
							|  |  |  |             yield 0.15 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = self.new_test_loop(gen) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         a = self.new_task(loop, asyncio.sleep(0.1)) | 
					
						
							|  |  |  |         b = self.new_task(loop, asyncio.sleep(0.15)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def foo(): | 
					
						
							|  |  |  |             done, pending = await asyncio.wait(iter([b, a])) | 
					
						
							|  |  |  |             self.assertEqual(done, set([a, b])) | 
					
						
							|  |  |  |             self.assertEqual(pending, set()) | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         res = loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-17 06:58:43 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_generator(self): | 
					
						
							|  |  |  |         async def func(a): | 
					
						
							|  |  |  |             return a | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = self.new_test_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             tasks = (self.new_task(loop, func(i)) for i in range(10)) | 
					
						
							|  |  |  |             done, pending = await asyncio.wait(tasks, return_when=asyncio.ALL_COMPLETED) | 
					
						
							|  |  |  |             self.assertEqual(len(done), 10) | 
					
						
							|  |  |  |             self.assertEqual(len(pending), 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop.run_until_complete(main()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_as_completed(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0.01 | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def sleeper(dt, x): | 
					
						
							|  |  |  |             nonlocal time_shifted | 
					
						
							|  |  |  |             await asyncio.sleep(dt) | 
					
						
							|  |  |  |             completed.add(x) | 
					
						
							|  |  |  |             if not time_shifted and 'a' in completed and 'b' in completed: | 
					
						
							|  |  |  |                 time_shifted = True | 
					
						
							|  |  |  |                 loop.advance_time(0.14) | 
					
						
							|  |  |  |             return x | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_iterator(awaitables): | 
					
						
							|  |  |  |             values = [] | 
					
						
							|  |  |  |             for f in asyncio.as_completed(awaitables): | 
					
						
							|  |  |  |                 values.append(await f) | 
					
						
							|  |  |  |             return values | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_async_iterator(awaitables): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             values = [] | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |             async for f in asyncio.as_completed(awaitables): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |                 values.append(await f) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return values | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         for foo in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |             with self.subTest(method=foo.__name__): | 
					
						
							|  |  |  |                 loop = self.new_test_loop(gen) | 
					
						
							|  |  |  |                 # disable "slow callback" warning | 
					
						
							|  |  |  |                 loop.slow_callback_duration = 1.0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 completed = set() | 
					
						
							|  |  |  |                 time_shifted = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 a = sleeper(0.01, 'a') | 
					
						
							|  |  |  |                 b = sleeper(0.01, 'b') | 
					
						
							|  |  |  |                 c = sleeper(0.15, 'c') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 res = loop.run_until_complete(self.new_task(loop, foo([b, c, a]))) | 
					
						
							|  |  |  |                 self.assertAlmostEqual(0.15, loop.time()) | 
					
						
							|  |  |  |                 self.assertTrue('a' in res[:2]) | 
					
						
							|  |  |  |                 self.assertTrue('b' in res[:2]) | 
					
						
							|  |  |  |                 self.assertEqual(res[2], 'c') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_as_completed_same_tasks_in_as_out(self): | 
					
						
							|  |  |  |         # Ensures that asynchronously iterating as_completed's iterator | 
					
						
							|  |  |  |         # yields awaitables are the same awaitables that were passed in when | 
					
						
							|  |  |  |         # those awaitables are futures. | 
					
						
							|  |  |  |         async def try_async_iterator(awaitables): | 
					
						
							|  |  |  |             awaitables_out = set() | 
					
						
							|  |  |  |             async for out_aw in asyncio.as_completed(awaitables): | 
					
						
							|  |  |  |                 awaitables_out.add(out_aw) | 
					
						
							|  |  |  |             return awaitables_out | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(i): | 
					
						
							|  |  |  |             return i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with contextlib.closing(asyncio.new_event_loop()) as loop: | 
					
						
							|  |  |  |             # Coroutines shouldn't be yielded back as finished coroutines | 
					
						
							|  |  |  |             # can't be re-used. | 
					
						
							|  |  |  |             awaitables_in = frozenset( | 
					
						
							|  |  |  |                 (coro(0), coro(1), coro(2), coro(3)) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             awaitables_out = loop.run_until_complete( | 
					
						
							|  |  |  |                 try_async_iterator(awaitables_in) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             if awaitables_in - awaitables_out != awaitables_in: | 
					
						
							|  |  |  |                 raise self.failureException('Got original coroutines ' | 
					
						
							|  |  |  |                                             'out of as_completed iterator.') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Tasks should be yielded back. | 
					
						
							|  |  |  |             coro_obj_a = coro('a') | 
					
						
							|  |  |  |             task_b = loop.create_task(coro('b')) | 
					
						
							|  |  |  |             coro_obj_c = coro('c') | 
					
						
							|  |  |  |             task_d = loop.create_task(coro('d')) | 
					
						
							|  |  |  |             awaitables_in = frozenset( | 
					
						
							|  |  |  |                 (coro_obj_a, task_b, coro_obj_c, task_d) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             awaitables_out = loop.run_until_complete( | 
					
						
							|  |  |  |                 try_async_iterator(awaitables_in) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             if awaitables_in & awaitables_out != {task_b, task_d}: | 
					
						
							|  |  |  |                 raise self.failureException('Only tasks should be yielded ' | 
					
						
							|  |  |  |                                             'from as_completed iterator ' | 
					
						
							|  |  |  |                                             'as-is.') | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_as_completed_with_timeout(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |             yield | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0.1 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_iterator(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             values = [] | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             for f in asyncio.as_completed([a, b], timeout=0.12): | 
					
						
							| 
									
										
										
										
											2014-02-12 17:58:19 -08:00
										 |  |  |                 if values: | 
					
						
							|  |  |  |                     loop.advance_time(0.02) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 try: | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                     v = await f | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                     values.append((1, v)) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |                 except asyncio.TimeoutError as exc: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                     values.append((2, exc)) | 
					
						
							|  |  |  |             return values | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_async_iterator(): | 
					
						
							|  |  |  |             values = [] | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 async for f in asyncio.as_completed([a, b], timeout=0.12): | 
					
						
							|  |  |  |                     v = await f | 
					
						
							|  |  |  |                     values.append((1, v)) | 
					
						
							|  |  |  |                     loop.advance_time(0.02) | 
					
						
							|  |  |  |             except asyncio.TimeoutError as exc: | 
					
						
							|  |  |  |                 values.append((2, exc)) | 
					
						
							|  |  |  |             return values | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         for foo in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |             with self.subTest(method=foo.__name__): | 
					
						
							|  |  |  |                 loop = self.new_test_loop(gen) | 
					
						
							|  |  |  |                 a = loop.create_task(asyncio.sleep(0.1, 'a')) | 
					
						
							|  |  |  |                 b = loop.create_task(asyncio.sleep(0.15, 'b')) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 res = loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							|  |  |  |                 self.assertEqual(len(res), 2, res) | 
					
						
							|  |  |  |                 self.assertEqual(res[0], (1, 'a')) | 
					
						
							|  |  |  |                 self.assertEqual(res[1][0], 2) | 
					
						
							|  |  |  |                 self.assertIsInstance(res[1][1], asyncio.TimeoutError) | 
					
						
							|  |  |  |                 self.assertAlmostEqual(0.12, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # move forward to close generator | 
					
						
							|  |  |  |                 loop.advance_time(10) | 
					
						
							|  |  |  |                 loop.run_until_complete(asyncio.wait([a, b])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-12 17:58:19 -08:00
										 |  |  |     def test_as_completed_with_unused_timeout(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             yield | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0.01 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_iterator(): | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             for f in asyncio.as_completed([a], timeout=1): | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                 v = await f | 
					
						
							| 
									
										
										
										
											2014-02-12 17:58:19 -08:00
										 |  |  |                 self.assertEqual(v, 'a') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_async_iterator(): | 
					
						
							|  |  |  |             async for f in asyncio.as_completed([a], timeout=1): | 
					
						
							|  |  |  |                 v = await f | 
					
						
							|  |  |  |                 self.assertEqual(v, 'a') | 
					
						
							| 
									
										
										
										
											2014-02-12 17:58:19 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         for foo in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |             with self.subTest(method=foo.__name__): | 
					
						
							|  |  |  |                 a = asyncio.sleep(0.01, 'a') | 
					
						
							|  |  |  |                 loop = self.new_test_loop(gen) | 
					
						
							|  |  |  |                 loop.run_until_complete(self.new_task(loop, foo())) | 
					
						
							|  |  |  |                 loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_as_completed_resume_iterator(self): | 
					
						
							|  |  |  |         # Test that as_completed returns an iterator that can be resumed | 
					
						
							|  |  |  |         # the next time iteration is performed (i.e. if __iter__ is called | 
					
						
							|  |  |  |         # again) | 
					
						
							|  |  |  |         async def try_iterator(awaitables): | 
					
						
							|  |  |  |             iterations = 0 | 
					
						
							|  |  |  |             iterator = asyncio.as_completed(awaitables) | 
					
						
							|  |  |  |             collected = [] | 
					
						
							|  |  |  |             for f in iterator: | 
					
						
							|  |  |  |                 collected.append(await f) | 
					
						
							|  |  |  |                 iterations += 1 | 
					
						
							|  |  |  |                 if iterations == 2: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |             self.assertEqual(len(collected), 2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Resume same iterator: | 
					
						
							|  |  |  |             for f in iterator: | 
					
						
							|  |  |  |                 collected.append(await f) | 
					
						
							|  |  |  |             return collected | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def try_async_iterator(awaitables): | 
					
						
							|  |  |  |             iterations = 0 | 
					
						
							|  |  |  |             iterator = asyncio.as_completed(awaitables) | 
					
						
							|  |  |  |             collected = [] | 
					
						
							|  |  |  |             async for f in iterator: | 
					
						
							|  |  |  |                 collected.append(await f) | 
					
						
							|  |  |  |                 iterations += 1 | 
					
						
							|  |  |  |                 if iterations == 2: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |             self.assertEqual(len(collected), 2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Resume same iterator: | 
					
						
							|  |  |  |             async for f in iterator: | 
					
						
							|  |  |  |                 collected.append(await f) | 
					
						
							|  |  |  |             return collected | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(i): | 
					
						
							|  |  |  |             return i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with contextlib.closing(asyncio.new_event_loop()) as loop: | 
					
						
							|  |  |  |             for foo in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |                 with self.subTest(method=foo.__name__): | 
					
						
							|  |  |  |                     results = loop.run_until_complete( | 
					
						
							|  |  |  |                         foo((coro(0), coro(1), coro(2), coro(3))) | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     self.assertCountEqual(results, (0, 1, 2, 3)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |     def test_as_completed_reverse_wait(self): | 
					
						
							|  |  |  |         # Tests the plain iterator style of as_completed iteration to | 
					
						
							|  |  |  |         # ensure that the first future awaited resolves to the first | 
					
						
							|  |  |  |         # completed awaitable from the set we passed in, even if it wasn't | 
					
						
							|  |  |  |         # the first future generated by as_completed. | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         def gen(): | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  |             yield 0.05 | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         a = asyncio.sleep(0.05, 'a') | 
					
						
							|  |  |  |         b = asyncio.sleep(0.10, 'b') | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fs = {a, b} | 
					
						
							| 
									
										
										
										
											2019-09-10 07:55:07 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         async def test(): | 
					
						
							|  |  |  |             futs = list(asyncio.as_completed(fs)) | 
					
						
							|  |  |  |             self.assertEqual(len(futs), 2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             x = await futs[1] | 
					
						
							|  |  |  |             self.assertEqual(x, 'a') | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.05, loop.time()) | 
					
						
							|  |  |  |             loop.advance_time(0.05) | 
					
						
							|  |  |  |             y = await futs[0] | 
					
						
							|  |  |  |             self.assertEqual(y, 'b') | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.10, loop.time()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         loop.run_until_complete(test()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_as_completed_concurrent(self): | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         # Ensure that more than one future or coroutine yielded from | 
					
						
							|  |  |  |         # as_completed can be awaited concurrently. | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.05, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.05, when) | 
					
						
							|  |  |  |             yield 0.05 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_iterator(fs): | 
					
						
							|  |  |  |             return list(asyncio.as_completed(fs)) | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_async_iterator(fs): | 
					
						
							|  |  |  |             return [f async for f in asyncio.as_completed(fs)] | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         for runner in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |             with self.subTest(method=runner.__name__): | 
					
						
							|  |  |  |                 a = asyncio.sleep(0.05, 'a') | 
					
						
							|  |  |  |                 b = asyncio.sleep(0.05, 'b') | 
					
						
							|  |  |  |                 fs = {a, b} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def test(): | 
					
						
							|  |  |  |                     futs = await runner(fs) | 
					
						
							|  |  |  |                     self.assertEqual(len(futs), 2) | 
					
						
							|  |  |  |                     done, pending = await asyncio.wait( | 
					
						
							|  |  |  |                         [asyncio.ensure_future(fut) for fut in futs] | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     self.assertEqual(set(f.result() for f in done), {'a', 'b'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 loop = self.new_test_loop(gen) | 
					
						
							|  |  |  |                 loop.run_until_complete(test()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |     def test_as_completed_duplicate_coroutines(self): | 
					
						
							| 
									
										
										
										
											2014-02-18 22:56:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def coro(s): | 
					
						
							|  |  |  |             return s | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_iterator(): | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |             result = [] | 
					
						
							|  |  |  |             c = coro('ham') | 
					
						
							|  |  |  |             for f in asyncio.as_completed([c, c, coro('spam')]): | 
					
						
							|  |  |  |                 result.append(await f) | 
					
						
							|  |  |  |             return result | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         async def try_async_iterator(): | 
					
						
							|  |  |  |             result = [] | 
					
						
							|  |  |  |             c = coro('ham') | 
					
						
							|  |  |  |             async for f in asyncio.as_completed([c, c, coro('spam')]): | 
					
						
							|  |  |  |                 result.append(await f) | 
					
						
							|  |  |  |             return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for runner in try_iterator, try_async_iterator: | 
					
						
							|  |  |  |             with self.subTest(method=runner.__name__): | 
					
						
							|  |  |  |                 fut = self.new_task(self.loop, runner()) | 
					
						
							|  |  |  |                 self.loop.run_until_complete(fut) | 
					
						
							|  |  |  |                 result = fut.result() | 
					
						
							|  |  |  |                 self.assertEqual(set(result), {'ham', 'spam'}) | 
					
						
							|  |  |  |                 self.assertEqual(len(result), 2) | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |     def test_as_completed_coroutine_without_loop(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         a = coro() | 
					
						
							|  |  |  |         self.addCleanup(a.close) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no current event loop'): | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |             futs = asyncio.as_completed([a]) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |             list(futs) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_as_completed_coroutine_use_running_loop(self): | 
					
						
							|  |  |  |         loop = self.new_test_loop() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def test(): | 
					
						
							|  |  |  |             futs = list(asyncio.as_completed([coro()])) | 
					
						
							|  |  |  |             self.assertEqual(len(futs), 1) | 
					
						
							|  |  |  |             self.assertEqual(await futs[0], 42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop.run_until_complete(test()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_sleep(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.05, when) | 
					
						
							|  |  |  |             when = yield 0.05 | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             yield 0.05 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def sleeper(dt, arg): | 
					
						
							|  |  |  |             await asyncio.sleep(dt/2) | 
					
						
							|  |  |  |             res = await asyncio.sleep(dt/2, arg) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(loop, sleeper(0.1, 'yeah')) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         loop.run_until_complete(t) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertEqual(t.result(), 'yeah') | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.1, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-13 04:29:02 +08:00
										 |  |  |     def test_sleep_when_delay_is_nan(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             yield | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = self.new_test_loop(gen) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def sleeper(): | 
					
						
							|  |  |  |             await asyncio.sleep(float("nan")) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t = self.new_task(loop, sleeper()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							|  |  |  |             loop.run_until_complete(t) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_sleep_cancel(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         t = self.new_task(loop, asyncio.sleep(10.0, 'yeah')) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         handle = None | 
					
						
							|  |  |  |         orig_call_later = loop.call_later | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-01 12:38:51 +02:00
										 |  |  |         def call_later(delay, callback, *args): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal handle | 
					
						
							| 
									
										
										
										
											2014-07-01 12:38:51 +02:00
										 |  |  |             handle = orig_call_later(delay, callback, *args) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return handle | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop.call_later = call_later | 
					
						
							|  |  |  |         test_utils.run_briefly(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertFalse(handle._cancelled) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         t.cancel() | 
					
						
							|  |  |  |         test_utils.run_briefly(loop) | 
					
						
							|  |  |  |         self.assertTrue(handle._cancelled) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_task_cancel_sleeping_task(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(0.1, when) | 
					
						
							|  |  |  |             when = yield 0 | 
					
						
							|  |  |  |             self.assertAlmostEqual(5000, when) | 
					
						
							|  |  |  |             yield 0.1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def sleep(dt): | 
					
						
							|  |  |  |             await asyncio.sleep(dt) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def doit(): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |             sleeper = self.new_task(loop, sleep(5000)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             loop.call_later(0.1, sleeper.cancel) | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |                 await sleeper | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 return 'cancelled' | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return 'slept in' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         doer = doit() | 
					
						
							|  |  |  |         self.assertEqual(loop.run_until_complete(doer), 'cancelled') | 
					
						
							|  |  |  |         self.assertAlmostEqual(0.1, loop.time()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_task_cancel_waiter_future(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							|  |  |  |             await fut | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(task._fut_waiter, fut) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task.cancel() | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertRaises( | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             asyncio.CancelledError, self.loop.run_until_complete, task) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIsNone(task._fut_waiter) | 
					
						
							|  |  |  |         self.assertTrue(fut.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     def test_task_set_methods(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmuch(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 'ko' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         gen = notmuch() | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task = self.new_task(self.loop, gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'not support set_result'): | 
					
						
							|  |  |  |             task.set_result('ok') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaisesRegex(RuntimeError, 'not support set_exception'): | 
					
						
							|  |  |  |             task.set_exception(ValueError()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |             self.loop.run_until_complete(task), | 
					
						
							|  |  |  |             'ko') | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_step_result_future(self): | 
					
						
							|  |  |  |         # If coroutine returns future, task waits on this future. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         class Fut(asyncio.Future): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             def __init__(self, *args, **kwds): | 
					
						
							|  |  |  |                 self.cb_added = False | 
					
						
							|  |  |  |                 super().__init__(*args, **kwds) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |             def add_done_callback(self, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 self.cb_added = True | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |                 super().add_done_callback(*args, **kwargs) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         fut = Fut(loop=self.loop) | 
					
						
							|  |  |  |         result = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def wait_for_future(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal result | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             result = await fut | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t = self.new_task(self.loop, wait_for_future()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.cb_added) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         res = object() | 
					
						
							|  |  |  |         fut.set_result(res) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(res, result) | 
					
						
							|  |  |  |         self.assertTrue(t.done()) | 
					
						
							|  |  |  |         self.assertIsNone(t.result()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_baseexception_during_cancel(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def gen(): | 
					
						
							|  |  |  |             when = yield | 
					
						
							|  |  |  |             self.assertAlmostEqual(10.0, when) | 
					
						
							|  |  |  |             yield 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         loop = self.new_test_loop(gen) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def sleeper(): | 
					
						
							|  |  |  |             await asyncio.sleep(10) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-27 14:45:12 +02:00
										 |  |  |         base_exc = SystemExit() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def notmutch(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |                 await sleeper() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 raise base_exc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task = self.new_task(loop, notmutch()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task.cancel() | 
					
						
							|  |  |  |         self.assertFalse(task.done()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-27 14:45:12 +02:00
										 |  |  |         self.assertRaises(SystemExit, test_utils.run_briefly, loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertTrue(task.done()) | 
					
						
							|  |  |  |         self.assertFalse(task.cancelled()) | 
					
						
							|  |  |  |         self.assertIs(task.exception(), base_exc) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 00:35:51 +08:00
										 |  |  |     @ignore_warnings(category=DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_iscoroutinefunction(self): | 
					
						
							|  |  |  |         def fn(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertFalse(asyncio.iscoroutinefunction(fn)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def fn1(): | 
					
						
							|  |  |  |             yield | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertFalse(asyncio.iscoroutinefunction(fn1)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def fn2(): | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertTrue(asyncio.iscoroutinefunction(fn2)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-15 15:20:34 -05:00
										 |  |  |         self.assertFalse(asyncio.iscoroutinefunction(mock.Mock())) | 
					
						
							| 
									
										
										
										
											2022-06-30 19:08:38 +02:00
										 |  |  |         self.assertTrue(asyncio.iscoroutinefunction(mock.AsyncMock())) | 
					
						
							| 
									
										
										
										
											2016-11-15 15:20:34 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 00:35:51 +08:00
										 |  |  |     @ignore_warnings(category=DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_coroutine_non_gen_function(self): | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def func(): | 
					
						
							|  |  |  |             return 'test' | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertTrue(asyncio.iscoroutinefunction(func)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         coro = func() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertTrue(asyncio.iscoroutine(coro)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(coro) | 
					
						
							|  |  |  |         self.assertEqual(res, 'test') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_coroutine_non_gen_function_return_future(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def func(): | 
					
						
							|  |  |  |             return fut | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             fut.set_result('test') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         t1 = self.new_task(self.loop, func()) | 
					
						
							|  |  |  |         t2 = self.new_task(self.loop, coro()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         res = self.loop.run_until_complete(t1) | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         self.assertEqual(res, fut) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIsNone(t2.result()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |     def test_current_task(self): | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop=self.loop)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(loop): | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             self.assertIs(asyncio.current_task(), task) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |             self.assertIs(asyncio.current_task(None), task) | 
					
						
							|  |  |  |             self.assertIs(asyncio.current_task(), task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro(self.loop)) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop=self.loop)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_current_task_with_interleaving_tasks(self): | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop=self.loop)) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut1 = self.new_future(self.loop) | 
					
						
							|  |  |  |         fut2 = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def coro1(loop): | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             self.assertTrue(asyncio.current_task() is task1) | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             await fut1 | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             self.assertTrue(asyncio.current_task() is task1) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  |             fut2.set_result(True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def coro2(loop): | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             self.assertTrue(asyncio.current_task() is task2) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  |             fut1.set_result(True) | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             await fut2 | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             self.assertTrue(asyncio.current_task() is task2) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task1 = self.new_task(self.loop, coro1(self.loop)) | 
					
						
							|  |  |  |         task2 = self.new_task(self.loop, coro2(self.loop)) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |         self.loop.run_until_complete(asyncio.wait((task1, task2))) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |         self.assertIsNone(asyncio.current_task(loop=self.loop)) | 
					
						
							| 
									
										
										
										
											2013-12-06 12:57:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     # Some thorough tests for cancellation propagation through | 
					
						
							|  |  |  |     # coroutines, tasks and wait(). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_yield_future_passes_cancel(self): | 
					
						
							|  |  |  |         # Cancelling outer() cancels inner() cancels waiter. | 
					
						
							|  |  |  |         proof = 0 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         waiter = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def inner(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                 await waiter | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 proof += 1 | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.fail('got past sleep() in inner()') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def outer(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |                 await inner() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             except asyncio.CancelledError: | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                 proof += 100  # Expect this path. | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 proof += 10 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(outer(), loop=self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         f.cancel() | 
					
						
							|  |  |  |         self.loop.run_until_complete(f) | 
					
						
							|  |  |  |         self.assertEqual(proof, 101) | 
					
						
							|  |  |  |         self.assertTrue(waiter.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_yield_wait_does_not_shield_cancel(self): | 
					
						
							|  |  |  |         # Cancelling outer() makes wait() return early, leaves inner() | 
					
						
							|  |  |  |         # running. | 
					
						
							|  |  |  |         proof = 0 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         waiter = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def inner(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             await waiter | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def outer(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							| 
									
										
										
										
											2019-12-30 06:50:19 -05:00
										 |  |  |             with self.assertWarns(DeprecationWarning): | 
					
						
							| 
									
										
										
										
											2022-03-17 22:51:40 +02:00
										 |  |  |                 d, p = await asyncio.wait([asyncio.create_task(inner())]) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 100 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(outer(), loop=self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         f.cancel() | 
					
						
							|  |  |  |         self.assertRaises( | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             asyncio.CancelledError, self.loop.run_until_complete, f) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         waiter.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertEqual(proof, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_result(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         inner = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         outer = asyncio.shield(inner) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         inner.set_result(42) | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(outer) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_exception(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         inner = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         outer = asyncio.shield(inner) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         exc = RuntimeError('expected') | 
					
						
							|  |  |  |         inner.set_exception(exc) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertIs(outer.exception(), exc) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-07 20:58:24 +02:00
										 |  |  |     def test_shield_cancel_inner(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         inner = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         outer = asyncio.shield(inner) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         inner.cancel() | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertTrue(outer.cancelled()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-07 20:58:24 +02:00
										 |  |  |     def test_shield_cancel_outer(self): | 
					
						
							|  |  |  |         inner = self.new_future(self.loop) | 
					
						
							|  |  |  |         outer = asyncio.shield(inner) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         outer.cancel() | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertTrue(outer.cancelled()) | 
					
						
							|  |  |  |         self.assertEqual(0, 0 if outer._callbacks is None else len(outer._callbacks)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_shield_shortcut(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.set_result(42) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         res = self.loop.run_until_complete(asyncio.shield(fut)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_effect(self): | 
					
						
							|  |  |  |         # Cancelling outer() does not affect inner(). | 
					
						
							|  |  |  |         proof = 0 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         waiter = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def inner(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |             await waiter | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  |         async def outer(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							| 
									
										
										
										
											2019-09-12 15:40:40 +03:00
										 |  |  |             await asyncio.shield(inner()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 100 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(outer(), loop=self.loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         f.cancel() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.loop.run_until_complete(f) | 
					
						
							|  |  |  |         waiter.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertEqual(proof, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_gather(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         child1 = self.new_future(self.loop) | 
					
						
							|  |  |  |         child2 = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2019-09-12 15:40:40 +03:00
										 |  |  |         parent = asyncio.gather(child1, child2) | 
					
						
							|  |  |  |         outer = asyncio.shield(parent) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         outer.cancel() | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertTrue(outer.cancelled()) | 
					
						
							|  |  |  |         child1.set_result(1) | 
					
						
							|  |  |  |         child2.set_result(2) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         self.assertEqual(parent.result(), [1, 2]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_gather_shield(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         child1 = self.new_future(self.loop) | 
					
						
							|  |  |  |         child2 = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2019-09-12 15:40:40 +03:00
										 |  |  |         inner1 = asyncio.shield(child1) | 
					
						
							|  |  |  |         inner2 = asyncio.shield(child2) | 
					
						
							|  |  |  |         parent = asyncio.gather(inner1, inner2) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         parent.cancel() | 
					
						
							|  |  |  |         # This should cancel inner1 and inner2 but bot child1 and child2. | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertIsInstance(parent.exception(), asyncio.CancelledError) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertTrue(inner1.cancelled()) | 
					
						
							|  |  |  |         self.assertTrue(inner2.cancelled()) | 
					
						
							|  |  |  |         child1.set_result(1) | 
					
						
							|  |  |  |         child2.set_result(2) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |     def test_shield_coroutine_without_loop(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         inner = coro() | 
					
						
							|  |  |  |         self.addCleanup(inner.close) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no current event loop'): | 
					
						
							|  |  |  |             asyncio.shield(inner) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_coroutine_use_running_loop(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def test(): | 
					
						
							|  |  |  |             return asyncio.shield(coro()) | 
					
						
							|  |  |  |         outer = self.loop.run_until_complete(test()) | 
					
						
							|  |  |  |         self.assertEqual(outer._loop, self.loop) | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(outer) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_shield_coroutine_use_global_loop(self): | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         # Deprecated in 3.10, undeprecated in 3.12 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 42 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         asyncio.set_event_loop(self.loop) | 
					
						
							|  |  |  |         self.addCleanup(asyncio.set_event_loop, None) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         outer = asyncio.shield(coro()) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.assertEqual(outer._loop, self.loop) | 
					
						
							|  |  |  |         res = self.loop.run_until_complete(outer) | 
					
						
							|  |  |  |         self.assertEqual(res, 42) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  |     def test_as_completed_invalid_args(self): | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         # as_completed() expects a list of futures, not a future instance | 
					
						
							|  |  |  |         # TypeError should be raised either on iterator construction or first | 
					
						
							|  |  |  |         # iteration | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Plain iterator | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             iterator = asyncio.as_completed(fut) | 
					
						
							|  |  |  |             next(iterator) | 
					
						
							|  |  |  |         coro = coroutine_function() | 
					
						
							|  |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             iterator = asyncio.as_completed(coro) | 
					
						
							|  |  |  |             next(iterator) | 
					
						
							|  |  |  |         coro.close() | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         # Async iterator | 
					
						
							|  |  |  |         async def try_async_iterator(aw): | 
					
						
							|  |  |  |             async for f in asyncio.as_completed(aw): | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fut = self.new_future(self.loop) | 
					
						
							|  |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             self.loop.run_until_complete(try_async_iterator(fut)) | 
					
						
							| 
									
										
										
										
											2014-06-28 01:19:11 +02:00
										 |  |  |         coro = coroutine_function() | 
					
						
							| 
									
										
										
										
											2024-04-01 12:07:29 -05:00
										 |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             self.loop.run_until_complete(try_async_iterator(coro)) | 
					
						
							| 
									
										
										
										
											2014-06-28 01:19:11 +02:00
										 |  |  |         coro.close() | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_wait_invalid_args(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(self.loop) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # wait() expects a list of futures, not a future instance | 
					
						
							|  |  |  |         self.assertRaises(TypeError, self.loop.run_until_complete, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait(fut)) | 
					
						
							| 
									
										
										
										
											2014-06-28 01:19:11 +02:00
										 |  |  |         coro = coroutine_function() | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  |         self.assertRaises(TypeError, self.loop.run_until_complete, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait(coro)) | 
					
						
							| 
									
										
										
										
											2014-06-28 01:19:11 +02:00
										 |  |  |         coro.close() | 
					
						
							| 
									
										
										
										
											2014-02-11 11:54:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # wait() expects at least a future | 
					
						
							|  |  |  |         self.assertRaises(ValueError, self.loop.run_until_complete, | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             asyncio.wait([])) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  |     def test_log_destroyed_pending_task(self): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         Task = self.__class__.Task | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def kill_me(loop): | 
					
						
							|  |  |  |             future = self.new_future(loop) | 
					
						
							|  |  |  |             await future | 
					
						
							|  |  |  |             # at this point, the only reference to kill_me() task is | 
					
						
							|  |  |  |             # the Task._wakeup() method in future._callbacks | 
					
						
							|  |  |  |             raise Exception("code never reached") | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         mock_handler = mock.Mock() | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |         self.loop.set_debug(True) | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  |         self.loop.set_exception_handler(mock_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # schedule the task | 
					
						
							|  |  |  |         coro = kill_me(self.loop) | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         task = asyncio.ensure_future(coro, loop=self.loop) | 
					
						
							| 
									
										
										
										
											2017-03-02 22:16:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop=self.loop), {task}) | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-02 05:41:21 +02:00
										 |  |  |         asyncio.set_event_loop(None) | 
					
						
							| 
									
										
										
										
											2017-03-02 22:16:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  |         # execute the task so it waits for future | 
					
						
							|  |  |  |         self.loop._run_once() | 
					
						
							|  |  |  |         self.assertEqual(len(self.loop._ready), 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         coro = None | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |         source_traceback = task._source_traceback | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  |         task = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # no more reference to kill_me() task: the task is destroyed by the GC | 
					
						
							|  |  |  |         support.gc_collect() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop=self.loop), set()) | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         mock_handler.assert_called_with(self.loop, { | 
					
						
							|  |  |  |             'message': 'Task was destroyed but it is pending!', | 
					
						
							|  |  |  |             'task': mock.ANY, | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |             'source_traceback': source_traceback, | 
					
						
							| 
									
										
										
										
											2014-06-24 22:37:53 +02:00
										 |  |  |         }) | 
					
						
							|  |  |  |         mock_handler.reset_mock() | 
					
						
							| 
									
										
										
										
											2014-04-27 10:44:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-11 13:49:18 +00:00
										 |  |  |     @mock.patch('asyncio.base_events.logger') | 
					
						
							|  |  |  |     def test_tb_logger_not_called_after_cancel(self, m_log): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2017-06-11 13:49:18 +00:00
										 |  |  |             raise TypeError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def runner(): | 
					
						
							| 
									
										
										
										
											2017-06-11 13:49:18 +00:00
										 |  |  |             task = self.new_task(loop, coro()) | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             await asyncio.sleep(0.05) | 
					
						
							| 
									
										
										
										
											2017-06-11 13:49:18 +00:00
										 |  |  |             task.cancel() | 
					
						
							|  |  |  |             task = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop.run_until_complete(runner()) | 
					
						
							|  |  |  |         self.assertFalse(m_log.error.called) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |     def test_task_source_traceback(self): | 
					
						
							|  |  |  |         self.loop.set_debug(True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         task = self.new_task(self.loop, coroutine_function()) | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |         lineno = sys._getframe().f_lineno - 1 | 
					
						
							|  |  |  |         self.assertIsInstance(task._source_traceback, list) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         self.assertEqual(task._source_traceback[-2][:3], | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  |                          (__file__, | 
					
						
							|  |  |  |                           lineno, | 
					
						
							|  |  |  |                           'test_task_source_traceback')) | 
					
						
							| 
									
										
										
										
											2014-06-30 14:39:11 +02:00
										 |  |  |         self.loop.run_until_complete(task) | 
					
						
							| 
									
										
										
										
											2014-06-27 13:52:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 17:20:02 -04:00
										 |  |  |     def test_cancel_gather_1(self): | 
					
						
							| 
									
										
										
										
											2016-10-21 17:22:17 -04:00
										 |  |  |         """Ensure that a gathering future refuses to be cancelled once all
 | 
					
						
							|  |  |  |         children are done"""
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.addCleanup(loop.close) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         fut = self.new_future(loop) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         async def create(): | 
					
						
							|  |  |  |             # The indirection fut->child_coro is needed since otherwise the | 
					
						
							|  |  |  |             # gathering task is done at the same time as the child future | 
					
						
							| 
									
										
										
										
											2022-12-09 09:22:18 +05:30
										 |  |  |             async def child_coro(): | 
					
						
							|  |  |  |                 return await fut | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |             gather_future = asyncio.gather(child_coro()) | 
					
						
							|  |  |  |             return asyncio.ensure_future(gather_future) | 
					
						
							|  |  |  |         gather_task = loop.run_until_complete(create()) | 
					
						
							| 
									
										
										
										
											2016-10-21 17:22:17 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         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]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 17:20:02 -04:00
										 |  |  |     def test_cancel_gather_2(self): | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |         cases = [ | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |             ((), ()), | 
					
						
							|  |  |  |             ((None,), ()), | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |             (('my message',), ('my message',)), | 
					
						
							|  |  |  |             # Non-string values should roundtrip. | 
					
						
							|  |  |  |             ((5,), (5,)), | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         for cancel_args, expected_args in cases: | 
					
						
							|  |  |  |             with self.subTest(cancel_args=cancel_args): | 
					
						
							|  |  |  |                 loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |                 self.addCleanup(loop.close) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def test(): | 
					
						
							|  |  |  |                     time = 0 | 
					
						
							|  |  |  |                     while True: | 
					
						
							|  |  |  |                         time += 0.05 | 
					
						
							|  |  |  |                         await asyncio.gather(asyncio.sleep(0.05), | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |                                              return_exceptions=True) | 
					
						
							| 
									
										
										
										
											2020-05-15 16:55:50 -07:00
										 |  |  |                         if time > 1: | 
					
						
							|  |  |  |                             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 async def main(): | 
					
						
							|  |  |  |                     qwe = self.new_task(loop, test()) | 
					
						
							|  |  |  |                     await asyncio.sleep(0.2) | 
					
						
							| 
									
										
										
										
											2022-10-06 17:30:27 -07:00
										 |  |  |                     qwe.cancel(*cancel_args) | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |                     await qwe | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     loop.run_until_complete(main()) | 
					
						
							|  |  |  |                 except asyncio.CancelledError as exc: | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                     self.assertEqual(exc.args, expected_args) | 
					
						
							|  |  |  |                     actual = get_innermost_context(exc) | 
					
						
							|  |  |  |                     self.assertEqual( | 
					
						
							|  |  |  |                         actual, | 
					
						
							|  |  |  |                         (asyncio.CancelledError, expected_args, 0), | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2020-05-17 22:47:31 -07:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2022-02-21 22:59:04 +02:00
										 |  |  |                     self.fail( | 
					
						
							|  |  |  |                         'gather() does not propagate CancelledError ' | 
					
						
							|  |  |  |                         'raised by inner task to the gather() caller.' | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2018-05-29 17:20:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 11:37:47 -05:00
										 |  |  |     def test_exception_traceback(self): | 
					
						
							|  |  |  |         # See http://bugs.python.org/issue28843 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2016-12-01 11:37:47 -05:00
										 |  |  |             1 / 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def main(): | 
					
						
							| 
									
										
										
										
											2016-12-01 11:37:47 -05:00
										 |  |  |             task = self.new_task(self.loop, foo()) | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             await asyncio.sleep(0)  # skip one loop iteration | 
					
						
							| 
									
										
										
										
											2016-12-01 11:37:47 -05:00
										 |  |  |             self.assertIsNotNone(task.exception().__traceback__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete(main()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     @mock.patch('asyncio.base_events.logger') | 
					
						
							|  |  |  |     def test_error_in_call_soon(self, m_log): | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |         def call_soon(callback, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |             raise ValueError | 
					
						
							|  |  |  |         self.loop.call_soon = call_soon | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def coro(): | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertFalse(m_log.error.called) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							| 
									
										
										
										
											2018-06-07 01:30:38 +02:00
										 |  |  |             gen = coro() | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 self.new_task(self.loop, gen) | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 gen.close() | 
					
						
							| 
									
										
										
										
											2021-08-29 14:04:40 +03:00
										 |  |  |         gc.collect()  # For PyPy or other GCs. | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertTrue(m_log.error.called) | 
					
						
							|  |  |  |         message = m_log.error.call_args[0][0] | 
					
						
							|  |  |  |         self.assertIn('Task was destroyed but it is pending', message) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(self.loop), set()) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-15 07:04:38 +02:00
										 |  |  |     def test_create_task_with_noncoroutine(self): | 
					
						
							|  |  |  |         with self.assertRaisesRegex(TypeError, | 
					
						
							|  |  |  |                                     "a coroutine was expected, got 123"): | 
					
						
							|  |  |  |             self.new_task(self.loop, 123) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-19 07:18:45 -05:00
										 |  |  |         # test it for the second time to ensure that caching | 
					
						
							|  |  |  |         # in asyncio.iscoroutine() doesn't break things. | 
					
						
							|  |  |  |         with self.assertRaisesRegex(TypeError, | 
					
						
							|  |  |  |                                     "a coroutine was expected, got 123"): | 
					
						
							|  |  |  |             self.new_task(self.loop, 123) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-15 07:04:38 +02:00
										 |  |  |     def test_create_task_with_async_function(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         self.assertIsInstance(task, self.Task) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-19 07:18:45 -05:00
										 |  |  |         # test it for the second time to ensure that caching | 
					
						
							|  |  |  |         # in asyncio.iscoroutine() doesn't break things. | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         self.assertIsInstance(task, self.Task) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_create_task_with_asynclike_function(self): | 
					
						
							|  |  |  |         task = self.new_task(self.loop, CoroLikeObject()) | 
					
						
							|  |  |  |         self.assertIsInstance(task, self.Task) | 
					
						
							|  |  |  |         self.assertEqual(self.loop.run_until_complete(task), 42) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # test it for the second time to ensure that caching | 
					
						
							|  |  |  |         # in asyncio.iscoroutine() doesn't break things. | 
					
						
							|  |  |  |         task = self.new_task(self.loop, CoroLikeObject()) | 
					
						
							|  |  |  |         self.assertIsInstance(task, self.Task) | 
					
						
							|  |  |  |         self.assertEqual(self.loop.run_until_complete(task), 42) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-15 07:04:38 +02:00
										 |  |  |     def test_bare_create_task(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def inner(): | 
					
						
							|  |  |  |             return 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             task = asyncio.create_task(inner()) | 
					
						
							|  |  |  |             self.assertIsInstance(task, self.Task) | 
					
						
							|  |  |  |             ret = await task | 
					
						
							|  |  |  |             self.assertEqual(1, ret) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete(coro()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 00:06:47 +03:00
										 |  |  |     def test_bare_create_named_task(self): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro_noop(): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             task = asyncio.create_task(coro_noop(), name='No-op') | 
					
						
							|  |  |  |             self.assertEqual(task.get_name(), 'No-op') | 
					
						
							|  |  |  |             await task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete(coro()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |     def test_context_1(self): | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar', default='nope') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def sub(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(0.01) | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |             self.assertEqual(cvar.get(), 'nope') | 
					
						
							|  |  |  |             cvar.set('something else') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             self.assertEqual(cvar.get(), 'nope') | 
					
						
							|  |  |  |             subtask = self.new_task(loop, sub()) | 
					
						
							|  |  |  |             cvar.set('yes') | 
					
						
							|  |  |  |             self.assertEqual(cvar.get(), 'yes') | 
					
						
							|  |  |  |             await subtask | 
					
						
							|  |  |  |             self.assertEqual(cvar.get(), 'yes') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, main()) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_context_2(self): | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar', default='nope') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             def fut_on_done(fut): | 
					
						
							|  |  |  |                 # This change must not pollute the context | 
					
						
							|  |  |  |                 # of the "main()" task. | 
					
						
							|  |  |  |                 cvar.set('something else') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.assertEqual(cvar.get(), 'nope') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for j in range(2): | 
					
						
							|  |  |  |                 fut = self.new_future(loop) | 
					
						
							|  |  |  |                 fut.add_done_callback(fut_on_done) | 
					
						
							|  |  |  |                 cvar.set(f'yes{j}') | 
					
						
							|  |  |  |                 loop.call_soon(fut.set_result, None) | 
					
						
							|  |  |  |                 await fut | 
					
						
							|  |  |  |                 self.assertEqual(cvar.get(), f'yes{j}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for i in range(3): | 
					
						
							|  |  |  |                     # Test that task passed its context to add_done_callback: | 
					
						
							|  |  |  |                     cvar.set(f'yes{i}-{j}') | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |                     await asyncio.sleep(0.001) | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |                     self.assertEqual(cvar.get(), f'yes{i}-{j}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, main()) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(cvar.get(), 'nope') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_context_3(self): | 
					
						
							|  |  |  |         # Run 100 Tasks in parallel, each modifying cvar. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar', default=-1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def sub(num): | 
					
						
							|  |  |  |             for i in range(10): | 
					
						
							|  |  |  |                 cvar.set(num + i) | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |                 await asyncio.sleep(random.uniform(0.001, 0.05)) | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |                 self.assertEqual(cvar.get(), num + i) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             tasks = [] | 
					
						
							|  |  |  |             for i in range(100): | 
					
						
							|  |  |  |                 task = loop.create_task(sub(random.randint(0, 10))) | 
					
						
							|  |  |  |                 tasks.append(task) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             await asyncio.gather(*tasks) | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             loop.run_until_complete(main()) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(cvar.get(), -1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-14 13:54:13 +02:00
										 |  |  |     def test_context_4(self): | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(val): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             cvar.set(val) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             ret = [] | 
					
						
							|  |  |  |             ctx = contextvars.copy_context() | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t1 = self.new_task(loop, coro(1), context=ctx) | 
					
						
							|  |  |  |             await t1 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t2 = self.new_task(loop, coro(2), context=ctx) | 
					
						
							|  |  |  |             await t2 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, main()) | 
					
						
							|  |  |  |             ret = loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual([None, 1, 2], ret) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_context_5(self): | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(val): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             cvar.set(val) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             ret = [] | 
					
						
							|  |  |  |             ctx = contextvars.copy_context() | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t1 = asyncio.create_task(coro(1), context=ctx) | 
					
						
							|  |  |  |             await t1 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t2 = asyncio.create_task(coro(2), context=ctx) | 
					
						
							|  |  |  |             await t2 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, main()) | 
					
						
							|  |  |  |             ret = loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual([None, 1, 2], ret) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_context_6(self): | 
					
						
							|  |  |  |         cvar = contextvars.ContextVar('cvar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(val): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             cvar.set(val) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             ret = [] | 
					
						
							|  |  |  |             ctx = contextvars.copy_context() | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t1 = loop.create_task(coro(1), context=ctx) | 
					
						
							|  |  |  |             await t1 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             t2 = loop.create_task(coro(2), context=ctx) | 
					
						
							|  |  |  |             await t2 | 
					
						
							|  |  |  |             ret.append(ctx.get(cvar)) | 
					
						
							|  |  |  |             return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = loop.create_task(main()) | 
					
						
							|  |  |  |             ret = loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual([None, 1, 2], ret) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-30 18:30:09 +03:00
										 |  |  |     def test_get_coro(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         coro = coroutine_function() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, coro) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |             self.assertIs(task.get_coro(), coro) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-04 23:49:10 -07:00
										 |  |  |     def test_get_context(self): | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         coro = coroutine_function() | 
					
						
							|  |  |  |         context = contextvars.copy_context() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             task = self.new_task(loop, coro, context=context) | 
					
						
							|  |  |  |             loop.run_until_complete(task) | 
					
						
							|  |  |  |             self.assertIs(task.get_context(), context) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-31 03:47:57 -04:00
										 |  |  |     def test_proper_refcounts(self): | 
					
						
							|  |  |  |         # see: https://github.com/python/cpython/issues/126083 | 
					
						
							|  |  |  |         class Break: | 
					
						
							|  |  |  |             def __str__(self): | 
					
						
							|  |  |  |                 raise RuntimeError("break") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         obj = object() | 
					
						
							|  |  |  |         initial_refcount = sys.getrefcount(obj) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         coro = coroutine_function() | 
					
						
							|  |  |  |         loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         task = asyncio.Task.__new__(asyncio.Task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for _ in range(5): | 
					
						
							|  |  |  |             with self.assertRaisesRegex(RuntimeError, 'break'): | 
					
						
							|  |  |  |                 task.__init__(coro, loop=loop, context=obj, name=Break()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         coro.close() | 
					
						
							|  |  |  |         del task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(sys.getrefcount(obj), initial_refcount) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | def add_subclass_tests(cls): | 
					
						
							|  |  |  |     BaseTask = cls.Task | 
					
						
							|  |  |  |     BaseFuture = cls.Future | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if BaseTask is None or BaseFuture is None: | 
					
						
							|  |  |  |         return cls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class CommonFuture: | 
					
						
							|  |  |  |         def __init__(self, *args, **kwargs): | 
					
						
							|  |  |  |             self.calls = collections.defaultdict(lambda: 0) | 
					
						
							|  |  |  |             super().__init__(*args, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |         def add_done_callback(self, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |             self.calls['add_done_callback'] += 1 | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |             return super().add_done_callback(*args, **kwargs) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     class Task(CommonFuture, BaseTask): | 
					
						
							| 
									
										
										
										
											2022-08-04 15:51:38 +02:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     class Future(CommonFuture, BaseFuture): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_subclasses_ctask_cfuture(self): | 
					
						
							|  |  |  |         fut = self.Future(loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def func(): | 
					
						
							|  |  |  |             self.loop.call_soon(lambda: fut.set_result('spam')) | 
					
						
							|  |  |  |             return await fut | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = self.Task(func(), loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         result = self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(result, 'spam') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |             dict(task.calls), | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |             {'add_done_callback': 1}) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |             dict(fut.calls), | 
					
						
							| 
									
										
										
										
											2018-01-24 11:31:01 -05:00
										 |  |  |             {'add_done_callback': 1}) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Add patched Task & Future back to the test case | 
					
						
							|  |  |  |     cls.Task = Task | 
					
						
							|  |  |  |     cls.Future = Future | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     cls.all_tasks = tasks.all_tasks | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Add an extra unit-test | 
					
						
							|  |  |  |     cls.test_subclasses_ctask_cfuture = test_subclasses_ctask_cfuture | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Disable the "test_task_source_traceback" test | 
					
						
							|  |  |  |     # (the test is hardcoded for a particular call stack, which | 
					
						
							|  |  |  |     # is slightly different for Task subclasses) | 
					
						
							|  |  |  |     cls.test_task_source_traceback = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return cls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | class SetMethodsTest: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_set_result_causes_invalid_state(self): | 
					
						
							|  |  |  |         Future = type(self).Future | 
					
						
							|  |  |  |         self.loop.call_exception_handler = exc_handler = mock.Mock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(0.1) | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |             return 10 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 01:09:31 -08:00
										 |  |  |         coro = foo() | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro) | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |         Future.set_result(task, 'spam') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |             self.loop.run_until_complete(task), | 
					
						
							|  |  |  |             'spam') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exc_handler.assert_called_once() | 
					
						
							|  |  |  |         exc = exc_handler.call_args[0][0]['exception'] | 
					
						
							|  |  |  |         with self.assertRaisesRegex(asyncio.InvalidStateError, | 
					
						
							|  |  |  |                                     r'step\(\): already done'): | 
					
						
							|  |  |  |             raise exc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 01:09:31 -08:00
										 |  |  |         coro.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     def test_set_exception_causes_invalid_state(self): | 
					
						
							|  |  |  |         class MyExc(Exception): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Future = type(self).Future | 
					
						
							|  |  |  |         self.loop.call_exception_handler = exc_handler = mock.Mock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def foo(): | 
					
						
							| 
									
										
										
										
											2018-10-02 13:53:06 -04:00
										 |  |  |             await asyncio.sleep(0.1) | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |             return 10 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 01:09:31 -08:00
										 |  |  |         coro = foo() | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro) | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |         Future.set_exception(task, MyExc()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with self.assertRaises(MyExc): | 
					
						
							|  |  |  |             self.loop.run_until_complete(task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exc_handler.assert_called_once() | 
					
						
							|  |  |  |         exc = exc_handler.call_args[0][0]['exception'] | 
					
						
							|  |  |  |         with self.assertRaisesRegex(asyncio.InvalidStateError, | 
					
						
							|  |  |  |                                     r'step\(\): already done'): | 
					
						
							|  |  |  |             raise exc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-23 01:09:31 -08:00
										 |  |  |         coro.close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | @unittest.skipUnless(hasattr(futures, '_CFuture') and | 
					
						
							|  |  |  |                      hasattr(tasks, '_CTask'), | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |                      'requires the C _asyncio module') | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | class CTask_CFuture_Tests(BaseTaskTests, SetMethodsTest, | 
					
						
							|  |  |  |                           test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     Task = getattr(tasks, '_CTask', None) | 
					
						
							|  |  |  |     Future = getattr(futures, '_CFuture', None) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = getattr(tasks, '_c_all_tasks', None) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 12:28:33 +02:00
										 |  |  |     @support.refcount_test | 
					
						
							|  |  |  |     def test_refleaks_in_task___init__(self): | 
					
						
							|  |  |  |         gettotalrefcount = support.get_attribute(sys, 'gettotalrefcount') | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2018-02-13 12:28:33 +02:00
										 |  |  |             pass | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         refs_before = gettotalrefcount() | 
					
						
							|  |  |  |         for i in range(100): | 
					
						
							|  |  |  |             task.__init__(coro(), loop=self.loop) | 
					
						
							|  |  |  |             self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         self.assertAlmostEqual(gettotalrefcount() - refs_before, 0, delta=10) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-17 07:52:45 -07:00
										 |  |  |     def test_del__log_destroy_pending_segfault(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2018-12-17 07:52:45 -07:00
										 |  |  |             pass | 
					
						
							|  |  |  |         task = self.new_task(self.loop, coro()) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         with self.assertRaises(AttributeError): | 
					
						
							|  |  |  |             del task._log_destroy_pending | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | @unittest.skipUnless(hasattr(futures, '_CFuture') and | 
					
						
							|  |  |  |                      hasattr(tasks, '_CTask'), | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |                      'requires the C _asyncio module') | 
					
						
							|  |  |  | @add_subclass_tests | 
					
						
							|  |  |  | class CTask_CFuture_SubclassTests(BaseTaskTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     Task = getattr(tasks, '_CTask', None) | 
					
						
							|  |  |  |     Future = getattr(futures, '_CFuture', None) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = getattr(tasks, '_c_all_tasks', None) | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(tasks, '_CTask'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							|  |  |  | @add_subclass_tests | 
					
						
							|  |  |  | class CTaskSubclass_PyFuture_Tests(BaseTaskTests, test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     Task = getattr(tasks, '_CTask', None) | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  |     Future = futures._PyFuture | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = getattr(tasks, '_c_all_tasks', None) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(futures, '_CFuture'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | @add_subclass_tests | 
					
						
							|  |  |  | class PyTask_CFutureSubclass_Tests(BaseTaskTests, test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     Future = getattr(futures, '_CFuture', None) | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  |     Task = tasks._PyTask | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = tasks._py_all_tasks | 
					
						
							| 
									
										
										
										
											2017-12-17 20:19:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(tasks, '_CTask'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | class CTask_PyFuture_Tests(BaseTaskTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     Task = getattr(tasks, '_CTask', None) | 
					
						
							|  |  |  |     Future = futures._PyFuture | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = getattr(tasks, '_c_all_tasks', None) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(futures, '_CFuture'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							|  |  |  | class PyTask_CFuture_Tests(BaseTaskTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     Task = tasks._PyTask | 
					
						
							|  |  |  |     Future = getattr(futures, '_CFuture', None) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = staticmethod(tasks._py_all_tasks) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  | class PyTask_PyFuture_Tests(BaseTaskTests, SetMethodsTest, | 
					
						
							|  |  |  |                             test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     Task = tasks._PyTask | 
					
						
							|  |  |  |     Future = futures._PyFuture | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = staticmethod(tasks._py_all_tasks) | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @add_subclass_tests | 
					
						
							|  |  |  | class PyTask_PyFuture_SubclassTests(BaseTaskTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2017-12-25 10:48:15 -05:00
										 |  |  |     Task = tasks._PyTask | 
					
						
							|  |  |  |     Future = futures._PyFuture | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 16:29:26 -05:00
										 |  |  | @unittest.skipUnless(hasattr(tasks, '_CTask'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							|  |  |  | class CTask_Future_Tests(test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_foobar(self): | 
					
						
							|  |  |  |         class Fut(asyncio.Future): | 
					
						
							|  |  |  |             @property | 
					
						
							|  |  |  |             def get_loop(self): | 
					
						
							|  |  |  |                 raise AttributeError | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             await fut | 
					
						
							|  |  |  |             return 'spam' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             fut = Fut(loop=self.loop) | 
					
						
							| 
									
										
										
										
											2018-01-24 21:15:14 +01:00
										 |  |  |             self.loop.call_later(0.1, fut.set_result, 1) | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |             task = self.loop.create_task(coro()) | 
					
						
							| 
									
										
										
										
											2017-12-23 16:29:26 -05:00
										 |  |  |             res = self.loop.run_until_complete(task) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             self.loop.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(res, 'spam') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | class BaseTaskIntrospectionTests: | 
					
						
							|  |  |  |     _register_task = None | 
					
						
							|  |  |  |     _unregister_task = None | 
					
						
							|  |  |  |     _enter_task = None | 
					
						
							|  |  |  |     _leave_task = None | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = None | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  |     def test__register_task_1(self): | 
					
						
							|  |  |  |         class TaskLike: | 
					
						
							|  |  |  |             @property | 
					
						
							|  |  |  |             def _loop(self): | 
					
						
							|  |  |  |                 return loop | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-28 17:54:02 -04:00
										 |  |  |             def done(self): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  |         task = TaskLike() | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |         loop = mock.Mock() | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  |         self._register_task(task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), {task}) | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  |         self._unregister_task(task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__register_task_2(self): | 
					
						
							|  |  |  |         class TaskLike: | 
					
						
							|  |  |  |             def get_loop(self): | 
					
						
							|  |  |  |                 return loop | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-28 17:54:02 -04:00
										 |  |  |             def done(self): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-23 15:42:27 -05:00
										 |  |  |         task = TaskLike() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |         self._register_task(task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), {task}) | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |         self._unregister_task(task) | 
					
						
							| 
									
										
										
										
											2018-05-28 17:54:02 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test__register_task_3(self): | 
					
						
							|  |  |  |         class TaskLike: | 
					
						
							|  |  |  |             def get_loop(self): | 
					
						
							|  |  |  |                 return loop | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def done(self): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         task = TaskLike() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2018-05-28 17:54:02 -04:00
										 |  |  |         self._register_task(task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2018-05-28 17:54:02 -04:00
										 |  |  |         self._unregister_task(task) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test__enter_task(self): | 
					
						
							|  |  |  |         task = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop)) | 
					
						
							|  |  |  |         self._enter_task(loop, task) | 
					
						
							|  |  |  |         self.assertIs(asyncio.current_task(loop), task) | 
					
						
							|  |  |  |         self._leave_task(loop, task) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__enter_task_failure(self): | 
					
						
							|  |  |  |         task1 = mock.Mock() | 
					
						
							|  |  |  |         task2 = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  |         self._enter_task(loop, task1) | 
					
						
							|  |  |  |         with self.assertRaises(RuntimeError): | 
					
						
							|  |  |  |             self._enter_task(loop, task2) | 
					
						
							|  |  |  |         self.assertIs(asyncio.current_task(loop), task1) | 
					
						
							|  |  |  |         self._leave_task(loop, task1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__leave_task(self): | 
					
						
							|  |  |  |         task = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  |         self._enter_task(loop, task) | 
					
						
							|  |  |  |         self._leave_task(loop, task) | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__leave_task_failure1(self): | 
					
						
							|  |  |  |         task1 = mock.Mock() | 
					
						
							|  |  |  |         task2 = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  |         self._enter_task(loop, task1) | 
					
						
							|  |  |  |         with self.assertRaises(RuntimeError): | 
					
						
							|  |  |  |             self._leave_task(loop, task2) | 
					
						
							|  |  |  |         self.assertIs(asyncio.current_task(loop), task1) | 
					
						
							|  |  |  |         self._leave_task(loop, task1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__leave_task_failure2(self): | 
					
						
							|  |  |  |         task = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							|  |  |  |         with self.assertRaises(RuntimeError): | 
					
						
							|  |  |  |             self._leave_task(loop, task) | 
					
						
							|  |  |  |         self.assertIsNone(asyncio.current_task(loop)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test__unregister_task(self): | 
					
						
							|  |  |  |         task = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |         task.get_loop = lambda: loop | 
					
						
							|  |  |  |         self._register_task(task) | 
					
						
							|  |  |  |         self._unregister_task(task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test__unregister_task_not_registered(self): | 
					
						
							|  |  |  |         task = mock.Mock() | 
					
						
							|  |  |  |         loop = mock.Mock() | 
					
						
							| 
									
										
										
										
											2017-12-23 15:04:15 -05:00
										 |  |  |         self._unregister_task(task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         self.assertEqual(self.all_tasks(loop), set()) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  | class PyIntrospectionTests(test_utils.TestCase, BaseTaskIntrospectionTests): | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |     _register_task = staticmethod(tasks._py_register_task) | 
					
						
							|  |  |  |     _unregister_task = staticmethod(tasks._py_unregister_task) | 
					
						
							|  |  |  |     _enter_task = staticmethod(tasks._py_enter_task) | 
					
						
							|  |  |  |     _leave_task = staticmethod(tasks._py_leave_task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |     all_tasks = staticmethod(tasks._py_all_tasks) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(tasks, '_c_register_task'), | 
					
						
							|  |  |  |                      'requires the C _asyncio module') | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  | class CIntrospectionTests(test_utils.TestCase, BaseTaskIntrospectionTests): | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |     if hasattr(tasks, '_c_register_task'): | 
					
						
							|  |  |  |         _register_task = staticmethod(tasks._c_register_task) | 
					
						
							|  |  |  |         _unregister_task = staticmethod(tasks._c_unregister_task) | 
					
						
							|  |  |  |         _enter_task = staticmethod(tasks._c_enter_task) | 
					
						
							|  |  |  |         _leave_task = staticmethod(tasks._c_leave_task) | 
					
						
							| 
									
										
										
										
											2024-06-22 23:28:35 +05:30
										 |  |  |         all_tasks = staticmethod(tasks._c_all_tasks) | 
					
						
							| 
									
										
										
										
											2018-01-22 19:11:18 -05:00
										 |  |  |     else: | 
					
						
							|  |  |  |         _register_task = _unregister_task = _enter_task = _leave_task = None | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BaseCurrentLoopTests: | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |     current_task = None | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							|  |  |  |         super().setUp() | 
					
						
							|  |  |  |         self.loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  |         self.set_event_loop(self.loop) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def new_task(self, coro): | 
					
						
							|  |  |  |         raise NotImplementedError | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_current_task_no_running_loop(self): | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |         self.assertIsNone(self.current_task(loop=self.loop)) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_current_task_no_running_loop_implicit(self): | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no running event loop'): | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |             self.current_task() | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_current_task_with_implicit_loop(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |             self.assertIs(self.current_task(loop=self.loop), task) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |             self.assertIs(self.current_task(None), task) | 
					
						
							|  |  |  |             self.assertIs(self.current_task(), task) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         task = self.new_task(coro()) | 
					
						
							|  |  |  |         self.loop.run_until_complete(task) | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |         self.assertIsNone(self.current_task(loop=self.loop)) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  | class PyCurrentLoopTests(BaseCurrentLoopTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |     current_task = staticmethod(tasks._py_current_task) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def new_task(self, coro): | 
					
						
							|  |  |  |         return tasks._PyTask(coro, loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  | @unittest.skipUnless(hasattr(tasks, '_CTask') and | 
					
						
							|  |  |  |                      hasattr(tasks, '_c_current_task'), | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |                      'requires the C _asyncio module') | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  | class CCurrentLoopTests(BaseCurrentLoopTests, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2022-12-22 15:38:12 +02:00
										 |  |  |     if hasattr(tasks, '_c_current_task'): | 
					
						
							|  |  |  |         current_task = staticmethod(tasks._c_current_task) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         current_task = None | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def new_task(self, coro): | 
					
						
							|  |  |  |         return getattr(tasks, '_CTask')(coro, loop=self.loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  | class GenericTaskTests(test_utils.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_future_subclass(self): | 
					
						
							|  |  |  |         self.assertTrue(issubclass(asyncio.Task, asyncio.Future)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-26 15:48:24 +03:00
										 |  |  |     @support.cpython_only | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |     def test_asyncio_module_compiled(self): | 
					
						
							|  |  |  |         # Because of circular imports it's easy to make _asyncio | 
					
						
							|  |  |  |         # module non-importable.  This is a simple test that will | 
					
						
							|  |  |  |         # fail on systems where C modules were successfully compiled | 
					
						
							| 
									
										
										
										
											2021-08-26 15:48:24 +03:00
										 |  |  |         # (hence the test for _functools etc), but _asyncio somehow didn't. | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-06-18 17:51:47 +02:00
										 |  |  |             import _functools  # noqa: F401 | 
					
						
							|  |  |  |             import _json       # noqa: F401 | 
					
						
							|  |  |  |             import _pickle     # noqa: F401 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         except ImportError: | 
					
						
							| 
									
										
										
										
											2021-08-26 15:48:24 +03:00
										 |  |  |             self.skipTest('C modules are not available') | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |         else: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2024-06-18 17:51:47 +02:00
										 |  |  |                 import _asyncio  # noqa: F401 | 
					
						
							| 
									
										
										
										
											2016-10-28 12:52:37 -04:00
										 |  |  |             except ImportError: | 
					
						
							|  |  |  |                 self.fail('_asyncio module is missing') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-14 22:21:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | class GatherTestsBase: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2016-11-04 14:29:28 -04:00
										 |  |  |         super().setUp() | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  |         self.one_loop = self.new_test_loop() | 
					
						
							|  |  |  |         self.other_loop = self.new_test_loop() | 
					
						
							|  |  |  |         self.set_event_loop(self.one_loop, cleanup=False) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _run_loop(self, loop): | 
					
						
							|  |  |  |         while loop._ready: | 
					
						
							|  |  |  |             test_utils.run_briefly(loop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _check_success(self, **kwargs): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a, b, c = [self.one_loop.create_future() for i in range(3)] | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         fut = self._gather(*self.wrap_futures(a, b, c), **kwargs) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:34:30 +01:00
										 |  |  |         cb = test_utils.MockCallback() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.add_done_callback(cb) | 
					
						
							|  |  |  |         b.set_result(1) | 
					
						
							|  |  |  |         a.set_result(2) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertEqual(cb.called, False) | 
					
						
							|  |  |  |         self.assertFalse(fut.done()) | 
					
						
							|  |  |  |         c.set_result(3) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         cb.assert_called_once_with(fut) | 
					
						
							|  |  |  |         self.assertEqual(fut.result(), [2, 1, 3]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_success(self): | 
					
						
							|  |  |  |         self._check_success() | 
					
						
							|  |  |  |         self._check_success(return_exceptions=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_result_exception_success(self): | 
					
						
							|  |  |  |         self._check_success(return_exceptions=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_one_exception(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a, b, c, d, e = [self.one_loop.create_future() for i in range(5)] | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         fut = self._gather(*self.wrap_futures(a, b, c, d, e)) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:34:30 +01:00
										 |  |  |         cb = test_utils.MockCallback() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.add_done_callback(cb) | 
					
						
							|  |  |  |         exc = ZeroDivisionError() | 
					
						
							|  |  |  |         a.set_result(1) | 
					
						
							|  |  |  |         b.set_exception(exc) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.done()) | 
					
						
							|  |  |  |         cb.assert_called_once_with(fut) | 
					
						
							|  |  |  |         self.assertIs(fut.exception(), exc) | 
					
						
							|  |  |  |         # Does nothing | 
					
						
							|  |  |  |         c.set_result(3) | 
					
						
							|  |  |  |         d.cancel() | 
					
						
							|  |  |  |         e.set_exception(RuntimeError()) | 
					
						
							| 
									
										
										
										
											2013-12-19 22:42:40 +01:00
										 |  |  |         e.exception() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_return_exceptions(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a, b, c, d = [self.one_loop.create_future() for i in range(4)] | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         fut = self._gather(*self.wrap_futures(a, b, c, d), | 
					
						
							|  |  |  |                            return_exceptions=True) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:34:30 +01:00
										 |  |  |         cb = test_utils.MockCallback() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.add_done_callback(cb) | 
					
						
							|  |  |  |         exc = ZeroDivisionError() | 
					
						
							|  |  |  |         exc2 = RuntimeError() | 
					
						
							|  |  |  |         b.set_result(1) | 
					
						
							|  |  |  |         c.set_exception(exc) | 
					
						
							|  |  |  |         a.set_result(3) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertFalse(fut.done()) | 
					
						
							|  |  |  |         d.set_exception(exc2) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.done()) | 
					
						
							|  |  |  |         cb.assert_called_once_with(fut) | 
					
						
							|  |  |  |         self.assertEqual(fut.result(), [3, 1, exc, exc2]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  |     def test_env_var_debug(self): | 
					
						
							|  |  |  |         code = '\n'.join(( | 
					
						
							| 
									
										
										
										
											2014-06-29 00:46:45 +02:00
										 |  |  |             'import asyncio.coroutines', | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |             'print(asyncio.coroutines._is_debug_mode())')) | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Test with -E to not fail if the unit test was run with | 
					
						
							|  |  |  |         # PYTHONASYNCIODEBUG set to a non-empty string | 
					
						
							| 
									
										
										
										
											2017-12-12 23:15:00 +01:00
										 |  |  |         sts, stdout, stderr = assert_python_ok('-E', '-c', code) | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  |         self.assertEqual(stdout.rstrip(), b'False') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sts, stdout, stderr = assert_python_ok('-c', code, | 
					
						
							| 
									
										
										
										
											2014-09-25 19:12:10 -04:00
										 |  |  |                                                PYTHONASYNCIODEBUG='', | 
					
						
							| 
									
										
										
										
											2017-12-12 23:15:00 +01:00
										 |  |  |                                                PYTHONDEVMODE='') | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  |         self.assertEqual(stdout.rstrip(), b'False') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sts, stdout, stderr = assert_python_ok('-c', code, | 
					
						
							| 
									
										
										
										
											2014-09-25 19:12:10 -04:00
										 |  |  |                                                PYTHONASYNCIODEBUG='1', | 
					
						
							| 
									
										
										
										
											2017-12-12 23:15:00 +01:00
										 |  |  |                                                PYTHONDEVMODE='') | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  |         self.assertEqual(stdout.rstrip(), b'True') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sts, stdout, stderr = assert_python_ok('-E', '-c', code, | 
					
						
							| 
									
										
										
										
											2014-09-25 19:12:10 -04:00
										 |  |  |                                                PYTHONASYNCIODEBUG='1', | 
					
						
							| 
									
										
										
										
											2017-12-12 23:15:00 +01:00
										 |  |  |                                                PYTHONDEVMODE='') | 
					
						
							| 
									
										
										
										
											2014-02-19 23:15:02 +01:00
										 |  |  |         self.assertEqual(stdout.rstrip(), b'False') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-12 23:15:00 +01:00
										 |  |  |         # -X dev | 
					
						
							| 
									
										
										
										
											2017-11-20 07:14:07 -08:00
										 |  |  |         sts, stdout, stderr = assert_python_ok('-E', '-X', 'dev', | 
					
						
							|  |  |  |                                                '-c', code) | 
					
						
							|  |  |  |         self.assertEqual(stdout.rstrip(), b'True') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  | class FutureGatherTests(GatherTestsBase, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def wrap_futures(self, *futures): | 
					
						
							|  |  |  |         return futures | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |     def _gather(self, *args, **kwargs): | 
					
						
							|  |  |  |         return asyncio.gather(*args, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_empty_sequence_without_loop(self): | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no current event loop'): | 
					
						
							|  |  |  |             asyncio.gather() | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_empty_sequence_use_running_loop(self): | 
					
						
							|  |  |  |         async def gather(): | 
					
						
							|  |  |  |             return asyncio.gather() | 
					
						
							|  |  |  |         fut = self.one_loop.run_until_complete(gather()) | 
					
						
							|  |  |  |         self.assertIsInstance(fut, asyncio.Future) | 
					
						
							|  |  |  |         self.assertIs(fut._loop, self.one_loop) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.done()) | 
					
						
							|  |  |  |         self.assertEqual(fut.result(), []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_empty_sequence_use_global_loop(self): | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         # Deprecated in 3.10, undeprecated in 3.12 | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         asyncio.set_event_loop(self.one_loop) | 
					
						
							|  |  |  |         self.addCleanup(asyncio.set_event_loop, None) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         fut = asyncio.gather() | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertIsInstance(fut, asyncio.Future) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(fut._loop, self.one_loop) | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.done()) | 
					
						
							|  |  |  |         self.assertEqual(fut.result(), []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_heterogenous_futures(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         fut1 = self.one_loop.create_future() | 
					
						
							|  |  |  |         fut2 = self.other_loop.create_future() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         with self.assertRaises(ValueError): | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |             asyncio.gather(fut1, fut2) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_homogenous_futures(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         children = [self.other_loop.create_future() for i in range(3)] | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         fut = asyncio.gather(*children) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(fut._loop, self.other_loop) | 
					
						
							|  |  |  |         self._run_loop(self.other_loop) | 
					
						
							|  |  |  |         self.assertFalse(fut.done()) | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |         fut = asyncio.gather(*children) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(fut._loop, self.other_loop) | 
					
						
							|  |  |  |         self._run_loop(self.other_loop) | 
					
						
							|  |  |  |         self.assertFalse(fut.done()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_one_cancellation(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a, b, c, d, e = [self.one_loop.create_future() for i in range(5)] | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         fut = asyncio.gather(a, b, c, d, e) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:34:30 +01:00
										 |  |  |         cb = test_utils.MockCallback() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.add_done_callback(cb) | 
					
						
							|  |  |  |         a.set_result(1) | 
					
						
							|  |  |  |         b.cancel() | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(fut.done()) | 
					
						
							|  |  |  |         cb.assert_called_once_with(fut) | 
					
						
							|  |  |  |         self.assertFalse(fut.cancelled()) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertIsInstance(fut.exception(), asyncio.CancelledError) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         # Does nothing | 
					
						
							|  |  |  |         c.set_result(3) | 
					
						
							|  |  |  |         d.cancel() | 
					
						
							|  |  |  |         e.set_exception(RuntimeError()) | 
					
						
							| 
									
										
										
										
											2013-12-19 22:42:40 +01:00
										 |  |  |         e.exception() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_result_exception_one_cancellation(self): | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a, b, c, d, e, f = [self.one_loop.create_future() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |                             for i in range(6)] | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         fut = asyncio.gather(a, b, c, d, e, f, return_exceptions=True) | 
					
						
							| 
									
										
										
										
											2014-02-11 11:34:30 +01:00
										 |  |  |         cb = test_utils.MockCallback() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         fut.add_done_callback(cb) | 
					
						
							|  |  |  |         a.set_result(1) | 
					
						
							|  |  |  |         zde = ZeroDivisionError() | 
					
						
							|  |  |  |         b.set_exception(zde) | 
					
						
							|  |  |  |         c.cancel() | 
					
						
							|  |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertFalse(fut.done()) | 
					
						
							|  |  |  |         d.set_result(3) | 
					
						
							|  |  |  |         e.cancel() | 
					
						
							|  |  |  |         rte = RuntimeError() | 
					
						
							|  |  |  |         f.set_exception(rte) | 
					
						
							|  |  |  |         res = self.one_loop.run_until_complete(fut) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         self.assertIsInstance(res[2], asyncio.CancelledError) | 
					
						
							|  |  |  |         self.assertIsInstance(res[4], asyncio.CancelledError) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         res[2] = res[4] = None | 
					
						
							|  |  |  |         self.assertEqual(res, [1, zde, None, 3, None, rte]) | 
					
						
							|  |  |  |         cb.assert_called_once_with(fut) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 01:36:32 +02:00
										 |  |  | class CoroutineGatherTests(GatherTestsBase, test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def wrap_futures(self, *futures): | 
					
						
							|  |  |  |         coros = [] | 
					
						
							|  |  |  |         for fut in futures: | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             async def coro(fut=fut): | 
					
						
							|  |  |  |                 return await fut | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             coros.append(coro()) | 
					
						
							|  |  |  |         return coros | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |     def _gather(self, *args, **kwargs): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return asyncio.gather(*args, **kwargs) | 
					
						
							|  |  |  |         return self.one_loop.run_until_complete(coro()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_without_loop(self): | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             return 'abc' | 
					
						
							|  |  |  |         gen1 = coro() | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.addCleanup(gen1.close) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         gen2 = coro() | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.addCleanup(gen2.close) | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         with self.assertRaisesRegex(RuntimeError, 'no current event loop'): | 
					
						
							|  |  |  |             asyncio.gather(gen1, gen2) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_constructor_use_running_loop(self): | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 'abc' | 
					
						
							|  |  |  |         gen1 = coro() | 
					
						
							|  |  |  |         gen2 = coro() | 
					
						
							|  |  |  |         async def gather(): | 
					
						
							|  |  |  |             return asyncio.gather(gen1, gen2) | 
					
						
							|  |  |  |         fut = self.one_loop.run_until_complete(gather()) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         self.assertIs(fut._loop, self.one_loop) | 
					
						
							| 
									
										
										
										
											2014-06-25 23:11:21 +02:00
										 |  |  |         self.one_loop.run_until_complete(fut) | 
					
						
							| 
									
										
										
										
											2014-06-23 00:12:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |     def test_constructor_use_global_loop(self): | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         # Deprecated in 3.10, undeprecated in 3.12 | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         async def coro(): | 
					
						
							|  |  |  |             return 'abc' | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |         asyncio.set_event_loop(self.other_loop) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.addCleanup(asyncio.set_event_loop, None) | 
					
						
							|  |  |  |         gen1 = coro() | 
					
						
							|  |  |  |         gen2 = coro() | 
					
						
							| 
									
										
										
										
											2022-12-06 19:42:12 +02:00
										 |  |  |         fut = asyncio.gather(gen1, gen2) | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         self.assertIs(fut._loop, self.other_loop) | 
					
						
							|  |  |  |         self.other_loop.run_until_complete(fut) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |     def test_duplicate_coroutines(self): | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |         async def coro(s): | 
					
						
							|  |  |  |             return s | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |         c = coro('abc') | 
					
						
							| 
									
										
										
										
											2021-04-25 13:40:44 +03:00
										 |  |  |         fut = self._gather(c, c, coro('def'), c) | 
					
						
							| 
									
										
										
										
											2014-02-06 22:06:16 -05:00
										 |  |  |         self._run_loop(self.one_loop) | 
					
						
							|  |  |  |         self.assertEqual(fut.result(), ['abc', 'abc', 'def', 'abc']) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |     def test_cancellation_broadcast(self): | 
					
						
							|  |  |  |         # Cancelling outer() cancels all children. | 
					
						
							|  |  |  |         proof = 0 | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         waiter = self.one_loop.create_future() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def inner(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             await waiter | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         child1 = asyncio.ensure_future(inner(), loop=self.one_loop) | 
					
						
							|  |  |  |         child2 = asyncio.ensure_future(inner(), loop=self.one_loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         gatherer = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def outer(): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             nonlocal proof, gatherer | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             gatherer = asyncio.gather(child1, child2) | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             await gatherer | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             proof += 100 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(outer(), loop=self.one_loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.one_loop) | 
					
						
							|  |  |  |         self.assertTrue(f.cancel()) | 
					
						
							| 
									
										
										
										
											2014-01-25 15:32:06 +01:00
										 |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             self.one_loop.run_until_complete(f) | 
					
						
							|  |  |  |         self.assertFalse(gatherer.cancel()) | 
					
						
							|  |  |  |         self.assertTrue(waiter.cancelled()) | 
					
						
							|  |  |  |         self.assertTrue(child1.cancelled()) | 
					
						
							|  |  |  |         self.assertTrue(child2.cancelled()) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.one_loop) | 
					
						
							|  |  |  |         self.assertEqual(proof, 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_exception_marking(self): | 
					
						
							|  |  |  |         # Test for the first line marked "Mark exception retrieved." | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def inner(f): | 
					
						
							|  |  |  |             await f | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |             raise RuntimeError('should not be ignored') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-11 16:07:37 +03:00
										 |  |  |         a = self.one_loop.create_future() | 
					
						
							|  |  |  |         b = self.one_loop.create_future() | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def outer(): | 
					
						
							| 
									
										
										
										
											2020-11-28 10:21:17 +02:00
										 |  |  |             await asyncio.gather(inner(a), inner(b)) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 14:48:38 -04:00
										 |  |  |         f = asyncio.ensure_future(outer(), loop=self.one_loop) | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  |         test_utils.run_briefly(self.one_loop) | 
					
						
							|  |  |  |         a.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.one_loop) | 
					
						
							|  |  |  |         b.set_result(None) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.one_loop) | 
					
						
							|  |  |  |         self.assertIsInstance(f.exception(), RuntimeError) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-20 13:24:00 +03:00
										 |  |  |     def test_issue46672(self): | 
					
						
							|  |  |  |         with mock.patch( | 
					
						
							|  |  |  |             'asyncio.base_events.BaseEventLoop.call_exception_handler', | 
					
						
							|  |  |  |         ): | 
					
						
							|  |  |  |             async def coro(s): | 
					
						
							|  |  |  |                 return s | 
					
						
							|  |  |  |             c = coro('abc') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             with self.assertRaises(TypeError): | 
					
						
							|  |  |  |                 self._gather(c, {}) | 
					
						
							|  |  |  |             self._run_loop(self.one_loop) | 
					
						
							|  |  |  |             # NameError should not happen: | 
					
						
							|  |  |  |             self.one_loop.call_exception_handler.assert_not_called() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | class RunCoroutineThreadsafeTests(test_utils.TestCase): | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |     """Test case for asyncio.run_coroutine_threadsafe.""" | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2016-11-04 14:29:28 -04:00
										 |  |  |         super().setUp() | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |         self.loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |         self.set_event_loop(self.loop) # Will cleanup properly | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |     async def add(self, a, b, fail=False, cancel=False): | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |         """Wait 0.05 second and return a + b.""" | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         await asyncio.sleep(0.05) | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         if fail: | 
					
						
							|  |  |  |             raise RuntimeError("Fail!") | 
					
						
							|  |  |  |         if cancel: | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |             asyncio.current_task(self.loop).cancel() | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         return a + b | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |     def target(self, fail=False, cancel=False, timeout=None, | 
					
						
							|  |  |  |                advance_coro=False): | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         """Run add coroutine in the event loop.""" | 
					
						
							|  |  |  |         coro = self.add(1, 2, fail=fail, cancel=cancel) | 
					
						
							|  |  |  |         future = asyncio.run_coroutine_threadsafe(coro, self.loop) | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |         if advance_coro: | 
					
						
							|  |  |  |             # this is for test_run_coroutine_threadsafe_task_factory_exception; | 
					
						
							|  |  |  |             # otherwise it spills errors and breaks **other** unittests, since | 
					
						
							|  |  |  |             # 'target' is interacting with threads. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 16:13:59 +03:00
										 |  |  |             # With this call, `coro` will be advanced. | 
					
						
							| 
									
										
										
										
											2015-11-13 12:28:48 -05:00
										 |  |  |             self.loop.call_soon_threadsafe(coro.send, None) | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return future.result(timeout) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             future.done() or future.cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_run_coroutine_threadsafe(self): | 
					
						
							|  |  |  |         """Test coroutine submission from a thread to an event loop.""" | 
					
						
							|  |  |  |         future = self.loop.run_in_executor(None, self.target) | 
					
						
							|  |  |  |         result = self.loop.run_until_complete(future) | 
					
						
							|  |  |  |         self.assertEqual(result, 3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_run_coroutine_threadsafe_with_exception(self): | 
					
						
							|  |  |  |         """Test coroutine submission from a thread to an event loop
 | 
					
						
							|  |  |  |         when an exception is raised."""
 | 
					
						
							|  |  |  |         future = self.loop.run_in_executor(None, self.target, True) | 
					
						
							|  |  |  |         with self.assertRaises(RuntimeError) as exc_context: | 
					
						
							|  |  |  |             self.loop.run_until_complete(future) | 
					
						
							|  |  |  |         self.assertIn("Fail!", exc_context.exception.args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_run_coroutine_threadsafe_with_timeout(self): | 
					
						
							|  |  |  |         """Test coroutine submission from a thread to an event loop
 | 
					
						
							|  |  |  |         when a timeout is raised."""
 | 
					
						
							|  |  |  |         callback = lambda: self.target(timeout=0) | 
					
						
							|  |  |  |         future = self.loop.run_in_executor(None, callback) | 
					
						
							|  |  |  |         with self.assertRaises(asyncio.TimeoutError): | 
					
						
							|  |  |  |             self.loop.run_until_complete(future) | 
					
						
							|  |  |  |         test_utils.run_briefly(self.loop) | 
					
						
							|  |  |  |         # Check that there's no pending task (add has been cancelled) | 
					
						
							| 
									
										
										
										
											2017-12-16 21:58:38 +02:00
										 |  |  |         for task in asyncio.all_tasks(self.loop): | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |             self.assertTrue(task.done()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_run_coroutine_threadsafe_task_cancelled(self): | 
					
						
							| 
									
										
										
										
											2021-10-07 01:13:48 +02:00
										 |  |  |         """Test coroutine submission from a thread to an event loop
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  |         when the task is cancelled."""
 | 
					
						
							|  |  |  |         callback = lambda: self.target(cancel=True) | 
					
						
							|  |  |  |         future = self.loop.run_in_executor(None, callback) | 
					
						
							|  |  |  |         with self.assertRaises(asyncio.CancelledError): | 
					
						
							|  |  |  |             self.loop.run_until_complete(future) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:20:00 -07:00
										 |  |  |     def test_run_coroutine_threadsafe_task_factory_exception(self): | 
					
						
							| 
									
										
										
										
											2021-10-07 01:13:48 +02:00
										 |  |  |         """Test coroutine submission from a thread to an event loop
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:20:00 -07:00
										 |  |  |         when the task factory raise an exception."""
 | 
					
						
							| 
									
										
										
										
											2017-12-14 20:53:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def task_factory(loop, coro): | 
					
						
							|  |  |  |             raise NameError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-28 14:09:40 -05:00
										 |  |  |         run = self.loop.run_in_executor( | 
					
						
							|  |  |  |             None, lambda: self.target(advance_coro=True)) | 
					
						
							| 
									
										
										
										
											2017-12-14 20:53:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:20:00 -07:00
										 |  |  |         # Set exception handler | 
					
						
							|  |  |  |         callback = test_utils.MockCallback() | 
					
						
							|  |  |  |         self.loop.set_exception_handler(callback) | 
					
						
							| 
									
										
										
										
											2017-12-14 20:53:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Set corrupted task factory | 
					
						
							| 
									
										
										
										
											2019-09-19 16:45:06 +02:00
										 |  |  |         self.addCleanup(self.loop.set_task_factory, | 
					
						
							|  |  |  |                         self.loop.get_task_factory()) | 
					
						
							| 
									
										
										
										
											2017-12-14 20:53:26 -05:00
										 |  |  |         self.loop.set_task_factory(task_factory) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:20:00 -07:00
										 |  |  |         # Run event loop | 
					
						
							|  |  |  |         with self.assertRaises(NameError) as exc_context: | 
					
						
							| 
									
										
										
										
											2017-12-14 20:53:26 -05:00
										 |  |  |             self.loop.run_until_complete(run) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:20:00 -07:00
										 |  |  |         # Check exceptions | 
					
						
							|  |  |  |         self.assertEqual(len(callback.call_args_list), 1) | 
					
						
							|  |  |  |         (loop, context), kwargs = callback.call_args | 
					
						
							|  |  |  |         self.assertEqual(context['exception'], exc_context.exception) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-03 08:31:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  | class SleepTests(test_utils.TestCase): | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2016-11-04 14:29:28 -04:00
										 |  |  |         super().setUp() | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  |         self.loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  |         self.set_event_loop(self.loop) | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-16 20:41:11 -05:00
										 |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         self.loop.close() | 
					
						
							|  |  |  |         self.loop = None | 
					
						
							| 
									
										
										
										
											2016-11-04 14:29:28 -04:00
										 |  |  |         super().tearDown() | 
					
						
							| 
									
										
										
										
											2015-12-16 20:41:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  |     def test_sleep_zero(self): | 
					
						
							|  |  |  |         result = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def inc_result(num): | 
					
						
							|  |  |  |             nonlocal result | 
					
						
							|  |  |  |             result += num | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  |             self.loop.call_soon(inc_result, 1) | 
					
						
							|  |  |  |             self.assertEqual(result, 0) | 
					
						
							| 
									
										
										
										
											2019-05-16 17:52:10 +03:00
										 |  |  |             num = await asyncio.sleep(0, result=10) | 
					
						
							| 
									
										
										
										
											2015-11-05 14:29:04 -05:00
										 |  |  |             self.assertEqual(result, 1) # inc'ed by call_soon | 
					
						
							|  |  |  |             inc_result(num) # num should be 11 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.loop.run_until_complete(coro()) | 
					
						
							|  |  |  |         self.assertEqual(result, 11) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 06:51:22 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  | class CompatibilityTests(test_utils.TestCase): | 
					
						
							|  |  |  |     # Tests for checking a bridge between old-styled coroutines | 
					
						
							|  |  |  |     # and async/await syntax | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							|  |  |  |         super().setUp() | 
					
						
							|  |  |  |         self.loop = asyncio.new_event_loop() | 
					
						
							| 
									
										
										
										
											2019-06-03 17:49:04 +02:00
										 |  |  |         self.set_event_loop(self.loop) | 
					
						
							| 
									
										
										
										
											2017-12-09 00:23:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         self.loop.close() | 
					
						
							|  |  |  |         self.loop = None | 
					
						
							|  |  |  |         super().tearDown() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-17 13:40:50 -07:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     unittest.main() |