|
2 | 2 |
|
3 | 3 | from dataclasses import dataclass
|
4 | 4 | from pathlib import Path
|
| 5 | +from typing import Iterable |
5 | 6 | from typing import Tuple
|
6 | 7 | from typing import cast
|
7 | 8 | from typing import Callable
|
8 | 9 | from typing import Optional
|
9 |
| -from typing import Collection |
| 10 | +from typing import Sequence |
10 | 11 |
|
11 | 12 | from memray import AllocationRecord
|
12 | 13 | from memray import FileReader
|
|
17 | 18 | from .utils import value_or_ini
|
18 | 19 |
|
19 | 20 | PytestSection = Tuple[str, str]
|
20 |
| -StackElement = Tuple[str, str, int] |
| 21 | + |
| 22 | + |
| 23 | +@dataclass |
| 24 | +class StackFrame: |
| 25 | + """One frame of a call stack. |
| 26 | +
|
| 27 | + Each frame has attributes to tell you what code was executing. |
| 28 | + """ |
| 29 | + |
| 30 | + function: str |
| 31 | + """The function being executed, or ``"???"`` if unknown.""" |
| 32 | + |
| 33 | + filename: str |
| 34 | + """The source file being executed, or ``"???"`` if unknown.""" |
| 35 | + |
| 36 | + lineno: int |
| 37 | + """The line number of the executing line, or ``0`` if unknown.""" |
21 | 38 |
|
22 | 39 |
|
23 | 40 | @dataclass
|
24 | 41 | class Stack:
|
25 |
| - frames: Collection[StackElement] |
| 42 | + """The call stack that led to some memory allocation. |
| 43 | +
|
| 44 | + You can inspect the frames which make up the call stack. |
| 45 | + """ |
| 46 | + |
| 47 | + frames: Sequence[StackFrame] |
| 48 | + """The frames that make up the call stack, most recent first.""" |
26 | 49 |
|
27 | 50 |
|
28 | 51 | LeaksFilteringFunction = Callable[[Stack], bool]
|
@@ -112,6 +135,16 @@ def long_repr(self) -> str:
|
112 | 135 | )
|
113 | 136 |
|
114 | 137 |
|
| 138 | +def passes_filter( |
| 139 | + stack: Iterable[Tuple[str, str, int]], filter_fn: Optional[LeaksFilteringFunction] |
| 140 | +) -> bool: |
| 141 | + if filter_fn is None: |
| 142 | + return True |
| 143 | + |
| 144 | + stack_elements = [StackFrame(*frame) for frame in stack] |
| 145 | + return filter_fn(Stack(stack_elements)) |
| 146 | + |
| 147 | + |
115 | 148 | def limit_memory(
|
116 | 149 | limit: str, *, _result_file: Path, _config: Config
|
117 | 150 | ) -> _MemoryInfo | None:
|
@@ -148,7 +181,7 @@ def limit_leaks(
|
148 | 181 | for allocation in allocations
|
149 | 182 | if (
|
150 | 183 | allocation.size >= memory_limit
|
151 |
| - and (filter_fn is None or filter_fn(Stack(allocation.hybrid_stack_trace()))) |
| 184 | + and passes_filter(allocation.hybrid_stack_trace(), filter_fn) |
152 | 185 | )
|
153 | 186 | )
|
154 | 187 | if not leaked_allocations:
|
|
0 commit comments