Skip to content

Commit 300bb6f

Browse files
authored
Limit cases where SQL origin is captured (#881)
* Limit backtraces for origin to 20 frames * Implement threshold for collecting sql origins * Add tests * Typo * Refactor resolving event origin to unify code
1 parent 3fd21f9 commit 300bb6f

File tree

6 files changed

+73
-35
lines changed

6 files changed

+73
-35
lines changed

config/sentry.php

+3
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,9 @@
8888
// Capture where the SQL query originated from on the SQL query spans
8989
'sql_origin' => env('SENTRY_TRACE_SQL_ORIGIN_ENABLED', true),
9090

91+
// Define a threshold in milliseconds for SQL queries to resolve their origin
92+
'sql_origin_threshold_ms' => env('SENTRY_TRACE_SQL_ORIGIN_THRESHOLD_MS', 100),
93+
9194
// Capture views rendered as spans
9295
'views' => env('SENTRY_TRACE_VIEWS_ENABLED', true),
9396

src/Sentry/Laravel/Features/CacheIntegration.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,7 @@ public function handleRedisCommand(RedisEvents\CommandExecuted $event): void
108108
$commandOrigin = $this->resolveEventOrigin();
109109

110110
if ($commandOrigin !== null) {
111-
$data['db.redis.origin'] = $commandOrigin;
111+
$data = array_merge($data, $commandOrigin);
112112
}
113113
}
114114

src/Sentry/Laravel/Features/Concerns/ResolvesEventOrigin.php

+19-3
Original file line numberDiff line numberDiff line change
@@ -12,19 +12,35 @@ protected function container(): Container
1212
return app();
1313
}
1414

15-
protected function resolveEventOrigin(): ?string
15+
protected function resolveEventOrigin(): ?array
1616
{
1717
$backtraceHelper = $this->makeBacktraceHelper();
1818

19-
$firstAppFrame = $backtraceHelper->findFirstInAppFrameForBacktrace(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS));
19+
// We limit the backtrace to 20 frames to prevent too much overhead and we'd reasonable expect the origin to be within the first 20 frames
20+
$firstAppFrame = $backtraceHelper->findFirstInAppFrameForBacktrace(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 20));
2021

2122
if ($firstAppFrame === null) {
2223
return null;
2324
}
2425

2526
$filePath = $backtraceHelper->getOriginalViewPathForFrameOfCompiledViewPath($firstAppFrame) ?? $firstAppFrame->getFile();
2627

27-
return "{$filePath}:{$firstAppFrame->getLine()}";
28+
return [
29+
'code.filepath' => $filePath,
30+
'code.function' => $firstAppFrame->getFunctionName(),
31+
'code.lineno' => $firstAppFrame->getLine(),
32+
];
33+
}
34+
35+
protected function resolveEventOriginAsString(): ?string
36+
{
37+
$origin = $this->resolveEventOrigin();
38+
39+
if ($origin === null) {
40+
return null;
41+
}
42+
43+
return "{$origin['code.filepath']}:{$origin['code.lineno']}";
2844
}
2945

3046
private function makeBacktraceHelper(): BacktraceHelper

src/Sentry/Laravel/Integration.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -322,7 +322,7 @@ public function __invoke(Model $model, string $relation): void
322322
$scope->setContext('violation', [
323323
'model' => get_class($model),
324324
'relation' => $relation,
325-
'origin' => $this->resolveEventOrigin(),
325+
'origin' => $this->resolveEventOriginAsString(),
326326
'kind' => 'lazy_loading',
327327
]);
328328

src/Sentry/Laravel/Tracing/EventHandler.php

+12-28
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,14 @@ class EventHandler
5353
*
5454
* @var bool
5555
*/
56-
private $traceSqlQueryOrigins;
56+
private $traceSqlQueryOrigin;
57+
58+
/**
59+
* The threshold in milliseconds to consider a SQL query origin.
60+
*
61+
* @var int
62+
*/
63+
private $traceSqlQueryOriginTreshHoldMs;
5764

5865
/**
5966
* Indicates if we should trace queue job spans.
@@ -90,7 +97,8 @@ public function __construct(array $config)
9097
{
9198
$this->traceSqlQueries = ($config['sql_queries'] ?? true) === true;
9299
$this->traceSqlBindings = ($config['sql_bindings'] ?? true) === true;
93-
$this->traceSqlQueryOrigins = ($config['sql_origin'] ?? true) === true;
100+
$this->traceSqlQueryOrigin = ($config['sql_origin'] ?? true) === true;
101+
$this->traceSqlQueryOriginTreshHoldMs = $config['sql_origin_threshold_ms'] ?? 100;
94102

95103
$this->traceQueueJobs = ($config['queue_jobs'] ?? false) === true;
96104
$this->traceQueueJobsAsTransactions = ($config['queue_job_transactions'] ?? false) === true;
@@ -180,8 +188,8 @@ protected function queryExecutedHandler(DatabaseEvents\QueryExecuted $query): vo
180188
]));
181189
}
182190

183-
if ($this->traceSqlQueryOrigins) {
184-
$queryOrigin = $this->resolveQueryOriginFromBacktrace();
191+
if ($this->traceSqlQueryOrigin && $query->time >= $this->traceSqlQueryOriginTreshHoldMs) {
192+
$queryOrigin = $this->resolveEventOrigin();
185193

186194
if ($queryOrigin !== null) {
187195
$context->setData(array_merge($context->getData(), $queryOrigin));
@@ -191,30 +199,6 @@ protected function queryExecutedHandler(DatabaseEvents\QueryExecuted $query): vo
191199
$parentSpan->startChild($context);
192200
}
193201

194-
/**
195-
* Try to find the origin of the SQL query that was just executed.
196-
*
197-
* @return string|null
198-
*/
199-
private function resolveQueryOriginFromBacktrace(): ?array
200-
{
201-
$backtraceHelper = $this->makeBacktraceHelper();
202-
203-
$firstAppFrame = $backtraceHelper->findFirstInAppFrameForBacktrace(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS));
204-
205-
if ($firstAppFrame === null) {
206-
return null;
207-
}
208-
209-
$filePath = $backtraceHelper->getOriginalViewPathForFrameOfCompiledViewPath($firstAppFrame) ?? $firstAppFrame->getFile();
210-
211-
return [
212-
'code.filepath' => $filePath,
213-
'code.function' => $firstAppFrame->getFunctionName(),
214-
'code.lineno' => $firstAppFrame->getLine(),
215-
];
216-
}
217-
218202
protected function responsePreparedHandler(RoutingEvents\ResponsePrepared $event): void
219203
{
220204
$span = $this->popSpan();

test/Sentry/Features/DatabaseIntegrationTest.php

+37-2
Original file line numberDiff line numberDiff line change
@@ -115,11 +115,46 @@ public function testSqlBindingsAreRecordedWhenDisabled(): void
115115
$this->assertFalse(isset($span->getData()['db.sql.bindings']));
116116
}
117117

118-
private function executeQueryAndRetrieveSpan(string $query, array $bindings = []): Span
118+
public function testSqlOriginIsResolvedWhenEnabledAndOverTreshold(): void
119+
{
120+
$this->resetApplicationWithConfig([
121+
'sentry.tracing.sql_origin' => true,
122+
'sentry.tracing.sql_origin_threshold_ms' => 10,
123+
]);
124+
125+
$span = $this->executeQueryAndRetrieveSpan('SELECT 1', [], 20);
126+
127+
$this->assertArrayHasKey('code.filepath', $span->getData());
128+
}
129+
130+
public function testSqlOriginIsNotResolvedWhenDisabled(): void
131+
{
132+
$this->resetApplicationWithConfig([
133+
'sentry.tracing.sql_origin' => false,
134+
]);
135+
136+
$span = $this->executeQueryAndRetrieveSpan('SELECT 1');
137+
138+
$this->assertArrayNotHasKey('code.filepath', $span->getData());
139+
}
140+
141+
public function testSqlOriginIsNotResolvedWhenUnderThreshold(): void
142+
{
143+
$this->resetApplicationWithConfig([
144+
'sentry.tracing.sql_origin' => true,
145+
'sentry.tracing.sql_origin_threshold_ms' => 10,
146+
]);
147+
148+
$span = $this->executeQueryAndRetrieveSpan('SELECT 1', [], 5);
149+
150+
$this->assertArrayNotHasKey('code.filepath', $span->getData());
151+
}
152+
153+
private function executeQueryAndRetrieveSpan(string $query, array $bindings = [], int $time = 123): Span
119154
{
120155
$transaction = $this->startTransaction();
121156

122-
$this->dispatchLaravelEvent(new QueryExecuted($query, $bindings, 123, DB::connection()));
157+
$this->dispatchLaravelEvent(new QueryExecuted($query, $bindings, $time, DB::connection()));
123158

124159
$spans = $transaction->getSpanRecorder()->getSpans();
125160

0 commit comments

Comments
 (0)