Coverage for Lib/asyncio/base_tasks.py: 63%
71 statements
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-15 02:02 +0000
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-15 02:02 +0000
1import linecache
2import reprlib
3import traceback
5from . import base_futures
6from . import coroutines
9def _task_repr_info(task):
10 info = base_futures._future_repr_info(task)
12 if task.cancelling() and not task.done():
13 # replace status
14 info[0] = 'cancelling'
16 info.insert(1, 'name=%r' % task.get_name())
18 if task._fut_waiter is not None:
19 info.insert(2, f'wait_for={task._fut_waiter!r}')
21 if task._coro: 21 ↛ 25line 21 didn't jump to line 25 because the condition on line 21 was always true
22 coro = coroutines._format_coroutine(task._coro)
23 info.insert(2, f'coro=<{coro}>')
25 return info
28@reprlib.recursive_repr()
29def _task_repr(task):
30 info = ' '.join(_task_repr_info(task))
31 return f'<{task.__class__.__name__} {info}>'
34def _task_get_stack(task, limit):
35 frames = []
36 if hasattr(task._coro, 'cr_frame'): 36 ↛ 39line 36 didn't jump to line 39 because the condition on line 36 was always true
37 # case 1: 'async def' coroutines
38 f = task._coro.cr_frame
39 elif hasattr(task._coro, 'gi_frame'):
40 # case 2: legacy coroutines
41 f = task._coro.gi_frame
42 elif hasattr(task._coro, 'ag_frame'):
43 # case 3: async generators
44 f = task._coro.ag_frame
45 else:
46 # case 4: unknown objects
47 f = None
48 if f is not None: 48 ↛ 57line 48 didn't jump to line 57 because the condition on line 48 was always true
49 while f is not None: 49 ↛ 56line 49 didn't jump to line 56 because the condition on line 49 was always true
50 if limit is not None: 50 ↛ 54line 50 didn't jump to line 54 because the condition on line 50 was always true
51 if limit <= 0:
52 break
53 limit -= 1
54 frames.append(f)
55 f = f.f_back
56 frames.reverse()
57 elif task._exception is not None:
58 tb = task._exception.__traceback__
59 while tb is not None:
60 if limit is not None:
61 if limit <= 0:
62 break
63 limit -= 1
64 frames.append(tb.tb_frame)
65 tb = tb.tb_next
66 return frames
69def _task_print_stack(task, limit, file):
70 extracted_list = []
71 checked = set()
72 for f in task.get_stack(limit=limit):
73 lineno = f.f_lineno
74 co = f.f_code
75 filename = co.co_filename
76 name = co.co_name
77 if filename not in checked: 77 ↛ 80line 77 didn't jump to line 80 because the condition on line 77 was always true
78 checked.add(filename)
79 linecache.checkcache(filename)
80 line = linecache.getline(filename, lineno, f.f_globals)
81 extracted_list.append((filename, lineno, name, line))
83 exc = task._exception
84 if not extracted_list: 84 ↛ 85line 84 didn't jump to line 85 because the condition on line 84 was never true
85 print(f'No stack for {task!r}', file=file)
86 elif exc is not None: 86 ↛ 87line 86 didn't jump to line 87 because the condition on line 86 was never true
87 print(f'Traceback for {task!r} (most recent call last):', file=file)
88 else:
89 print(f'Stack for {task!r} (most recent call last):', file=file)
91 traceback.print_list(extracted_list, file=file)
92 if exc is not None: 92 ↛ 93line 92 didn't jump to line 93 because the condition on line 92 was never true
93 for line in traceback.format_exception_only(exc.__class__, exc):
94 print(line, file=file, end='')