Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions packages/core/src/asyncContext/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import type { Span } from '../types/span';
import type { getTraceData } from '../utils/traceData';
import type {
continueTrace,
isTracingSuppressed,
startInactiveSpan,
startNewTrace,
startSpan,
Expand Down Expand Up @@ -86,6 +87,9 @@ export interface AsyncContextStrategy {
/** Suppress tracing in the given callback, ensuring no spans are generated inside of it. */
suppressTracing?: typeof suppressTracing;

/** If tracing is suppressed in the given scope. */
isTracingSuppressed?: typeof isTracingSuppressed;

/** Get trace data as serialized string values for propagation via `sentry-trace` and `baggage`. */
getTraceData?: typeof getTraceData;

Expand Down
1 change: 1 addition & 0 deletions packages/core/src/tracing/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ export {
continueTrace,
withActiveSpan,
suppressTracing,
isTracingSuppressed,
startNewTrace,
SUPPRESS_TRACING_KEY,
} from './trace';
Expand Down
29 changes: 18 additions & 11 deletions packages/core/src/tracing/trace.ts
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,17 @@ export function suppressTracing<T>(callback: () => T): T {
});
}

/** Check if tracing is suppressed. */
export function isTracingSuppressed(scope = getCurrentScope()): boolean {
const acs = getAcs();

if (acs.isTracingSuppressed) {
return acs.isTracingSuppressed(scope);
}

return scope.getScopeData().sdkProcessingMetadata[SUPPRESS_TRACING_KEY] === true;
}

/**
* Starts a new trace for the duration of the provided callback. Spans started within the
* callback will be part of the new trace instead of a potentially previously started trace.
Expand Down Expand Up @@ -372,7 +383,7 @@ function createChildOrRootSpan({

const client = getClient();
if (_shouldIgnoreStreamedSpan(client, spanArguments)) {
if (!_isTracingSuppressed(scope)) {
if (!isTracingSuppressed(scope)) {
// if tracing is actively suppressed (Sentry.suppressTracing(...)),
// we don't want to record a client outcome for the ignored span
client?.recordDroppedEvent('ignored', 'span');
Expand Down Expand Up @@ -489,9 +500,9 @@ function _startRootSpan(
const finalAttributes = mutableSpanSamplingData.spanAttributes;

const currentPropagationContext = scope.getPropagationContext();
const isTracingSuppressed = _isTracingSuppressed(scope);
const _isTracingSuppressed = isTracingSuppressed(scope);

const [sampled, sampleRate, localSampleRateWasApplied] = isTracingSuppressed
const [sampled, sampleRate, localSampleRateWasApplied] = _isTracingSuppressed
? [false]
: sampleSpan(
options,
Expand All @@ -515,7 +526,7 @@ function _startRootSpan(
sampled,
});

if (!sampled && client && !isTracingSuppressed) {
if (!sampled && client && !_isTracingSuppressed) {
DEBUG_BUILD && debug.log('[Tracing] Discarding root span because its trace was not chosen to be sampled.');
client.recordDroppedEvent('sample_rate', hasSpanStreamingEnabled(client) ? 'span' : 'transaction');
}
Expand All @@ -540,8 +551,8 @@ function _startChildSpan(
isolationScope: Scope,
): Span {
const { spanId, traceId } = parentSpan.spanContext();
const isTracingSuppressed = _isTracingSuppressed(scope);
const sampled = isTracingSuppressed ? false : spanIsSampled(parentSpan);
const _isTracingSuppressed = isTracingSuppressed(scope);
const sampled = _isTracingSuppressed ? false : spanIsSampled(parentSpan);

const childSpan = sampled
? new SentrySpan({
Expand Down Expand Up @@ -569,7 +580,7 @@ function _startChildSpan(
// record a client outcome for the child.
childSpan.dropReason = parentSpan.dropReason;
client.recordDroppedEvent(parentSpan.dropReason, 'span');
} else if (!isTracingSuppressed) {
} else if (!_isTracingSuppressed) {
// Otherwise, the child is not sampled due to sampling of the parent span,
// hence we record a sample_rate client outcome for the child.
childSpan.dropReason = 'sample_rate';
Expand Down Expand Up @@ -642,7 +653,3 @@ function _shouldIgnoreStreamedSpan(client: Client | undefined, spanArguments: Se
function _isIgnoredSpan(span: Span): span is SentryNonRecordingSpan {
return spanIsNonRecordingSpan(span) && span.dropReason === 'ignored';
}

function _isTracingSuppressed(scope: Scope): boolean {
return scope.getScopeData().sdkProcessingMetadata[SUPPRESS_TRACING_KEY] === true;
}
56 changes: 56 additions & 0 deletions packages/core/test/lib/tracing/trace.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,13 @@ import { getAsyncContextStrategy } from '../../../src/asyncContext';
import {
continueTrace,
getDynamicSamplingContextFromSpan,
isTracingSuppressed,
registerSpanErrorInstrumentation,
SentrySpan,
startInactiveSpan,
startSpan,
startSpanManual,
SUPPRESS_TRACING_KEY,
suppressTracing,
withActiveSpan,
} from '../../../src/tracing';
Expand Down Expand Up @@ -2561,6 +2563,60 @@ describe('suppressTracing', () => {
});
});

describe('isTracingSuppressed', () => {
beforeEach(() => {
getCurrentScope().clear();
getIsolationScope().clear();
getGlobalScope().clear();

setAsyncContextStrategy(undefined);

const options = getDefaultTestClientOptions({ tracesSampleRate: 1 });
client = new TestClient(options);
setCurrentClient(client);
client.init();
});

afterEach(() => {
vi.clearAllMocks();
});

it('returns false when tracing is not suppressed', () => {
expect(isTracingSuppressed()).toBe(false);
});

it('returns true while inside suppressTracing', () => {
const suppressed = suppressTracing(() => isTracingSuppressed());
expect(suppressed).toBe(true);
});

it('returns false again after suppressTracing has finished', () => {
suppressTracing(() => {
expect(isTracingSuppressed()).toBe(true);
});

expect(isTracingSuppressed()).toBe(false);
});

it('only suppresses tracing within the active scope', () => {
withScope(() => {
const suppressed = suppressTracing(() => isTracingSuppressed());
expect(suppressed).toBe(true);
});

// Outside of the suppressed scope, tracing is no longer suppressed
expect(isTracingSuppressed()).toBe(false);
});

it('respects a scope passed in explicitly', () => {
const scope = getCurrentScope().clone();
scope.setSDKProcessingMetadata({ [SUPPRESS_TRACING_KEY]: true });

expect(isTracingSuppressed(scope)).toBe(true);
expect(isTracingSuppressed()).toBe(false);
});
});

describe('startNewTrace', () => {
beforeEach(() => {
getCurrentScope().clear();
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { subscribe } from 'node:diagnostics_channel';
import { context, trace } from '@opentelemetry/api';
import { isTracingSuppressed } from '@opentelemetry/core';
import type { InstrumentationConfig } from '@opentelemetry/instrumentation';
import { InstrumentationBase, InstrumentationNodeModuleDefinition } from '@opentelemetry/instrumentation';
import type { ClientRequest, IncomingMessage, ServerResponse } from 'node:http';
Expand All @@ -11,7 +10,13 @@ import type {
HttpModuleExport,
Span,
} from '@sentry/core';
import { getHttpClientSubscriptions, patchHttpModuleClient, SDK_VERSION, getRequestOptions } from '@sentry/core';
import {
getHttpClientSubscriptions,
patchHttpModuleClient,
SDK_VERSION,
getRequestOptions,
isTracingSuppressed,
} from '@sentry/core';
import { INSTRUMENTATION_NAME } from './constants';
import { HTTP_ON_CLIENT_REQUEST } from '@sentry/core';
import { NODE_VERSION } from '../../nodeVersion';
Expand Down Expand Up @@ -172,8 +177,7 @@ export class SentryHttpInstrumentation extends InstrumentationBase<SentryHttpIns
spans: options.createSpansForOutgoingRequests && (options.spans ?? true),
ignoreOutgoingRequests(url, request) {
return (
isTracingSuppressed(context.active()) ||
!!options.ignoreOutgoingRequests?.(url, getRequestOptions(request as ClientRequest))
isTracingSuppressed() || !!options.ignoreOutgoingRequests?.(url, getRequestOptions(request as ClientRequest))
);
},
outgoingRequestHook(span, request) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { errorMonitor } from 'node:events';
import type { IncomingHttpHeaders } from 'node:http';
import { context, SpanKind, trace } from '@opentelemetry/api';
import type { RPCMetadata } from '@opentelemetry/core';
import { getRPCMetadata, isTracingSuppressed, RPCType, setRPCMetadata } from '@opentelemetry/core';
import { getRPCMetadata, RPCType, setRPCMetadata } from '@opentelemetry/core';
import {
HTTP_RESPONSE_STATUS_CODE,
HTTP_ROUTE,
Expand Down Expand Up @@ -32,6 +32,7 @@ import {
SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN,
SPAN_STATUS_ERROR,
stripUrlQueryAndFragment,
isTracingSuppressed,
} from '@sentry/core';
import { DEBUG_BUILD } from '../../debug-build';
import type { NodeClient } from '../../sdk/client';
Expand Down Expand Up @@ -306,7 +307,7 @@ function shouldIgnoreSpansForIncomingRequest(
ignoreIncomingRequests?: (urlPath: string, request: HttpIncomingMessage) => boolean;
},
): boolean {
if (isTracingSuppressed(context.active())) {
if (isTracingSuppressed()) {
return true;
}

Expand Down
Original file line number Diff line number Diff line change
@@ -1,8 +1,6 @@
import { context } from '@opentelemetry/api';
import { isTracingSuppressed } from '@opentelemetry/core';
import type { InstrumentationConfig } from '@opentelemetry/instrumentation';
import { InstrumentationBase } from '@opentelemetry/instrumentation';
import { LRUMap, SDK_VERSION } from '@sentry/core';
import { LRUMap, SDK_VERSION, isTracingSuppressed } from '@sentry/core';
import * as diagch from 'diagnostics_channel';
import { NODE_MAJOR, NODE_MINOR } from '../../nodeVersion';
import {
Expand Down Expand Up @@ -184,7 +182,7 @@ export class SentryNodeFetchInstrumentation extends InstrumentationBase<SentryNo
* Check if the given outgoing request should be ignored.
*/
private _shouldIgnoreOutgoingRequest(request: UndiciRequest): boolean {
if (isTracingSuppressed(context.active())) {
if (isTracingSuppressed()) {
return true;
}

Expand Down
3 changes: 2 additions & 1 deletion packages/opentelemetry/src/asyncContextStrategy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ import type { CurrentScopes } from './types';
import { getContextFromScope, getScopesFromContext } from './utils/contextData';
import { getActiveSpan } from './utils/getActiveSpan';
import { getTraceData } from './utils/getTraceData';
import { suppressTracing } from './utils/suppressTracing';
import { suppressTracing, isTracingSuppressed } from './utils/suppressTracing';

interface ContextApi {
_getContextManager(): {
Expand Down Expand Up @@ -110,6 +110,7 @@ export function setOpenTelemetryContextAsyncContextStrategy(): void {
startInactiveSpan,
getActiveSpan,
suppressTracing,
isTracingSuppressed,
getTraceData,
continueTrace,
startNewTrace,
Expand Down
12 changes: 11 additions & 1 deletion packages/opentelemetry/src/utils/suppressTracing.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,18 @@
import { context } from '@opentelemetry/api';
import { suppressTracing as suppressTracingImpl } from '@opentelemetry/core';
import {
suppressTracing as suppressTracingImpl,
isTracingSuppressed as isTracingSuppressedImpl,
} from '@opentelemetry/core';
import type { Scope } from '@sentry/core';
import { getContextFromScope } from './contextData';

/** Suppress tracing in the given callback, ensuring no spans are generated inside of it. */
export function suppressTracing<T>(callback: () => T): T {
const ctx = suppressTracingImpl(context.active());
return context.with(ctx, callback);
}

export function isTracingSuppressed(scope?: Scope): boolean {
const ctx = scope ? getContextFromScope(scope) : context.active();
return ctx ? isTracingSuppressedImpl(ctx) : false;

Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Missing active context fallback

Medium Severity

The OpenTelemetry isTracingSuppressed helper returns false when getContextFromScope(scope) is missing, instead of falling back to context.active(). Node HTTP/fetch instrumentation now uses this path rather than checking the active OTel context directly, so suppression can be missed and outgoing spans or trace headers may still be created.

Additional Locations (2)
Fix in Cursor Fix in Web

Reviewed by Cursor Bugbot for commit 620084c. Configure here.

Copy link
Copy Markdown
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IMHO this is the safer approach, if that is missing.

Comment thread
sentry[bot] marked this conversation as resolved.
}
37 changes: 37 additions & 0 deletions packages/opentelemetry/test/trace.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import {
getDynamicSamplingContextFromClient,
getDynamicSamplingContextFromSpan,
getRootSpan,
isTracingSuppressed,
SEMANTIC_ATTRIBUTE_SENTRY_OP,
SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN,
SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE,
Expand Down Expand Up @@ -2107,6 +2108,42 @@ describe('suppressTracing', () => {
});
});

describe('isTracingSuppressed', () => {
beforeEach(() => {
mockSdkInit({ tracesSampleRate: 1 });
});

afterEach(async () => {
await cleanupOtel();
});

it('returns false when tracing is not suppressed', () => {
expect(isTracingSuppressed()).toBe(false);
});

it('returns true while inside suppressTracing', () => {
const suppressed = suppressTracing(() => isTracingSuppressed());
expect(suppressed).toBe(true);
});

it('returns false again after suppressTracing has finished', () => {
suppressTracing(() => {
expect(isTracingSuppressed()).toBe(true);
});

expect(isTracingSuppressed()).toBe(false);
});

it('stays suppressed across async boundaries within suppressTracing', async () => {
const suppressed = await suppressTracing(async () => {
await new Promise(resolve => setTimeout(resolve, 10));
return isTracingSuppressed();
});

expect(suppressed).toBe(true);
});
});

describe('span.end() timestamp conversion', () => {
beforeEach(() => {
mockSdkInit({ tracesSampleRate: 1 });
Expand Down
Loading