Skip to content

Commit 223c261

Browse files
authored
fix(profiling): call before_flush() before calling ddup.upload() (#11258)
## Checklist - [x] PR author has checked that all the criteria below are met - The PR description includes an overview of the change - The PR description articulates the motivation for the change - The change includes tests OR the PR description describes a testing strategy - The PR description notes risks associated with the change, if any - Newly-added code is easy to change - The change follows the [library release note guidelines](https://ddtrace.readthedocs.io/en/stable/releasenotes.html) - The change includes or references documentation updates if necessary - Backport labels are set (if [applicable](https://ddtrace.readthedocs.io/en/latest/contributing.html#backporting)) ## Reviewer Checklist - [x] Reviewer has checked that all the criteria below are met - Title is accurate - All changes are related to the pull request's stated goal - Avoids breaking [API](https://ddtrace.readthedocs.io/en/stable/versioning.html#interfaces) changes - Testing strategy adequately addresses listed risks - Newly-added code is easy to change - Release note makes sense to a user of the library - If necessary, author has acknowledged and discussed the performance implications of this PR as reported in the benchmarks PR comment - Backport labels are set in a manner that is consistent with the [release branch maintenance policy](https://ddtrace.readthedocs.io/en/latest/contributing.html#backporting)
1 parent 18653fa commit 223c261

File tree

4 files changed

+55
-15
lines changed

4 files changed

+55
-15
lines changed

ddtrace/profiling/collector/memalloc.py

+13-10
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
import os
55
import threading
66
import typing # noqa:F401
7+
from typing import Optional
78

89

910
try:
@@ -69,20 +70,22 @@ def __init__(
6970
self,
7071
recorder: Recorder,
7172
_interval: float = _DEFAULT_INTERVAL,
72-
_max_events: int = config.memory.events_buffer,
73-
max_nframe: int = config.max_frames,
74-
heap_sample_size: int = config.heap.sample_size,
75-
ignore_profiler: bool = config.ignore_profiler,
76-
_export_libdd_enabled: bool = config.export.libdd_enabled,
73+
_max_events: Optional[int] = None,
74+
max_nframe: Optional[int] = None,
75+
heap_sample_size: Optional[int] = None,
76+
ignore_profiler: Optional[bool] = None,
77+
_export_libdd_enabled: Optional[bool] = None,
7778
):
7879
super().__init__(recorder=recorder)
7980
self._interval: float = _interval
8081
# TODO make this dynamic based on the 1. interval and 2. the max number of events allowed in the Recorder
81-
self._max_events: int = _max_events
82-
self.max_nframe: int = max_nframe
83-
self.heap_sample_size: int = heap_sample_size
84-
self.ignore_profiler: bool = ignore_profiler
85-
self._export_libdd_enabled: bool = _export_libdd_enabled
82+
self._max_events: int = _max_events if _max_events is not None else config.memory.events_buffer
83+
self.max_nframe: int = max_nframe if max_nframe is not None else config.max_frames
84+
self.heap_sample_size: int = heap_sample_size if heap_sample_size is not None else config.heap.sample_size
85+
self.ignore_profiler: bool = ignore_profiler if ignore_profiler is not None else config.ignore_profiler
86+
self._export_libdd_enabled: bool = (
87+
_export_libdd_enabled if _export_libdd_enabled is not None else config.export.libdd_enabled
88+
)
8689

8790
def _start_service(self):
8891
# type: (...) -> None

ddtrace/profiling/scheduler.py

+6-5
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,12 @@ def flush(self):
5252
# type: (...) -> None
5353
"""Flush events from recorder to exporters."""
5454
LOG.debug("Flushing events")
55+
if self.before_flush is not None:
56+
try:
57+
self.before_flush()
58+
except Exception:
59+
LOG.error("Scheduler before_flush hook failed", exc_info=True)
60+
5561
if self._export_libdd_enabled:
5662
ddup.upload()
5763

@@ -61,11 +67,6 @@ def flush(self):
6167
self._last_export = compat.time_ns()
6268
return
6369

64-
if self.before_flush is not None:
65-
try:
66-
self.before_flush()
67-
except Exception:
68-
LOG.error("Scheduler before_flush hook failed", exc_info=True)
6970
events: EventsType = {}
7071
if self.recorder:
7172
events = self.recorder.reset()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
fixes:
3+
- |
4+
profiling: fixes an issue where enabling native exporter via
5+
``DD_PROFILING_EXPORT_LIBDD_ENABLED``, ``DD_PROFILING_TIMELINE_ENABLED``
6+
or ``DD_PROFILING_STACK_V2_ENABLED`` turned off live heap profiling.
7+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import os
2+
3+
from ddtrace.profiling import Profiler
4+
from ddtrace.settings.profiling import config
5+
from tests.profiling.collector import pprof_utils
6+
7+
8+
def _allocate_1k():
9+
return [object() for _ in range(1000)]
10+
11+
12+
def test_heap_samples_collected(tmp_path, monkeypatch):
13+
# Test for https://github.com/DataDog/dd-trace-py/issues/11069
14+
test_name = "test_heap"
15+
pprof_prefix = str(tmp_path / test_name)
16+
monkeypatch.setattr(config, "output_pprof", pprof_prefix)
17+
monkeypatch.setattr(config, "max_frames", 32)
18+
monkeypatch.setattr(config.memory, "events_buffer", 10)
19+
monkeypatch.setattr(config.heap, "sample_size", 1024)
20+
output_filename = pprof_prefix + "." + str(os.getpid())
21+
22+
p = Profiler()
23+
p.start()
24+
x = _allocate_1k() # noqa: F841
25+
p.stop()
26+
27+
profile = pprof_utils.parse_profile(output_filename)
28+
samples = pprof_utils.get_samples_with_value_type(profile, "heap-space")
29+
assert len(samples) > 0

0 commit comments

Comments
 (0)