UNPKG

@sentry/node

Version:

Sentry Node SDK using OpenTelemetry for performance instrumentation

155 lines (126 loc) 5.1 kB
import * as os from 'node:os'; import { trace } from '@opentelemetry/api'; import { registerInstrumentations } from '@opentelemetry/instrumentation'; import { ServerRuntimeClient, applySdkMetadata, logger, _INTERNAL_flushLogsBuffer, SDK_VERSION } from '@sentry/core'; import { getTraceContextForScope } from '@sentry/opentelemetry'; import { threadId, isMainThread } from 'worker_threads'; import { DEBUG_BUILD } from '../debug-build.js'; const DEFAULT_CLIENT_REPORT_FLUSH_INTERVAL_MS = 60000; // 60s was chosen arbitrarily /** A client for using Sentry with Node & OpenTelemetry. */ class NodeClient extends ServerRuntimeClient { constructor(options) { const serverName = options.includeServerName === false ? undefined : options.serverName || global.process.env.SENTRY_NAME || os.hostname(); const clientOptions = { ...options, platform: 'node', runtime: { name: 'node', version: global.process.version }, serverName, }; if (options.openTelemetryInstrumentations) { registerInstrumentations({ instrumentations: options.openTelemetryInstrumentations, }); } applySdkMetadata(clientOptions, 'node'); logger.log( `Initializing Sentry: process: ${process.pid}, thread: ${isMainThread ? 'main' : `worker-${threadId}`}.`, ); super(clientOptions); if (this.getOptions()._experiments?.enableLogs) { this._logOnExitFlushListener = () => { _INTERNAL_flushLogsBuffer(this); }; if (serverName) { this.on('beforeCaptureLog', log => { log.attributes = { ...log.attributes, 'server.address': serverName, }; }); } process.on('beforeExit', this._logOnExitFlushListener); } } /** Get the OTEL tracer. */ get tracer() { if (this._tracer) { return this._tracer; } const name = '@sentry/node'; const version = SDK_VERSION; const tracer = trace.getTracer(name, version); this._tracer = tracer; return tracer; } // Eslint ignore explanation: This is already documented in super. // eslint-disable-next-line jsdoc/require-jsdoc async flush(timeout) { const provider = this.traceProvider; const spanProcessor = provider?.activeSpanProcessor; if (spanProcessor) { await spanProcessor.forceFlush(); } if (this.getOptions().sendClientReports) { this._flushOutcomes(); } return super.flush(timeout); } // Eslint ignore explanation: This is already documented in super. // eslint-disable-next-line jsdoc/require-jsdoc close(timeout) { if (this._clientReportInterval) { clearInterval(this._clientReportInterval); } if (this._clientReportOnExitFlushListener) { process.off('beforeExit', this._clientReportOnExitFlushListener); } if (this._logOnExitFlushListener) { process.off('beforeExit', this._logOnExitFlushListener); } return super.close(timeout); } /** * Will start tracking client reports for this client. * * NOTICE: This method will create an interval that is periodically called and attach a `process.on('beforeExit')` * hook. To clean up these resources, call `.close()` when you no longer intend to use the client. Not doing so will * result in a memory leak. */ // The reason client reports need to be manually activated with this method instead of just enabling them in a // constructor, is that if users periodically and unboundedly create new clients, we will create more and more // intervals and beforeExit listeners, thus leaking memory. In these situations, users are required to call // `client.close()` in order to dispose of the acquired resources. // We assume that calling this method in Sentry.init() is a sensible default, because calling Sentry.init() over and // over again would also result in memory leaks. // Note: We have experimented with using `FinalizationRegisty` to clear the interval when the client is garbage // collected, but it did not work, because the cleanup function never got called. startClientReportTracking() { const clientOptions = this.getOptions(); if (clientOptions.sendClientReports) { this._clientReportOnExitFlushListener = () => { this._flushOutcomes(); }; this._clientReportInterval = setInterval(() => { DEBUG_BUILD && logger.log('Flushing client reports based on interval.'); this._flushOutcomes(); }, clientOptions.clientReportFlushInterval ?? DEFAULT_CLIENT_REPORT_FLUSH_INTERVAL_MS) // Unref is critical for not preventing the process from exiting because the interval is active. .unref(); process.on('beforeExit', this._clientReportOnExitFlushListener); } } /** Custom implementation for OTEL, so we can handle scope-span linking. */ _getTraceInfoFromScope( scope, ) { if (!scope) { return [undefined, undefined]; } return getTraceContextForScope(this, scope); } } export { NodeClient }; //# sourceMappingURL=client.js.map