UNPKG

@sentry/node

Version:

Sentry Node SDK using OpenTelemetry for performance instrumentation

111 lines (95 loc) 3.63 kB
import { InstrumentationBase, InstrumentationNodeModuleDefinition } from '@opentelemetry/instrumentation'; import { SDK_VERSION, _INTERNAL_shouldSkipAiProviderWrapping, OPENAI_INTEGRATION_NAME, getClient, instrumentOpenAiClient } from '@sentry/core'; const supportedVersions = ['>=4.0.0 <7']; /** * Sentry OpenAI instrumentation using OpenTelemetry. */ class SentryOpenAiInstrumentation extends InstrumentationBase { constructor(config = {}) { super('@sentry/instrumentation-openai', SDK_VERSION, config); } /** * Initializes the instrumentation by defining the modules to be patched. */ init() { const module = new InstrumentationNodeModuleDefinition('openai', supportedVersions, this._patch.bind(this)); return module; } /** * Core patch logic applying instrumentation to the OpenAI and AzureOpenAI client constructors. */ _patch(exports) { let result = exports; result = this._patchClient(result, 'OpenAI'); result = this._patchClient(result, 'AzureOpenAI'); return result; } /** * Patch logic applying instrumentation to the specified client constructor. */ _patchClient(exports, exportKey) { const Original = exports[exportKey]; if (!Original) { return exports; } const config = this.getConfig(); const WrappedOpenAI = function ( ...args) { // Check if wrapping should be skipped (e.g., when LangChain is handling instrumentation) if (_INTERNAL_shouldSkipAiProviderWrapping(OPENAI_INTEGRATION_NAME)) { return Reflect.construct(Original, args) ; } const instance = Reflect.construct(Original, args); const client = getClient(); const defaultPii = Boolean(client?.getOptions().sendDefaultPii); const recordInputs = config.recordInputs ?? defaultPii; const recordOutputs = config.recordOutputs ?? defaultPii; return instrumentOpenAiClient(instance , { recordInputs, recordOutputs, }); } ; // Preserve static and prototype chains Object.setPrototypeOf(WrappedOpenAI, Original); Object.setPrototypeOf(WrappedOpenAI.prototype, Original.prototype); for (const key of Object.getOwnPropertyNames(Original)) { if (!['length', 'name', 'prototype'].includes(key)) { const descriptor = Object.getOwnPropertyDescriptor(Original, key); if (descriptor) { Object.defineProperty(WrappedOpenAI, key, descriptor); } } } // Constructor replacement - handle read-only properties // The OpenAI property might have only a getter, so use defineProperty try { exports[exportKey] = WrappedOpenAI; } catch (error) { // If direct assignment fails, override the property descriptor Object.defineProperty(exports, exportKey, { value: WrappedOpenAI, writable: true, configurable: true, enumerable: true, }); } // Wrap the default export if it points to the original constructor // Constructor replacement - handle read-only properties // The OpenAI property might have only a getter, so use defineProperty if (exports.default === Original) { try { exports.default = WrappedOpenAI; } catch (error) { // If direct assignment fails, override the property descriptor Object.defineProperty(exports, 'default', { value: WrappedOpenAI, writable: true, configurable: true, enumerable: true, }); } } return exports; } } export { SentryOpenAiInstrumentation }; //# sourceMappingURL=instrumentation.js.map