UNPKG

@sentry/node

Version:

Sentry Node SDK using OpenTelemetry for performance instrumentation

151 lines (129 loc) 5.8 kB
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); const api = require('@opentelemetry/api'); const resources = require('@opentelemetry/resources'); const sdkTraceBase = require('@opentelemetry/sdk-trace-base'); const semanticConventions = require('@opentelemetry/semantic-conventions'); const core = require('@sentry/core'); const nodeCore = require('@sentry/node-core'); const opentelemetry = require('@sentry/opentelemetry'); const importInTheMiddle = require('import-in-the-middle'); const moduleModule = require('module'); const debugBuild = require('../debug-build.js'); const index = require('../integrations/tracing/index.js'); var _documentCurrentScript = typeof document !== 'undefined' ? document.currentScript : null; // About 277h - this must fit into new Array(len)! const MAX_MAX_SPAN_WAIT_DURATION = 1000000; /** * Initialize OpenTelemetry for Node. */ function initOpenTelemetry(client, options = {}) { if (client.getOptions().debug) { nodeCore.setupOpenTelemetryLogger(); } const provider = setupOtel(client, options); client.traceProvider = provider; } /** Initialize the ESM loader. */ function maybeInitializeEsmLoader() { const [nodeMajor = 0, nodeMinor = 0] = process.versions.node.split('.').map(Number); // Register hook was added in v20.6.0 and v18.19.0 if (nodeMajor >= 21 || (nodeMajor === 20 && nodeMinor >= 6) || (nodeMajor === 18 && nodeMinor >= 19)) { if (!core.GLOBAL_OBJ._sentryEsmLoaderHookRegistered) { try { const { addHookMessagePort } = importInTheMiddle.createAddHookMessageChannel(); // @ts-expect-error register is available in these versions moduleModule.default.register('import-in-the-middle/hook.mjs', (typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('sdk/initOtel.js', document.baseURI).href)), { data: { addHookMessagePort, include: [] }, transferList: [addHookMessagePort], }); } catch (error) { core.debug.warn('Failed to register ESM hook', error); } } } else { core.consoleSandbox(() => { // eslint-disable-next-line no-console console.warn( `[Sentry] You are using Node.js v${process.versions.node} in ESM mode ("import syntax"). The Sentry Node.js SDK is not compatible with ESM in Node.js versions before 18.19.0 or before 20.6.0. Please either build your application with CommonJS ("require() syntax"), or upgrade your Node.js version.`, ); }); } } /** * Preload OpenTelemetry for Node. * This can be used to preload instrumentation early, but set up Sentry later. * By preloading the OTEL instrumentation wrapping still happens early enough that everything works. */ function preloadOpenTelemetry(options = {}) { const { debug } = options; if (debug) { core.debug.enable(); } if (!nodeCore.isCjs()) { maybeInitializeEsmLoader(); } // These are all integrations that we need to pre-load to ensure they are set up before any other code runs getPreloadMethods(options.integrations).forEach(fn => { fn(); if (debug) { core.debug.log(`[Sentry] Preloaded ${fn.id} instrumentation`); } }); } function getPreloadMethods(integrationNames) { const instruments = index.getOpenTelemetryInstrumentationToPreload(); if (!integrationNames) { return instruments; } return instruments.filter(instrumentation => integrationNames.includes(instrumentation.id)); } /** Just exported for tests. */ function setupOtel(client, options = {}) { // Create and configure NodeTracerProvider const provider = new sdkTraceBase.BasicTracerProvider({ sampler: new opentelemetry.SentrySampler(client), resource: resources.defaultResource().merge( resources.resourceFromAttributes({ [semanticConventions.ATTR_SERVICE_NAME]: 'node', // eslint-disable-next-line deprecation/deprecation [semanticConventions.SEMRESATTRS_SERVICE_NAMESPACE]: 'sentry', [semanticConventions.ATTR_SERVICE_VERSION]: core.SDK_VERSION, }), ), forceFlushTimeoutMillis: 500, spanProcessors: [ new opentelemetry.SentrySpanProcessor({ timeout: _clampSpanProcessorTimeout(client.getOptions().maxSpanWaitDuration), }), ...(options.spanProcessors || []), ], }); // Register as globals api.trace.setGlobalTracerProvider(provider); api.propagation.setGlobalPropagator(new opentelemetry.SentryPropagator()); api.context.setGlobalContextManager(new nodeCore.SentryContextManager()); return provider; } /** Just exported for tests. */ function _clampSpanProcessorTimeout(maxSpanWaitDuration) { if (maxSpanWaitDuration == null) { return undefined; } // We guard for a max. value here, because we create an array with this length // So if this value is too large, this would fail if (maxSpanWaitDuration > MAX_MAX_SPAN_WAIT_DURATION) { debugBuild.DEBUG_BUILD && core.debug.warn(`\`maxSpanWaitDuration\` is too high, using the maximum value of ${MAX_MAX_SPAN_WAIT_DURATION}`); return MAX_MAX_SPAN_WAIT_DURATION; } else if (maxSpanWaitDuration <= 0 || Number.isNaN(maxSpanWaitDuration)) { debugBuild.DEBUG_BUILD && core.debug.warn('`maxSpanWaitDuration` must be a positive number, using default value instead.'); return undefined; } return maxSpanWaitDuration; } exports._clampSpanProcessorTimeout = _clampSpanProcessorTimeout; exports.initOpenTelemetry = initOpenTelemetry; exports.maybeInitializeEsmLoader = maybeInitializeEsmLoader; exports.preloadOpenTelemetry = preloadOpenTelemetry; exports.setupOtel = setupOtel; //# sourceMappingURL=initOtel.js.map