| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | import asyncio | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  | import threading | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | import unittest | 
					
						
							|  |  |  | from threading import Thread | 
					
						
							|  |  |  | from unittest import TestCase | 
					
						
							| 
									
										
										
										
											2025-02-07 00:21:07 +05:30
										 |  |  | import weakref | 
					
						
							|  |  |  | from test import support | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | from test.support import threading_helper | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | threading_helper.requires_working_threading(module=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-13 21:06:55 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | class MyException(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | def tearDownModule(): | 
					
						
							|  |  |  |     asyncio._set_event_loop_policy(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestFreeThreading: | 
					
						
							|  |  |  |     def test_all_tasks_race(self) -> None: | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             loop = asyncio.get_running_loop() | 
					
						
							|  |  |  |             future = loop.create_future() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def coro(): | 
					
						
							|  |  |  |                 await future | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             tasks = set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async with asyncio.TaskGroup() as tg: | 
					
						
							|  |  |  |                 for _ in range(100): | 
					
						
							|  |  |  |                     tasks.add(tg.create_task(coro())) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |                 all_tasks = asyncio.all_tasks(loop) | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  |                 self.assertEqual(len(all_tasks), 101) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for task in all_tasks: | 
					
						
							|  |  |  |                     self.assertEqual(task.get_loop(), loop) | 
					
						
							|  |  |  |                     self.assertFalse(task.done()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |                 current = asyncio.current_task() | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  |                 self.assertEqual(current.get_loop(), loop) | 
					
						
							|  |  |  |                 self.assertSetEqual(all_tasks, tasks | {current}) | 
					
						
							|  |  |  |                 future.set_result(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def runner(): | 
					
						
							|  |  |  |             with asyncio.Runner() as runner: | 
					
						
							|  |  |  |                 loop = runner.get_loop() | 
					
						
							|  |  |  |                 loop.set_task_factory(self.factory) | 
					
						
							|  |  |  |                 runner.run(main()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         threads = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for _ in range(10): | 
					
						
							|  |  |  |             thread = Thread(target=runner) | 
					
						
							|  |  |  |             threads.append(thread) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with threading_helper.start_threads(threads): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  |     def test_all_tasks_different_thread(self) -> None: | 
					
						
							|  |  |  |         loop = None | 
					
						
							|  |  |  |         started = threading.Event() | 
					
						
							| 
									
										
										
										
											2025-01-24 23:10:24 +05:30
										 |  |  |         done = threading.Event() # used for main task not finishing early | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  |         async def coro(): | 
					
						
							| 
									
										
										
										
											2025-01-24 23:10:24 +05:30
										 |  |  |             await asyncio.Future() | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |         lock = threading.Lock() | 
					
						
							|  |  |  |         tasks = set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             nonlocal tasks, loop | 
					
						
							|  |  |  |             loop = asyncio.get_running_loop() | 
					
						
							|  |  |  |             started.set() | 
					
						
							|  |  |  |             for i in range(1000): | 
					
						
							|  |  |  |                 with lock: | 
					
						
							|  |  |  |                     asyncio.create_task(coro()) | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |                     tasks = asyncio.all_tasks(loop) | 
					
						
							| 
									
										
										
										
											2025-01-24 23:10:24 +05:30
										 |  |  |             done.wait() | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |         runner = threading.Thread(target=lambda: asyncio.run(main())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def check(): | 
					
						
							|  |  |  |             started.wait() | 
					
						
							|  |  |  |             with lock: | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |                 self.assertSetEqual(tasks & asyncio.all_tasks(loop), tasks) | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |         threads = [threading.Thread(target=check) for _ in range(10)] | 
					
						
							| 
									
										
										
										
											2025-01-24 23:10:24 +05:30
										 |  |  |         runner.start() | 
					
						
							| 
									
										
										
										
											2025-01-24 21:12:56 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |         with threading_helper.start_threads(threads): | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-24 23:10:24 +05:30
										 |  |  |         done.set() | 
					
						
							|  |  |  |         runner.join() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-07 00:21:07 +05:30
										 |  |  |     def test_task_different_thread_finalized(self) -> None: | 
					
						
							|  |  |  |         task = None | 
					
						
							|  |  |  |         async def func(): | 
					
						
							|  |  |  |             nonlocal task | 
					
						
							|  |  |  |             task = asyncio.current_task() | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |         def runner(): | 
					
						
							|  |  |  |             with asyncio.Runner() as runner: | 
					
						
							|  |  |  |                 loop = runner.get_loop() | 
					
						
							|  |  |  |                 loop.set_task_factory(self.factory) | 
					
						
							|  |  |  |                 runner.run(func()) | 
					
						
							|  |  |  |         thread = Thread(target=runner) | 
					
						
							| 
									
										
										
										
											2025-02-07 00:21:07 +05:30
										 |  |  |         thread.start() | 
					
						
							|  |  |  |         thread.join() | 
					
						
							|  |  |  |         wr = weakref.ref(task) | 
					
						
							|  |  |  |         del thread | 
					
						
							|  |  |  |         del task | 
					
						
							|  |  |  |         # task finalization in different thread shouldn't crash | 
					
						
							|  |  |  |         support.gc_collect() | 
					
						
							|  |  |  |         self.assertIsNone(wr()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-13 21:06:55 +05:30
										 |  |  |     def test_run_coroutine_threadsafe(self) -> None: | 
					
						
							|  |  |  |         results = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def in_thread(loop: asyncio.AbstractEventLoop): | 
					
						
							|  |  |  |             coro = asyncio.sleep(0.1, result=42) | 
					
						
							|  |  |  |             fut = asyncio.run_coroutine_threadsafe(coro, loop) | 
					
						
							|  |  |  |             result = fut.result() | 
					
						
							|  |  |  |             self.assertEqual(result, 42) | 
					
						
							|  |  |  |             results.append(result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             loop = asyncio.get_running_loop() | 
					
						
							|  |  |  |             async with asyncio.TaskGroup() as tg: | 
					
						
							|  |  |  |                 for _ in range(10): | 
					
						
							|  |  |  |                     tg.create_task(asyncio.to_thread(in_thread, loop)) | 
					
						
							|  |  |  |             self.assertEqual(results, [42] * 10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with asyncio.Runner() as r: | 
					
						
							|  |  |  |             loop = r.get_loop() | 
					
						
							|  |  |  |             loop.set_task_factory(self.factory) | 
					
						
							|  |  |  |             r.run(main()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_run_coroutine_threadsafe_exception(self) -> None: | 
					
						
							|  |  |  |         async def coro(): | 
					
						
							|  |  |  |             await asyncio.sleep(0) | 
					
						
							|  |  |  |             raise MyException("test") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def in_thread(loop: asyncio.AbstractEventLoop): | 
					
						
							|  |  |  |             fut = asyncio.run_coroutine_threadsafe(coro(), loop) | 
					
						
							|  |  |  |             return fut.result() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         async def main(): | 
					
						
							|  |  |  |             loop = asyncio.get_running_loop() | 
					
						
							|  |  |  |             tasks = [] | 
					
						
							|  |  |  |             for _ in range(10): | 
					
						
							|  |  |  |                 task = loop.create_task(asyncio.to_thread(in_thread, loop)) | 
					
						
							|  |  |  |                 tasks.append(task) | 
					
						
							|  |  |  |             results = await asyncio.gather(*tasks, return_exceptions=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.assertEqual(len(results), 10) | 
					
						
							|  |  |  |             for result in results: | 
					
						
							|  |  |  |                 self.assertIsInstance(result, MyException) | 
					
						
							|  |  |  |                 self.assertEqual(str(result), "test") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with asyncio.Runner() as r: | 
					
						
							|  |  |  |             loop = r.get_loop() | 
					
						
							|  |  |  |             loop.set_task_factory(self.factory) | 
					
						
							|  |  |  |             r.run(main()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | class TestPyFreeThreading(TestFreeThreading, TestCase): | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							|  |  |  |         self._old_current_task = asyncio.current_task | 
					
						
							|  |  |  |         asyncio.current_task = asyncio.tasks.current_task = asyncio.tasks._py_current_task | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |         self._old_all_tasks = asyncio.all_tasks | 
					
						
							|  |  |  |         asyncio.all_tasks = asyncio.tasks.all_tasks = asyncio.tasks._py_all_tasks | 
					
						
							|  |  |  |         self._old_Task = asyncio.Task | 
					
						
							|  |  |  |         asyncio.Task = asyncio.tasks.Task = asyncio.tasks._PyTask | 
					
						
							|  |  |  |         self._old_Future = asyncio.Future | 
					
						
							|  |  |  |         asyncio.Future = asyncio.futures.Future = asyncio.futures._PyFuture | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |         return super().setUp() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         asyncio.current_task = asyncio.tasks.current_task = self._old_current_task | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |         asyncio.all_tasks = asyncio.tasks.all_tasks = self._old_all_tasks | 
					
						
							|  |  |  |         asyncio.Task = asyncio.tasks.Task = self._old_Task | 
					
						
							|  |  |  |         asyncio.Future = asyncio.tasks.Future = self._old_Future | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |         return super().tearDown() | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 16:53:55 +00:00
										 |  |  |     def factory(self, loop, coro, **kwargs): | 
					
						
							|  |  |  |         return asyncio.tasks._PyTask(coro, loop=loop, **kwargs) | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(asyncio.tasks, "_c_all_tasks"), "requires _asyncio") | 
					
						
							|  |  |  | class TestCFreeThreading(TestFreeThreading, TestCase): | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							|  |  |  |         self._old_current_task = asyncio.current_task | 
					
						
							|  |  |  |         asyncio.current_task = asyncio.tasks.current_task = asyncio.tasks._c_current_task | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |         self._old_all_tasks = asyncio.all_tasks | 
					
						
							|  |  |  |         asyncio.all_tasks = asyncio.tasks.all_tasks = asyncio.tasks._c_all_tasks | 
					
						
							|  |  |  |         self._old_Task = asyncio.Task | 
					
						
							|  |  |  |         asyncio.Task = asyncio.tasks.Task = asyncio.tasks._CTask | 
					
						
							|  |  |  |         self._old_Future = asyncio.Future | 
					
						
							|  |  |  |         asyncio.Future = asyncio.futures.Future = asyncio.futures._CFuture | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |         return super().setUp() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         asyncio.current_task = asyncio.tasks.current_task = self._old_current_task | 
					
						
							| 
									
										
										
										
											2025-02-24 21:33:52 +05:30
										 |  |  |         asyncio.all_tasks = asyncio.tasks.all_tasks = self._old_all_tasks | 
					
						
							|  |  |  |         asyncio.Task = asyncio.tasks.Task = self._old_Task | 
					
						
							|  |  |  |         asyncio.Future = asyncio.futures.Future = self._old_Future | 
					
						
							| 
									
										
										
										
											2025-02-09 17:35:39 +05:30
										 |  |  |         return super().tearDown() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 16:53:55 +00:00
										 |  |  |     def factory(self, loop, coro, **kwargs): | 
					
						
							|  |  |  |         return asyncio.tasks._CTask(coro, loop=loop, **kwargs) | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestEagerPyFreeThreading(TestPyFreeThreading): | 
					
						
							| 
									
										
										
										
											2025-01-20 16:53:55 +00:00
										 |  |  |     def factory(self, loop, coro, eager_start=True, **kwargs): | 
					
						
							|  |  |  |         return asyncio.tasks._PyTask(coro, loop=loop, **kwargs, eager_start=eager_start) | 
					
						
							| 
									
										
										
										
											2025-01-04 14:18:22 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @unittest.skipUnless(hasattr(asyncio.tasks, "_c_all_tasks"), "requires _asyncio") | 
					
						
							|  |  |  | class TestEagerCFreeThreading(TestCFreeThreading, TestCase): | 
					
						
							| 
									
										
										
										
											2025-01-20 16:53:55 +00:00
										 |  |  |     def factory(self, loop, coro, eager_start=True, **kwargs): | 
					
						
							|  |  |  |         return asyncio.tasks._CTask(coro, loop=loop, **kwargs, eager_start=eager_start) |