src/plugins/PgPlugin.ts (64 lines of code) (raw):

/*! * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ import SwPlugin, { wrapEmit, wrapCallback, wrapPromise } from '../core/SwPlugin'; import ContextManager from '../trace/context/ContextManager'; import { Component } from '../trace/Component'; import Tag from '../Tag'; import { SpanLayer } from '../proto/language-agent/Tracing_pb'; import PluginInstaller from '../core/PluginInstaller'; import agentConfig from '../config/AgentConfig'; class MySQLPlugin implements SwPlugin { readonly module = 'pg'; readonly versions = '*'; install(installer: PluginInstaller): void { const Client = installer.require?.('pg/lib/client') ?? require('pg/lib/client'); let Cursor: any; try { Cursor = installer.require?.('pg-cursor') ?? require('pg-cursor'); } catch { /* Linter food */ } const _query = Client.prototype.query; Client.prototype.query = function (config: any, values: any, callback: any) { let query: any; const host = `${this.host}:${this.port}`; const span = ContextManager.current.newExitSpan('pg/query', Component.POSTGRESQL); span.start(); try { span.component = Component.POSTGRESQL; span.layer = SpanLayer.DATABASE; span.peer = host; span.tag(Tag.dbType('PostgreSQL')); span.tag(Tag.dbInstance(`${this.connectionParameters.database}`)); let _sql: any; let _values: any; if (typeof config === 'string') _sql = config; else if (config !== null && config !== undefined) { _sql = config.text; _values = config.values; if (typeof config.callback === 'function') config.callback = wrapCallback(span, config.callback, 0); } if (typeof values === 'function') values = wrapCallback(span, values, 0); else if (_values !== undefined) _values = values; if (typeof callback === 'function') callback = wrapCallback(span, callback, 0); span.tag(Tag.dbStatement(`${_sql}`)); if (agentConfig.sqlTraceParameters && _values) { let vals = _values.map((v: any) => (v === undefined ? 'undefined' : JSON.stringify(v))).join(', '); if (vals.length > agentConfig.sqlParametersMaxLength) vals = vals.slice(0, agentConfig.sqlParametersMaxLength) + ' ...'; span.tag(Tag.dbSqlParameters(`[${vals}]`)); } query = _query.call(this, config, values, callback); if (query) { if (Cursor && query instanceof Cursor) wrapEmit(span, query, true, 'end'); else if (typeof query.then === 'function') // generic Promise check query = wrapPromise(span, query); // else we assume there was a callback } } catch (e) { span.error(e); span.stop(); throw e; } span.async(); return query; }; } } // noinspection JSUnusedGlobalSymbols export default new MySQLPlugin();