Skip to content

Commit 05cf2cf

Browse files
committed
fixup! fixup! fixup! Add a new marker to check for memory leaks
1 parent b71d68d commit 05cf2cf

File tree

4 files changed

+44
-10
lines changed

4 files changed

+44
-10
lines changed

docs/api.rst

+3-2
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,10 @@ pytest-memray API
66
Types
77
-----
88

9-
.. autoclass:: StackElement
9+
.. autoclass:: StackFrame()
10+
:members:
1011

11-
.. autoclass:: Stack
12+
.. autoclass:: Stack()
1213
:members:
1314

1415
.. autoclass:: LeaksFilteringFunction

src/pytest_memray/__init__.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
from __future__ import annotations
22

33
from ._version import __version__ as __version__
4-
from .marks import StackElement
4+
from .marks import StackFrame
55
from .marks import Stack
66
from .marks import LeaksFilteringFunction
77

88
__all__ = [
99
"__version__",
1010
"Stack",
11-
"StackElement",
11+
"StackFrame",
1212
"LeaksFilteringFunction",
1313
]

src/pytest_memray/marks.py

+37-4
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,12 @@
22

33
from dataclasses import dataclass
44
from pathlib import Path
5+
from typing import Iterable
56
from typing import Tuple
67
from typing import cast
78
from typing import Callable
89
from typing import Optional
9-
from typing import Collection
10+
from typing import Sequence
1011

1112
from memray import AllocationRecord
1213
from memray import FileReader
@@ -17,12 +18,34 @@
1718
from .utils import value_or_ini
1819

1920
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."""
2138

2239

2340
@dataclass
2441
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."""
2649

2750

2851
LeaksFilteringFunction = Callable[[Stack], bool]
@@ -112,6 +135,16 @@ def long_repr(self) -> str:
112135
)
113136

114137

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+
115148
def limit_memory(
116149
limit: str, *, _result_file: Path, _config: Config
117150
) -> _MemoryInfo | None:
@@ -148,7 +181,7 @@ def limit_leaks(
148181
for allocation in allocations
149182
if (
150183
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)
152185
)
153186
)
154187
if not leaked_allocations:

tests/test_pytest_memray.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -671,8 +671,8 @@ def this_should_not_be_there():
671671
# No free call here
672672
673673
def filtering_function(stack):
674-
for fn, _, _ in stack.frames:
675-
if fn == "this_should_not_be_there":
674+
for frame in stack.frames:
675+
if frame.function == "this_should_not_be_there":
676676
return False
677677
return True
678678

0 commit comments

Comments
 (0)