Skip to content

Commit 05391d0

Browse files
authored
feat(node): Only add span listeners for instrumentation when used (#15802)
Related to #15725 (comment), this PR is an idea to avoid registering `on('spanStart')` hooks for all the node libraries that do not have proper hooks, unless they are used. Today, for OTEL instrumentation that does not provide span hooks, we fall back to `client.on('spanStart')` to enhance/mutate spans emitted by the instrumentation. This PR improved this by only registering the `spanStart` client hook if we detect that the OTEL instrumentation is actually wrapping something. This avoids us calling a bunch of span callbacks each time a span is started, when it is not even needed. For this, a new `instrumentWhenWrapped` utility is added which can be passed an instrumentation. This works by monkey-patching `_wrap` on the instrumentation, and ensuring a callback is only called conditionally. Note: For some (e.g. connect, fastify) we register `spanStart` in the error handler, which is even better, so there we do not need this logic.
1 parent c38315e commit 05391d0

File tree

10 files changed

+403
-206
lines changed

10 files changed

+403
-206
lines changed

packages/node/src/integrations/tracing/connect.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ function addConnectSpanAttributes(span: Span): void {
104104
[SEMANTIC_ATTRIBUTE_SENTRY_OP]: `${type}.connect`,
105105
});
106106

107-
// Also update the name, we don't need to "middleware - " prefix
107+
// Also update the name, we don't need the "middleware - " prefix
108108
const name = attributes['connect.name'];
109109
if (typeof name === 'string') {
110110
span.updateName(name);

packages/node/src/integrations/tracing/dataloader.ts

+25-19
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import {
66
spanToJSON,
77
} from '@sentry/core';
88
import type { IntegrationFn } from '@sentry/core';
9-
import { generateInstrumentOnce } from '../../otel/instrument';
9+
import { instrumentWhenWrapped, generateInstrumentOnce } from '../../otel/instrument';
1010

1111
const INTEGRATION_NAME = 'Dataloader';
1212

@@ -19,31 +19,37 @@ export const instrumentDataloader = generateInstrumentOnce(
1919
);
2020

2121
const _dataloaderIntegration = (() => {
22+
let instrumentationWrappedCallback: undefined | ((callback: () => void) => void);
23+
2224
return {
2325
name: INTEGRATION_NAME,
2426
setupOnce() {
25-
instrumentDataloader();
27+
const instrumentation = instrumentDataloader();
28+
instrumentationWrappedCallback = instrumentWhenWrapped(instrumentation);
2629
},
2730

2831
setup(client) {
29-
client.on('spanStart', span => {
30-
const spanJSON = spanToJSON(span);
31-
if (spanJSON.description?.startsWith('dataloader')) {
32-
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.dataloader');
33-
}
32+
// This is called either immediately or when the instrumentation is wrapped
33+
instrumentationWrappedCallback?.(() => {
34+
client.on('spanStart', span => {
35+
const spanJSON = spanToJSON(span);
36+
if (spanJSON.description?.startsWith('dataloader')) {
37+
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.dataloader');
38+
}
3439

35-
// These are all possible dataloader span descriptions
36-
// Still checking for the future versions
37-
// in case they add support for `clear` and `prime`
38-
if (
39-
spanJSON.description === 'dataloader.load' ||
40-
spanJSON.description === 'dataloader.loadMany' ||
41-
spanJSON.description === 'dataloader.batch'
42-
) {
43-
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_OP, 'cache.get');
44-
// TODO: We can try adding `key` to the `data` attribute upstream.
45-
// Or alternatively, we can add `requestHook` to the dataloader instrumentation.
46-
}
40+
// These are all possible dataloader span descriptions
41+
// Still checking for the future versions
42+
// in case they add support for `clear` and `prime`
43+
if (
44+
spanJSON.description === 'dataloader.load' ||
45+
spanJSON.description === 'dataloader.loadMany' ||
46+
spanJSON.description === 'dataloader.batch'
47+
) {
48+
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_OP, 'cache.get');
49+
// TODO: We can try adding `key` to the `data` attribute upstream.
50+
// Or alternatively, we can add `requestHook` to the dataloader instrumentation.
51+
}
52+
});
4753
});
4854
},
4955
};

packages/node/src/integrations/tracing/genericPool.ts

+17-12
Original file line numberDiff line numberDiff line change
@@ -1,33 +1,38 @@
11
import { GenericPoolInstrumentation } from '@opentelemetry/instrumentation-generic-pool';
22
import { SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, defineIntegration, spanToJSON } from '@sentry/core';
33
import type { IntegrationFn } from '@sentry/core';
4-
import { generateInstrumentOnce } from '../../otel/instrument';
4+
import { generateInstrumentOnce, instrumentWhenWrapped } from '../../otel/instrument';
55

66
const INTEGRATION_NAME = 'GenericPool';
77

88
export const instrumentGenericPool = generateInstrumentOnce(INTEGRATION_NAME, () => new GenericPoolInstrumentation({}));
99

1010
const _genericPoolIntegration = (() => {
11+
let instrumentationWrappedCallback: undefined | ((callback: () => void) => void);
12+
1113
return {
1214
name: INTEGRATION_NAME,
1315
setupOnce() {
14-
instrumentGenericPool();
16+
const instrumentation = instrumentGenericPool();
17+
instrumentationWrappedCallback = instrumentWhenWrapped(instrumentation);
1518
},
1619

1720
setup(client) {
18-
client.on('spanStart', span => {
19-
const spanJSON = spanToJSON(span);
21+
instrumentationWrappedCallback?.(() =>
22+
client.on('spanStart', span => {
23+
const spanJSON = spanToJSON(span);
2024

21-
const spanDescription = spanJSON.description;
25+
const spanDescription = spanJSON.description;
2226

23-
// typo in emitted span for version <= 0.38.0 of @opentelemetry/instrumentation-generic-pool
24-
const isGenericPoolSpan =
25-
spanDescription === 'generic-pool.aquire' || spanDescription === 'generic-pool.acquire';
27+
// typo in emitted span for version <= 0.38.0 of @opentelemetry/instrumentation-generic-pool
28+
const isGenericPoolSpan =
29+
spanDescription === 'generic-pool.aquire' || spanDescription === 'generic-pool.acquire';
2630

27-
if (isGenericPoolSpan) {
28-
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.generic_pool');
29-
}
30-
});
31+
if (isGenericPoolSpan) {
32+
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.generic_pool');
33+
}
34+
}),
35+
);
3136
},
3237
};
3338
}) satisfies IntegrationFn;

packages/node/src/integrations/tracing/knex.ts

+15-10
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { KnexInstrumentation } from '@opentelemetry/instrumentation-knex';
22
import { SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, defineIntegration, spanToJSON } from '@sentry/core';
33
import type { IntegrationFn } from '@sentry/core';
4-
import { generateInstrumentOnce } from '../../otel/instrument';
4+
import { generateInstrumentOnce, instrumentWhenWrapped } from '../../otel/instrument';
55

66
const INTEGRATION_NAME = 'Knex';
77

@@ -11,21 +11,26 @@ export const instrumentKnex = generateInstrumentOnce(
1111
);
1212

1313
const _knexIntegration = (() => {
14+
let instrumentationWrappedCallback: undefined | ((callback: () => void) => void);
15+
1416
return {
1517
name: INTEGRATION_NAME,
1618
setupOnce() {
17-
instrumentKnex();
19+
const instrumentation = instrumentKnex();
20+
instrumentationWrappedCallback = instrumentWhenWrapped(instrumentation);
1821
},
1922

2023
setup(client) {
21-
client.on('spanStart', span => {
22-
const { data } = spanToJSON(span);
23-
// knex.version is always set in the span data
24-
// https://github.com/open-telemetry/opentelemetry-js-contrib/blob/0309caeafc44ac9cb13a3345b790b01b76d0497d/plugins/node/opentelemetry-instrumentation-knex/src/instrumentation.ts#L138
25-
if ('knex.version' in data) {
26-
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.knex');
27-
}
28-
});
24+
instrumentationWrappedCallback?.(() =>
25+
client.on('spanStart', span => {
26+
const { data } = spanToJSON(span);
27+
// knex.version is always set in the span data
28+
// https://github.com/open-telemetry/opentelemetry-js-contrib/blob/0309caeafc44ac9cb13a3345b790b01b76d0497d/plugins/node/opentelemetry-instrumentation-knex/src/instrumentation.ts#L138
29+
if ('knex.version' in data) {
30+
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.knex');
31+
}
32+
}),
33+
);
2934
},
3035
};
3136
}) satisfies IntegrationFn;

packages/node/src/integrations/tracing/prisma.ts

+19-7
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,18 @@ function isPrismaV6TracingHelper(helper: unknown): helper is PrismaV6TracingHelp
1919
return !!helper && typeof helper === 'object' && 'dispatchEngineSpans' in helper;
2020
}
2121

22+
function getPrismaTracingHelper(): unknown | undefined {
23+
const prismaInstrumentationObject = (globalThis as Record<string, unknown>).PRISMA_INSTRUMENTATION;
24+
const prismaTracingHelper =
25+
prismaInstrumentationObject &&
26+
typeof prismaInstrumentationObject === 'object' &&
27+
'helper' in prismaInstrumentationObject
28+
? prismaInstrumentationObject.helper
29+
: undefined;
30+
31+
return prismaTracingHelper;
32+
}
33+
2234
class SentryPrismaInteropInstrumentation extends EsmInteropPrismaInstrumentation {
2335
public constructor() {
2436
super();
@@ -30,13 +42,7 @@ class SentryPrismaInteropInstrumentation extends EsmInteropPrismaInstrumentation
3042
// The PrismaIntegration (super class) defines a global variable `global["PRISMA_INSTRUMENTATION"]` when `enable()` is called. This global variable holds a "TracingHelper" which Prisma uses internally to create tracing data. It's their way of not depending on OTEL with their main package. The sucky thing is, prisma broke the interface of the tracing helper with the v6 major update. This means that if you use Prisma 5 with the v6 instrumentation (or vice versa) Prisma just blows up, because tries to call methods on the helper that no longer exist.
3143
// Because we actually want to use the v6 instrumentation and not blow up in Prisma 5 user's faces, what we're doing here is backfilling the v5 method (`createEngineSpan`) with a noop so that no longer crashes when it attempts to call that function.
3244
// We still won't fully emit all the spans, but this could potentially be implemented in the future.
33-
const prismaInstrumentationObject = (globalThis as Record<string, unknown>).PRISMA_INSTRUMENTATION;
34-
const prismaTracingHelper =
35-
prismaInstrumentationObject &&
36-
typeof prismaInstrumentationObject === 'object' &&
37-
'helper' in prismaInstrumentationObject
38-
? prismaInstrumentationObject.helper
39-
: undefined;
45+
const prismaTracingHelper = getPrismaTracingHelper();
4046

4147
let emittedWarning = false;
4248

@@ -119,6 +125,12 @@ export const prismaIntegration = defineIntegration(
119125
instrumentPrisma({ prismaInstrumentation });
120126
},
121127
setup(client) {
128+
// If no tracing helper exists, we skip any work here
129+
// this means that prisma is not being used
130+
if (!getPrismaTracingHelper()) {
131+
return;
132+
}
133+
122134
client.on('spanStart', span => {
123135
const spanJSON = spanToJSON(span);
124136
if (spanJSON.description?.startsWith('prisma:')) {

packages/node/src/integrations/tracing/tedious.ts

+19-14
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { TediousInstrumentation } from '@opentelemetry/instrumentation-tedious';
22
import { SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, defineIntegration, spanToJSON } from '@sentry/core';
33
import type { IntegrationFn } from '@sentry/core';
4-
import { generateInstrumentOnce } from '../../otel/instrument';
4+
import { generateInstrumentOnce, instrumentWhenWrapped } from '../../otel/instrument';
55

66
const TEDIUS_INSTRUMENTED_METHODS = new Set([
77
'callProcedure',
@@ -17,25 +17,30 @@ const INTEGRATION_NAME = 'Tedious';
1717
export const instrumentTedious = generateInstrumentOnce(INTEGRATION_NAME, () => new TediousInstrumentation({}));
1818

1919
const _tediousIntegration = (() => {
20+
let instrumentationWrappedCallback: undefined | ((callback: () => void) => void);
21+
2022
return {
2123
name: INTEGRATION_NAME,
2224
setupOnce() {
23-
instrumentTedious();
25+
const instrumentation = instrumentTedious();
26+
instrumentationWrappedCallback = instrumentWhenWrapped(instrumentation);
2427
},
2528

2629
setup(client) {
27-
client.on('spanStart', span => {
28-
const { description, data } = spanToJSON(span);
29-
// Tedius integration always set a span name and `db.system` attribute to `mssql`.
30-
if (!description || data['db.system'] !== 'mssql') {
31-
return;
32-
}
33-
34-
const operation = description.split(' ')[0] || '';
35-
if (TEDIUS_INSTRUMENTED_METHODS.has(operation)) {
36-
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.tedious');
37-
}
38-
});
30+
instrumentationWrappedCallback?.(() =>
31+
client.on('spanStart', span => {
32+
const { description, data } = spanToJSON(span);
33+
// Tedius integration always set a span name and `db.system` attribute to `mssql`.
34+
if (!description || data['db.system'] !== 'mssql') {
35+
return;
36+
}
37+
38+
const operation = description.split(' ')[0] || '';
39+
if (TEDIUS_INSTRUMENTED_METHODS.has(operation)) {
40+
span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, 'auto.db.otel.tedious');
41+
}
42+
}),
43+
);
3944
},
4045
};
4146
}) satisfies IntegrationFn;

0 commit comments

Comments
 (0)