| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | import unittest | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import textwrap | 
					
						
							|  |  |  | import importlib | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  | import socket | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  | import threading | 
					
						
							| 
									
										
										
										
											2025-07-08 13:23:31 +01:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  | from asyncio import staggered, taskgroups, base_events, tasks | 
					
						
							| 
									
										
										
										
											2025-05-04 02:51:57 +02:00
										 |  |  | from unittest.mock import ANY | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  | from test.support import ( | 
					
						
							|  |  |  |     os_helper, | 
					
						
							|  |  |  |     SHORT_TIMEOUT, | 
					
						
							|  |  |  |     busy_retry, | 
					
						
							|  |  |  |     requires_gil_enabled, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | from test.support.script_helper import make_script | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  | from test.support.socket_helper import find_unused_port | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import subprocess | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PROCESS_VM_READV_SUPPORTED = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | try: | 
					
						
							| 
									
										
										
										
											2025-05-05 01:30:14 +01:00
										 |  |  |     from _remote_debugging import PROCESS_VM_READV_SUPPORTED | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |     from _remote_debugging import RemoteUnwinder | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |     from _remote_debugging import FrameInfo, CoroInfo, TaskInfo | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | except ImportError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |     raise unittest.SkipTest( | 
					
						
							|  |  |  |         "Test only runs when _remote_debugging is available" | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def _make_test_script(script_dir, script_basename, source): | 
					
						
							|  |  |  |     to_return = make_script(script_dir, script_basename, source) | 
					
						
							|  |  |  |     importlib.invalidate_caches() | 
					
						
							|  |  |  |     return to_return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | skip_if_not_supported = unittest.skipIf( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |     ( | 
					
						
							|  |  |  |         sys.platform != "darwin" | 
					
						
							|  |  |  |         and sys.platform != "linux" | 
					
						
							|  |  |  |         and sys.platform != "win32" | 
					
						
							|  |  |  |     ), | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     "Test only runs on Linux, Windows and MacOS", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  | def get_stack_trace(pid): | 
					
						
							| 
									
										
										
										
											2025-05-26 15:31:47 +01:00
										 |  |  |     unwinder = RemoteUnwinder(pid, all_threads=True, debug=True) | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |     return unwinder.get_stack_trace() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_async_stack_trace(pid): | 
					
						
							| 
									
										
										
										
											2025-05-26 15:31:47 +01:00
										 |  |  |     unwinder = RemoteUnwinder(pid, debug=True) | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |     return unwinder.get_async_stack_trace() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_all_awaited_by(pid): | 
					
						
							| 
									
										
										
										
											2025-05-26 15:31:47 +01:00
										 |  |  |     unwinder = RemoteUnwinder(pid, debug=True) | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |     return unwinder.get_all_awaited_by() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | class TestGetStackTrace(unittest.TestCase): | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |     maxDiff = None | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |     def test_remote_stack_trace(self): | 
					
						
							|  |  |  |         # Spawn a process with some realistic Python code | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             import time, sys, socket, threading | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |             def bar(): | 
					
						
							|  |  |  |                 for x in range(100): | 
					
						
							|  |  |  |                     if x == 50: | 
					
						
							|  |  |  |                         baz() | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |             def baz(): | 
					
						
							|  |  |  |                 foo() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def foo(): | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |                 sock.sendall(b"ready:thread\\n"); time.sleep(10_000)  # same line number | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             t = threading.Thread(target=bar) | 
					
						
							|  |  |  |             t.start() | 
					
						
							|  |  |  |             sock.sendall(b"ready:main\\n"); t.join()  # same line number | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             client_socket = None | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |                 response = b"" | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 while ( | 
					
						
							|  |  |  |                     b"ready:main" not in response | 
					
						
							|  |  |  |                     or b"ready:thread" not in response | 
					
						
							|  |  |  |                 ): | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |                     response += client_socket.recv(1024) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |                 stack_trace = get_stack_trace(p.pid) | 
					
						
							|  |  |  |             except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             thread_expected_stack_trace = [ | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 FrameInfo([script_name, 15, "foo"]), | 
					
						
							|  |  |  |                 FrameInfo([script_name, 12, "baz"]), | 
					
						
							|  |  |  |                 FrameInfo([script_name, 9, "bar"]), | 
					
						
							|  |  |  |                 FrameInfo([threading.__file__, ANY, "Thread.run"]), | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |             ] | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             # Is possible that there are more threads, so we check that the | 
					
						
							|  |  |  |             # expected stack traces are in the result (looking at you Windows!) | 
					
						
							|  |  |  |             self.assertIn((ANY, thread_expected_stack_trace), stack_trace) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Check that the main thread stack trace is in the result | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |             frame = FrameInfo([script_name, 19, "<module>"]) | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             for _, stack in stack_trace: | 
					
						
							|  |  |  |                 if frame in stack: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.fail("Main thread stack trace not found in result") | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     def test_async_remote_stack_trace(self): | 
					
						
							|  |  |  |         # Spawn a process with some realistic Python code | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             import asyncio | 
					
						
							|  |  |  |             import time | 
					
						
							|  |  |  |             import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             import socket | 
					
						
							|  |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             def c5(): | 
					
						
							| 
									
										
										
										
											2025-05-05 14:38:51 +02:00
										 |  |  |                 sock.sendall(b"ready"); time.sleep(10_000)  # same line number | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def c4(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							|  |  |  |                 c5() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def c3(): | 
					
						
							|  |  |  |                 await c4() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def c2(): | 
					
						
							|  |  |  |                 await c3() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def c1(task): | 
					
						
							|  |  |  |                 await task | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def main(): | 
					
						
							|  |  |  |                 async with asyncio.TaskGroup() as tg: | 
					
						
							|  |  |  |                     task = tg.create_task(c2(), name="c2_root") | 
					
						
							|  |  |  |                     tg.create_task(c1(task), name="sub_main_1") | 
					
						
							|  |  |  |                     tg.create_task(c1(task), name="sub_main_2") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def new_eager_loop(): | 
					
						
							|  |  |  |                 loop = asyncio.new_event_loop() | 
					
						
							|  |  |  |                 eager_task_factory = asyncio.create_eager_task_factory( | 
					
						
							|  |  |  |                     asyncio.Task) | 
					
						
							|  |  |  |                 loop.set_task_factory(eager_task_factory) | 
					
						
							|  |  |  |                 return loop | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             asyncio.run(main(), loop_factory={{TASK_FACTORY}}) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         for task_factory_variant in "asyncio.new_event_loop", "new_eager_loop": | 
					
						
							|  |  |  |             with ( | 
					
						
							|  |  |  |                 self.subTest(task_factory_variant=task_factory_variant), | 
					
						
							|  |  |  |                 os_helper.temp_dir() as work_dir, | 
					
						
							|  |  |  |             ): | 
					
						
							|  |  |  |                 script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |                 os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 server_socket = socket.socket( | 
					
						
							|  |  |  |                     socket.AF_INET, socket.SOCK_STREAM | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 server_socket.setsockopt( | 
					
						
							|  |  |  |                     socket.SOL_SOCKET, socket.SO_REUSEADDR, 1 | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |                 server_socket.listen(1) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 script_name = _make_test_script( | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     script_dir, | 
					
						
							|  |  |  |                     "script", | 
					
						
							|  |  |  |                     script.format(TASK_FACTORY=task_factory_variant), | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 client_socket = None | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 try: | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                     client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                     server_socket.close() | 
					
						
							|  |  |  |                     response = client_socket.recv(1024) | 
					
						
							|  |  |  |                     self.assertEqual(response, b"ready") | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                     stack_trace = get_async_stack_trace(p.pid) | 
					
						
							|  |  |  |                 except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                     self.skipTest( | 
					
						
							|  |  |  |                         "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                     if client_socket is not None: | 
					
						
							|  |  |  |                         client_socket.close() | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                     p.kill() | 
					
						
							|  |  |  |                     p.terminate() | 
					
						
							|  |  |  |                     p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                 # First check all the tasks are present | 
					
						
							|  |  |  |                 tasks_names = [ | 
					
						
							|  |  |  |                     task.task_name for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |                 for task_name in ["c2_root", "sub_main_1", "sub_main_2"]: | 
					
						
							|  |  |  |                     self.assertIn(task_name, tasks_names) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Now ensure that the awaited_by_relationships are correct | 
					
						
							|  |  |  |                 id_to_task = { | 
					
						
							|  |  |  |                     task.task_id: task for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 task_name_to_awaited_by = { | 
					
						
							|  |  |  |                     task.task_name: set( | 
					
						
							|  |  |  |                         id_to_task[awaited.task_name].task_name | 
					
						
							|  |  |  |                         for awaited in task.awaited_by | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 self.assertEqual( | 
					
						
							|  |  |  |                     task_name_to_awaited_by, | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         "c2_root": {"Task-1", "sub_main_1", "sub_main_2"}, | 
					
						
							|  |  |  |                         "Task-1": set(), | 
					
						
							|  |  |  |                         "sub_main_1": {"Task-1"}, | 
					
						
							|  |  |  |                         "sub_main_2": {"Task-1"}, | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                 # Now ensure that the coroutine stacks are correct | 
					
						
							|  |  |  |                 coroutine_stacks = { | 
					
						
							|  |  |  |                     task.task_name: sorted( | 
					
						
							|  |  |  |                         tuple(tuple(frame) for frame in coro.call_stack) | 
					
						
							|  |  |  |                         for coro in task.coroutine_stack | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 self.assertEqual( | 
					
						
							|  |  |  |                     coroutine_stacks, | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         "Task-1": [ | 
					
						
							|  |  |  |                             ( | 
					
						
							|  |  |  |                                 tuple( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         taskgroups.__file__, | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                         "TaskGroup._aexit", | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                                 tuple( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         taskgroups.__file__, | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                         "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                                 tuple([script_name, 26, "main"]), | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                         "c2_root": [ | 
					
						
							|  |  |  |                             ( | 
					
						
							|  |  |  |                                 tuple([script_name, 10, "c5"]), | 
					
						
							|  |  |  |                                 tuple([script_name, 14, "c4"]), | 
					
						
							|  |  |  |                                 tuple([script_name, 17, "c3"]), | 
					
						
							|  |  |  |                                 tuple([script_name, 20, "c2"]), | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                         "sub_main_1": [(tuple([script_name, 23, "c1"]),)], | 
					
						
							|  |  |  |                         "sub_main_2": [(tuple([script_name, 23, "c1"]),)], | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Now ensure the coroutine stacks for the awaited_by relationships are correct. | 
					
						
							|  |  |  |                 awaited_by_coroutine_stacks = { | 
					
						
							|  |  |  |                     task.task_name: sorted( | 
					
						
							|  |  |  |                         ( | 
					
						
							|  |  |  |                             id_to_task[coro.task_name].task_name, | 
					
						
							|  |  |  |                             tuple(tuple(frame) for frame in coro.call_stack), | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         for coro in task.awaited_by | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 self.assertEqual( | 
					
						
							|  |  |  |                     awaited_by_coroutine_stacks, | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         "Task-1": [], | 
					
						
							|  |  |  |                         "c2_root": [ | 
					
						
							|  |  |  |                             ( | 
					
						
							|  |  |  |                                 "Task-1", | 
					
						
							|  |  |  |                                 ( | 
					
						
							|  |  |  |                                     tuple( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                         [ | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                             "TaskGroup._aexit", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                                     tuple( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                         [ | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                             "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                                     tuple([script_name, 26, "main"]), | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                             ), | 
					
						
							|  |  |  |                             ("sub_main_1", (tuple([script_name, 23, "c1"]),)), | 
					
						
							|  |  |  |                             ("sub_main_2", (tuple([script_name, 23, "c1"]),)), | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                         "sub_main_1": [ | 
					
						
							|  |  |  |                             ( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                 "Task-1", | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                                 ( | 
					
						
							|  |  |  |                                     tuple( | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							|  |  |  |                                             "TaskGroup._aexit", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							|  |  |  |                                     tuple( | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							|  |  |  |                                             "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							|  |  |  |                                     tuple([script_name, 26, "main"]), | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                         "sub_main_2": [ | 
					
						
							|  |  |  |                             ( | 
					
						
							|  |  |  |                                 "Task-1", | 
					
						
							|  |  |  |                                 ( | 
					
						
							|  |  |  |                                     tuple( | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							|  |  |  |                                             "TaskGroup._aexit", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							|  |  |  |                                     tuple( | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             taskgroups.__file__, | 
					
						
							|  |  |  |                                             ANY, | 
					
						
							|  |  |  |                                             "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                         ] | 
					
						
							|  |  |  |                                     ), | 
					
						
							|  |  |  |                                     tuple([script_name, 26, "main"]), | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         ], | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     def test_asyncgen_remote_stack_trace(self): | 
					
						
							|  |  |  |         # Spawn a process with some realistic Python code | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             import asyncio | 
					
						
							|  |  |  |             import time | 
					
						
							|  |  |  |             import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             import socket | 
					
						
							|  |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def gen_nested_call(): | 
					
						
							| 
									
										
										
										
											2025-05-05 14:38:51 +02:00
										 |  |  |                 sock.sendall(b"ready"); time.sleep(10_000)  # same line number | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def gen(): | 
					
						
							|  |  |  |                 for num in range(2): | 
					
						
							|  |  |  |                     yield num | 
					
						
							|  |  |  |                     if num == 1: | 
					
						
							|  |  |  |                         await gen_nested_call() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def main(): | 
					
						
							|  |  |  |                 async for el in gen(): | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             asyncio.run(main()) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             client_socket = None | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							|  |  |  |                 response = client_socket.recv(1024) | 
					
						
							|  |  |  |                 self.assertEqual(response, b"ready") | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 stack_trace = get_async_stack_trace(p.pid) | 
					
						
							|  |  |  |             except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             # For this simple asyncgen test, we only expect one task with the full coroutine stack | 
					
						
							|  |  |  |             self.assertEqual(len(stack_trace[0].awaited_by), 1) | 
					
						
							|  |  |  |             task = stack_trace[0].awaited_by[0] | 
					
						
							|  |  |  |             self.assertEqual(task.task_name, "Task-1") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Check the coroutine stack - based on actual output, only shows main | 
					
						
							|  |  |  |             coroutine_stack = sorted( | 
					
						
							|  |  |  |                 tuple(tuple(frame) for frame in coro.call_stack) | 
					
						
							|  |  |  |                 for coro in task.coroutine_stack | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 coroutine_stack, | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 [ | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                     ( | 
					
						
							|  |  |  |                         tuple([script_name, 10, "gen_nested_call"]), | 
					
						
							|  |  |  |                         tuple([script_name, 16, "gen"]), | 
					
						
							|  |  |  |                         tuple([script_name, 19, "main"]), | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 ], | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # No awaited_by relationships expected for this simple case | 
					
						
							|  |  |  |             self.assertEqual(task.awaited_by, []) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |     def test_async_gather_remote_stack_trace(self): | 
					
						
							|  |  |  |         # Spawn a process with some realistic Python code | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             import asyncio | 
					
						
							|  |  |  |             import time | 
					
						
							|  |  |  |             import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             import socket | 
					
						
							|  |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def deep(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2025-05-05 14:38:51 +02:00
										 |  |  |                 sock.sendall(b"ready"); time.sleep(10_000)  # same line number | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def c1(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							|  |  |  |                 await deep() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def c2(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def main(): | 
					
						
							|  |  |  |                 await asyncio.gather(c1(), c2()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             asyncio.run(main()) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             client_socket = None | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							|  |  |  |                 response = client_socket.recv(1024) | 
					
						
							|  |  |  |                 self.assertEqual(response, b"ready") | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 stack_trace = get_async_stack_trace(p.pid) | 
					
						
							|  |  |  |             except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             # First check all the tasks are present | 
					
						
							|  |  |  |             tasks_names = [ | 
					
						
							|  |  |  |                 task.task_name for task in stack_trace[0].awaited_by | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  |             ] | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             for task_name in ["Task-1", "Task-2"]: | 
					
						
							|  |  |  |                 self.assertIn(task_name, tasks_names) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure that the awaited_by_relationships are correct | 
					
						
							|  |  |  |             id_to_task = { | 
					
						
							|  |  |  |                 task.task_id: task for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             task_name_to_awaited_by = { | 
					
						
							|  |  |  |                 task.task_name: set( | 
					
						
							|  |  |  |                     id_to_task[awaited.task_name].task_name | 
					
						
							|  |  |  |                     for awaited in task.awaited_by | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 task_name_to_awaited_by, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": set(), | 
					
						
							|  |  |  |                     "Task-2": {"Task-1"}, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure that the coroutine stacks are correct | 
					
						
							|  |  |  |             coroutine_stacks = { | 
					
						
							|  |  |  |                 task.task_name: sorted( | 
					
						
							|  |  |  |                     tuple(tuple(frame) for frame in coro.call_stack) | 
					
						
							|  |  |  |                     for coro in task.coroutine_stack | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 coroutine_stacks, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": [(tuple([script_name, 21, "main"]),)], | 
					
						
							|  |  |  |                     "Task-2": [ | 
					
						
							|  |  |  |                         ( | 
					
						
							|  |  |  |                             tuple([script_name, 11, "deep"]), | 
					
						
							|  |  |  |                             tuple([script_name, 15, "c1"]), | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure the coroutine stacks for the awaited_by relationships are correct. | 
					
						
							|  |  |  |             awaited_by_coroutine_stacks = { | 
					
						
							|  |  |  |                 task.task_name: sorted( | 
					
						
							|  |  |  |                     ( | 
					
						
							|  |  |  |                         id_to_task[coro.task_name].task_name, | 
					
						
							|  |  |  |                         tuple(tuple(frame) for frame in coro.call_stack), | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     for coro in task.awaited_by | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 awaited_by_coroutine_stacks, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": [], | 
					
						
							|  |  |  |                     "Task-2": [ | 
					
						
							|  |  |  |                         ("Task-1", (tuple([script_name, 21, "main"]),)) | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-01-22 08:25:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |     def test_async_staggered_race_remote_stack_trace(self): | 
					
						
							|  |  |  |         # Spawn a process with some realistic Python code | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |             import asyncio.staggered | 
					
						
							|  |  |  |             import time | 
					
						
							|  |  |  |             import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             import socket | 
					
						
							|  |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def deep(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2025-05-05 14:38:51 +02:00
										 |  |  |                 sock.sendall(b"ready"); time.sleep(10_000)  # same line number | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def c1(): | 
					
						
							|  |  |  |                 await asyncio.sleep(0) | 
					
						
							|  |  |  |                 await deep() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def c2(): | 
					
						
							| 
									
										
										
										
											2025-05-05 14:38:51 +02:00
										 |  |  |                 await asyncio.sleep(10_000) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             async def main(): | 
					
						
							|  |  |  |                 await asyncio.staggered.staggered_race( | 
					
						
							|  |  |  |                     [c1, c2], | 
					
						
							|  |  |  |                     delay=None, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             asyncio.run(main()) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             client_socket = None | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							|  |  |  |                 response = client_socket.recv(1024) | 
					
						
							|  |  |  |                 self.assertEqual(response, b"ready") | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |                 stack_trace = get_async_stack_trace(p.pid) | 
					
						
							|  |  |  |             except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             # First check all the tasks are present | 
					
						
							|  |  |  |             tasks_names = [ | 
					
						
							|  |  |  |                 task.task_name for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |             for task_name in ["Task-1", "Task-2"]: | 
					
						
							|  |  |  |                 self.assertIn(task_name, tasks_names) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure that the awaited_by_relationships are correct | 
					
						
							|  |  |  |             id_to_task = { | 
					
						
							|  |  |  |                 task.task_id: task for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             task_name_to_awaited_by = { | 
					
						
							|  |  |  |                 task.task_name: set( | 
					
						
							|  |  |  |                     id_to_task[awaited.task_name].task_name | 
					
						
							|  |  |  |                     for awaited in task.awaited_by | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 task_name_to_awaited_by, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": set(), | 
					
						
							|  |  |  |                     "Task-2": {"Task-1"}, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure that the coroutine stacks are correct | 
					
						
							|  |  |  |             coroutine_stacks = { | 
					
						
							|  |  |  |                 task.task_name: sorted( | 
					
						
							|  |  |  |                     tuple(tuple(frame) for frame in coro.call_stack) | 
					
						
							|  |  |  |                     for coro in task.coroutine_stack | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 coroutine_stacks, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": [ | 
					
						
							|  |  |  |                         ( | 
					
						
							|  |  |  |                             tuple([staggered.__file__, ANY, "staggered_race"]), | 
					
						
							|  |  |  |                             tuple([script_name, 21, "main"]), | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                     "Task-2": [ | 
					
						
							|  |  |  |                         ( | 
					
						
							|  |  |  |                             tuple([script_name, 11, "deep"]), | 
					
						
							|  |  |  |                             tuple([script_name, 15, "c1"]), | 
					
						
							|  |  |  |                             tuple( | 
					
						
							|  |  |  |                                 [ | 
					
						
							|  |  |  |                                     staggered.__file__, | 
					
						
							|  |  |  |                                     ANY, | 
					
						
							|  |  |  |                                     "staggered_race.<locals>.run_one_coro", | 
					
						
							|  |  |  |                                 ] | 
					
						
							|  |  |  |                             ), | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now ensure the coroutine stacks for the awaited_by relationships are correct. | 
					
						
							|  |  |  |             awaited_by_coroutine_stacks = { | 
					
						
							|  |  |  |                 task.task_name: sorted( | 
					
						
							|  |  |  |                     ( | 
					
						
							|  |  |  |                         id_to_task[coro.task_name].task_name, | 
					
						
							|  |  |  |                         tuple(tuple(frame) for frame in coro.call_stack), | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     for coro in task.awaited_by | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 for task in stack_trace[0].awaited_by | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 awaited_by_coroutine_stacks, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Task-1": [], | 
					
						
							|  |  |  |                     "Task-2": [ | 
					
						
							|  |  |  |                         ( | 
					
						
							|  |  |  |                             "Task-1", | 
					
						
							|  |  |  |                             ( | 
					
						
							|  |  |  |                                 tuple( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                                     [staggered.__file__, ANY, "staggered_race"] | 
					
						
							|  |  |  |                                 ), | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                                 tuple([script_name, 21, "main"]), | 
					
						
							|  |  |  |                             ), | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-01-26 15:44:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |     def test_async_global_awaited_by(self): | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |         port = find_unused_port() | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             import asyncio | 
					
						
							|  |  |  |             import os | 
					
						
							|  |  |  |             import random | 
					
						
							|  |  |  |             import sys | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             import socket | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             from string import ascii_lowercase, digits | 
					
						
							|  |  |  |             from test.support import socket_helper, SHORT_TIMEOUT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             HOST = '127.0.0.1' | 
					
						
							|  |  |  |             PORT = socket_helper.find_unused_port() | 
					
						
							|  |  |  |             connections = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             class EchoServerProtocol(asyncio.Protocol): | 
					
						
							|  |  |  |                 def connection_made(self, transport): | 
					
						
							|  |  |  |                     global connections | 
					
						
							|  |  |  |                     connections += 1 | 
					
						
							|  |  |  |                     self.transport = transport | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 def data_received(self, data): | 
					
						
							|  |  |  |                     self.transport.write(data) | 
					
						
							|  |  |  |                     self.transport.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def echo_client(message): | 
					
						
							|  |  |  |                 reader, writer = await asyncio.open_connection(HOST, PORT) | 
					
						
							|  |  |  |                 writer.write(message.encode()) | 
					
						
							|  |  |  |                 await writer.drain() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 data = await reader.read(100) | 
					
						
							|  |  |  |                 assert message == data.decode() | 
					
						
							|  |  |  |                 writer.close() | 
					
						
							|  |  |  |                 await writer.wait_closed() | 
					
						
							| 
									
										
										
										
											2025-05-04 18:52:20 +02:00
										 |  |  |                 # Signal we are ready to sleep | 
					
						
							|  |  |  |                 sock.sendall(b"ready") | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                 await asyncio.sleep(SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def echo_client_spam(server): | 
					
						
							|  |  |  |                 async with asyncio.TaskGroup() as tg: | 
					
						
							|  |  |  |                     while connections < 1000: | 
					
						
							|  |  |  |                         msg = list(ascii_lowercase + digits) | 
					
						
							|  |  |  |                         random.shuffle(msg) | 
					
						
							|  |  |  |                         tg.create_task(echo_client("".join(msg))) | 
					
						
							|  |  |  |                         await asyncio.sleep(0) | 
					
						
							| 
									
										
										
										
											2025-05-04 18:52:20 +02:00
										 |  |  |                     # at least a 1000 tasks created. Each task will signal | 
					
						
							|  |  |  |                     # when is ready to avoid the race caused by the fact that | 
					
						
							|  |  |  |                     # tasks are waited on tg.__exit__ and we cannot signal when | 
					
						
							|  |  |  |                     # that happens otherwise | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                 # at this point all client tasks completed without assertion errors | 
					
						
							|  |  |  |                 # let's wrap up the test | 
					
						
							|  |  |  |                 server.close() | 
					
						
							|  |  |  |                 await server.wait_closed() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             async def main(): | 
					
						
							|  |  |  |                 loop = asyncio.get_running_loop() | 
					
						
							|  |  |  |                 server = await loop.create_server(EchoServerProtocol, HOST, PORT) | 
					
						
							|  |  |  |                 async with server: | 
					
						
							|  |  |  |                     async with asyncio.TaskGroup() as tg: | 
					
						
							|  |  |  |                         tg.create_task(server.serve_forever(), name="server task") | 
					
						
							|  |  |  |                         tg.create_task(echo_client_spam(server), name="echo client spam") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             asyncio.run(main()) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |         stack_trace = None | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |             client_socket = None | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							| 
									
										
										
										
											2025-05-04 18:52:20 +02:00
										 |  |  |                 for _ in range(1000): | 
					
						
							|  |  |  |                     expected_response = b"ready" | 
					
						
							|  |  |  |                     response = client_socket.recv(len(expected_response)) | 
					
						
							|  |  |  |                     self.assertEqual(response, expected_response) | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                 for _ in busy_retry(SHORT_TIMEOUT): | 
					
						
							|  |  |  |                     try: | 
					
						
							|  |  |  |                         all_awaited_by = get_all_awaited_by(p.pid) | 
					
						
							|  |  |  |                     except RuntimeError as re: | 
					
						
							|  |  |  |                         # This call reads a linked list in another process with | 
					
						
							|  |  |  |                         # no synchronization. That occasionally leads to invalid | 
					
						
							|  |  |  |                         # reads. Here we avoid making the test flaky. | 
					
						
							|  |  |  |                         msg = str(re) | 
					
						
							|  |  |  |                         if msg.startswith("Task list appears corrupted"): | 
					
						
							|  |  |  |                             continue | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                         elif msg.startswith( | 
					
						
							|  |  |  |                             "Invalid linked list structure reading remote memory" | 
					
						
							|  |  |  |                         ): | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                             continue | 
					
						
							|  |  |  |                         elif msg.startswith("Unknown error reading memory"): | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         elif msg.startswith("Unhandled frame owner"): | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         raise  # Unrecognized exception, safest not to ignore it | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                 # expected: a list of two elements: 1 thread, 1 interp | 
					
						
							|  |  |  |                 self.assertEqual(len(all_awaited_by), 2) | 
					
						
							|  |  |  |                 # expected: a tuple with the thread ID and the awaited_by list | 
					
						
							|  |  |  |                 self.assertEqual(len(all_awaited_by[0]), 2) | 
					
						
							|  |  |  |                 # expected: no tasks in the fallback per-interp task list | 
					
						
							|  |  |  |                 self.assertEqual(all_awaited_by[1], (0, [])) | 
					
						
							|  |  |  |                 entries = all_awaited_by[0][1] | 
					
						
							|  |  |  |                 # expected: at least 1000 pending tasks | 
					
						
							|  |  |  |                 self.assertGreaterEqual(len(entries), 1000) | 
					
						
							|  |  |  |                 # the first three tasks stem from the code structure | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 main_stack = [ | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                     FrameInfo([taskgroups.__file__, ANY, "TaskGroup._aexit"]), | 
					
						
							|  |  |  |                     FrameInfo( | 
					
						
							|  |  |  |                         [taskgroups.__file__, ANY, "TaskGroup.__aexit__"] | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     ), | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                     FrameInfo([script_name, 60, "main"]), | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 ] | 
					
						
							|  |  |  |                 self.assertIn( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                     TaskInfo( | 
					
						
							|  |  |  |                         [ANY, "Task-1", [CoroInfo([main_stack, ANY])], []] | 
					
						
							|  |  |  |                     ), | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     entries, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 self.assertIn( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                     TaskInfo( | 
					
						
							|  |  |  |                         [ | 
					
						
							|  |  |  |                             ANY, | 
					
						
							|  |  |  |                             "server task", | 
					
						
							|  |  |  |                             [ | 
					
						
							|  |  |  |                                 CoroInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     base_events.__file__, | 
					
						
							|  |  |  |                                                     ANY, | 
					
						
							|  |  |  |                                                     "Server.serve_forever", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ) | 
					
						
							|  |  |  |                                         ], | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ) | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                             [ | 
					
						
							|  |  |  |                                 CoroInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     taskgroups.__file__, | 
					
						
							|  |  |  |                                                     ANY, | 
					
						
							|  |  |  |                                                     "TaskGroup._aexit", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     taskgroups.__file__, | 
					
						
							|  |  |  |                                                     ANY, | 
					
						
							|  |  |  |                                                     "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [script_name, ANY, "main"] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                         ], | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ) | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                         ] | 
					
						
							|  |  |  |                     ), | 
					
						
							|  |  |  |                     entries, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 self.assertIn( | 
					
						
							|  |  |  |                     TaskInfo( | 
					
						
							|  |  |  |                         [ | 
					
						
							|  |  |  |                             ANY, | 
					
						
							|  |  |  |                             "Task-4", | 
					
						
							|  |  |  |                             [ | 
					
						
							|  |  |  |                                 CoroInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [tasks.__file__, ANY, "sleep"] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     script_name, | 
					
						
							|  |  |  |                                                     38, | 
					
						
							|  |  |  |                                                     "echo_client", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                         ], | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ) | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                             [ | 
					
						
							|  |  |  |                                 CoroInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         [ | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     taskgroups.__file__, | 
					
						
							|  |  |  |                                                     ANY, | 
					
						
							|  |  |  |                                                     "TaskGroup._aexit", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     taskgroups.__file__, | 
					
						
							|  |  |  |                                                     ANY, | 
					
						
							|  |  |  |                                                     "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                             FrameInfo( | 
					
						
							|  |  |  |                                                 [ | 
					
						
							|  |  |  |                                                     script_name, | 
					
						
							|  |  |  |                                                     41, | 
					
						
							|  |  |  |                                                     "echo_client_spam", | 
					
						
							|  |  |  |                                                 ] | 
					
						
							|  |  |  |                                             ), | 
					
						
							|  |  |  |                                         ], | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ) | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                         ] | 
					
						
							|  |  |  |                     ), | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     entries, | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 expected_awaited_by = [ | 
					
						
							|  |  |  |                     CoroInfo( | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                         [ | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                             [ | 
					
						
							|  |  |  |                                 FrameInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         taskgroups.__file__, | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                         "TaskGroup._aexit", | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                                 FrameInfo( | 
					
						
							|  |  |  |                                     [ | 
					
						
							|  |  |  |                                         taskgroups.__file__, | 
					
						
							|  |  |  |                                         ANY, | 
					
						
							|  |  |  |                                         "TaskGroup.__aexit__", | 
					
						
							|  |  |  |                                     ] | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                                 FrameInfo( | 
					
						
							|  |  |  |                                     [script_name, 41, "echo_client_spam"] | 
					
						
							|  |  |  |                                 ), | 
					
						
							|  |  |  |                             ], | 
					
						
							|  |  |  |                             ANY, | 
					
						
							|  |  |  |                         ] | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 ] | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 tasks_with_awaited = [ | 
					
						
							|  |  |  |                     task | 
					
						
							|  |  |  |                     for task in entries | 
					
						
							|  |  |  |                     if task.awaited_by == expected_awaited_by | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                 ] | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.assertGreaterEqual(len(tasks_with_awaited), 1000) | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                 # the final task will have some random number, but it should for | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 # sure be one of the echo client spam horde (In windows this is not true | 
					
						
							|  |  |  |                 # for some reason) | 
					
						
							|  |  |  |                 if sys.platform != "win32": | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     self.assertEqual( | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                         tasks_with_awaited[-1].awaited_by, | 
					
						
							|  |  |  |                         entries[-1].awaited_by, | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             except PermissionError: | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							| 
									
										
										
										
											2025-04-23 19:22:29 +02:00
										 |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-25 14:12:16 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  |     def test_self_trace(self): | 
					
						
							|  |  |  |         stack_trace = get_stack_trace(os.getpid()) | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |         # Is possible that there are more threads, so we check that the | 
					
						
							|  |  |  |         # expected stack traces are in the result (looking at you Windows!) | 
					
						
							|  |  |  |         this_tread_stack = None | 
					
						
							|  |  |  |         for thread_id, stack in stack_trace: | 
					
						
							|  |  |  |             if thread_id == threading.get_native_id(): | 
					
						
							|  |  |  |                 this_tread_stack = stack | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         self.assertIsNotNone(this_tread_stack) | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         self.assertEqual( | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |             stack[:2], | 
					
						
							|  |  |  |             [ | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 FrameInfo( | 
					
						
							|  |  |  |                     [ | 
					
						
							|  |  |  |                         __file__, | 
					
						
							|  |  |  |                         get_stack_trace.__code__.co_firstlineno + 2, | 
					
						
							|  |  |  |                         "get_stack_trace", | 
					
						
							|  |  |  |                     ] | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |                 FrameInfo( | 
					
						
							|  |  |  |                     [ | 
					
						
							|  |  |  |                         __file__, | 
					
						
							|  |  |  |                         self.test_self_trace.__code__.co_firstlineno + 6, | 
					
						
							|  |  |  |                         "TestGetStackTrace.test_self_trace", | 
					
						
							|  |  |  |                     ] | 
					
						
							| 
									
										
										
										
											2025-05-25 21:19:29 +01:00
										 |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2025-06-14 13:48:25 +01:00
										 |  |  |             ], | 
					
						
							| 
									
										
										
										
											2025-05-04 23:33:37 +02:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-28 14:11:31 +01:00
										 |  |  |     @skip_if_not_supported | 
					
						
							|  |  |  |     @unittest.skipIf( | 
					
						
							|  |  |  |         sys.platform == "linux" and not PROCESS_VM_READV_SUPPORTED, | 
					
						
							|  |  |  |         "Test only runs on Linux with process_vm_readv support", | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     @requires_gil_enabled("Free threaded builds don't have an 'active thread'") | 
					
						
							|  |  |  |     def test_only_active_thread(self): | 
					
						
							|  |  |  |         # Test that only_active_thread parameter works correctly | 
					
						
							|  |  |  |         port = find_unused_port() | 
					
						
							|  |  |  |         script = textwrap.dedent( | 
					
						
							|  |  |  |             f"""\
 | 
					
						
							|  |  |  |             import time, sys, socket, threading | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Connect to the test process | 
					
						
							|  |  |  |             sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             sock.connect(('localhost', {port})) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def worker_thread(name, barrier, ready_event): | 
					
						
							|  |  |  |                 barrier.wait()  # Synchronize thread start | 
					
						
							|  |  |  |                 ready_event.wait()  # Wait for main thread signal | 
					
						
							|  |  |  |                 # Sleep to keep thread alive | 
					
						
							|  |  |  |                 time.sleep(10_000) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             def main_work(): | 
					
						
							|  |  |  |                 # Do busy work to hold the GIL | 
					
						
							|  |  |  |                 sock.sendall(b"working\\n") | 
					
						
							|  |  |  |                 count = 0 | 
					
						
							|  |  |  |                 while count < 100000000: | 
					
						
							|  |  |  |                     count += 1 | 
					
						
							|  |  |  |                     if count % 10000000 == 0: | 
					
						
							|  |  |  |                         pass  # Keep main thread busy | 
					
						
							|  |  |  |                 sock.sendall(b"done\\n") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Create synchronization primitives | 
					
						
							|  |  |  |             num_threads = 3 | 
					
						
							|  |  |  |             barrier = threading.Barrier(num_threads + 1)  # +1 for main thread | 
					
						
							|  |  |  |             ready_event = threading.Event() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Start worker threads | 
					
						
							|  |  |  |             threads = [] | 
					
						
							|  |  |  |             for i in range(num_threads): | 
					
						
							|  |  |  |                 t = threading.Thread(target=worker_thread, args=(f"Worker-{{i}}", barrier, ready_event)) | 
					
						
							|  |  |  |                 t.start() | 
					
						
							|  |  |  |                 threads.append(t) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Wait for all threads to be ready | 
					
						
							|  |  |  |             barrier.wait() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Signal ready to parent process | 
					
						
							|  |  |  |             sock.sendall(b"ready\\n") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Signal threads to start waiting | 
					
						
							|  |  |  |             ready_event.set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Now do busy work to hold the GIL | 
					
						
							|  |  |  |             main_work() | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with os_helper.temp_dir() as work_dir: | 
					
						
							|  |  |  |             script_dir = os.path.join(work_dir, "script_pkg") | 
					
						
							|  |  |  |             os.mkdir(script_dir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Create a socket server to communicate with the target process | 
					
						
							|  |  |  |             server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
					
						
							|  |  |  |             server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) | 
					
						
							|  |  |  |             server_socket.bind(("localhost", port)) | 
					
						
							|  |  |  |             server_socket.settimeout(SHORT_TIMEOUT) | 
					
						
							|  |  |  |             server_socket.listen(1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             script_name = _make_test_script(script_dir, "script", script) | 
					
						
							|  |  |  |             client_socket = None | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 p = subprocess.Popen([sys.executable, script_name]) | 
					
						
							|  |  |  |                 client_socket, _ = server_socket.accept() | 
					
						
							|  |  |  |                 server_socket.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Wait for ready signal | 
					
						
							|  |  |  |                 response = b"" | 
					
						
							|  |  |  |                 while b"ready" not in response: | 
					
						
							|  |  |  |                     response += client_socket.recv(1024) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Wait for the main thread to start its busy work | 
					
						
							|  |  |  |                 while b"working" not in response: | 
					
						
							|  |  |  |                     response += client_socket.recv(1024) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Get stack trace with all threads | 
					
						
							|  |  |  |                 unwinder_all = RemoteUnwinder(p.pid, all_threads=True) | 
					
						
							| 
									
										
										
										
											2025-07-08 13:23:31 +01:00
										 |  |  |                 for _ in range(10): | 
					
						
							|  |  |  |                     # Wait for the main thread to start its busy work | 
					
						
							|  |  |  |                     all_traces = unwinder_all.get_stack_trace() | 
					
						
							|  |  |  |                     found = False | 
					
						
							|  |  |  |                     for thread_id, stack in all_traces: | 
					
						
							|  |  |  |                         if not stack: | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         current_frame = stack[0] | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                         if ( | 
					
						
							|  |  |  |                             current_frame.funcname == "main_work" | 
					
						
							|  |  |  |                             and current_frame.lineno > 15 | 
					
						
							|  |  |  |                         ): | 
					
						
							| 
									
										
										
										
											2025-07-08 13:23:31 +01:00
										 |  |  |                             found = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if found: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     # Give a bit of time to take the next sample | 
					
						
							|  |  |  |                     time.sleep(0.1) | 
					
						
							|  |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |                     self.fail( | 
					
						
							|  |  |  |                         "Main thread did not start its busy work on time" | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-06-28 14:11:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 # Get stack trace with only GIL holder | 
					
						
							|  |  |  |                 unwinder_gil = RemoteUnwinder(p.pid, only_active_thread=True) | 
					
						
							|  |  |  |                 gil_traces = unwinder_gil.get_stack_trace() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             except PermissionError: | 
					
						
							|  |  |  |                 self.skipTest( | 
					
						
							|  |  |  |                     "Insufficient permissions to read the stack trace" | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 if client_socket is not None: | 
					
						
							|  |  |  |                     client_socket.close() | 
					
						
							|  |  |  |                 p.kill() | 
					
						
							|  |  |  |                 p.terminate() | 
					
						
							|  |  |  |                 p.wait(timeout=SHORT_TIMEOUT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Verify we got multiple threads in all_traces | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             self.assertGreater( | 
					
						
							|  |  |  |                 len(all_traces), 1, "Should have multiple threads" | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-06-28 14:11:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Verify we got exactly one thread in gil_traces | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             self.assertEqual( | 
					
						
							|  |  |  |                 len(gil_traces), 1, "Should have exactly one GIL holder" | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-06-28 14:11:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # The GIL holder should be in the all_traces list | 
					
						
							|  |  |  |             gil_thread_id = gil_traces[0][0] | 
					
						
							|  |  |  |             all_thread_ids = [trace[0] for trace in all_traces] | 
					
						
							| 
									
										
										
										
											2025-07-10 00:11:17 +01:00
										 |  |  |             self.assertIn( | 
					
						
							|  |  |  |                 gil_thread_id, | 
					
						
							|  |  |  |                 all_thread_ids, | 
					
						
							|  |  |  |                 "GIL holder should be among all threads", | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-06-28 14:11:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 10:17:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     unittest.main() |