autotel-plugins 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,469 @@
1
+ 'use strict';
2
+
3
+ var api = require('@opentelemetry/api');
4
+ var traceHelpers = require('autotel/trace-helpers');
5
+
6
+ // src/drizzle/index.ts
7
+
8
+ // src/common/constants.ts
9
+ var SEMATTRS_DB_SYSTEM = "db.system";
10
+ var SEMATTRS_DB_OPERATION = "db.operation";
11
+ var SEMATTRS_DB_STATEMENT = "db.statement";
12
+ var SEMATTRS_DB_NAME = "db.name";
13
+ var SEMATTRS_NET_PEER_NAME = "net.peer.name";
14
+ var SEMATTRS_NET_PEER_PORT = "net.peer.port";
15
+ var DEFAULT_TRACER_NAME = "autotel-plugins/drizzle";
16
+ var DEFAULT_DB_SYSTEM = "postgresql";
17
+ var INSTRUMENTED_FLAG = "__autotelDrizzleInstrumented";
18
+ function extractQueryText(queryArg) {
19
+ if (typeof queryArg === "string") {
20
+ return queryArg;
21
+ }
22
+ if (queryArg && typeof queryArg === "object") {
23
+ if (typeof queryArg.sql === "string") {
24
+ return queryArg.sql;
25
+ }
26
+ if (typeof queryArg.text === "string") {
27
+ return queryArg.text;
28
+ }
29
+ if (typeof queryArg.queryChunks === "object") {
30
+ const drizzleQuery = queryArg;
31
+ if (typeof drizzleQuery.sql === "string") {
32
+ return drizzleQuery.sql;
33
+ }
34
+ }
35
+ }
36
+ return void 0;
37
+ }
38
+ function sanitizeQueryText(queryText, maxLength) {
39
+ if (queryText.length <= maxLength) {
40
+ return queryText;
41
+ }
42
+ return `${queryText.slice(0, Math.max(0, maxLength))}...`;
43
+ }
44
+ function extractOperation(queryText) {
45
+ const trimmed = queryText.trimStart();
46
+ const match = /^(?<op>\w+)/u.exec(trimmed);
47
+ return match?.groups?.op?.toUpperCase();
48
+ }
49
+ function instrumentDrizzle(client, config) {
50
+ if (!client) {
51
+ return client;
52
+ }
53
+ const hasQuery = typeof client.query === "function";
54
+ const hasExecute = typeof client.execute === "function";
55
+ if (!hasQuery && !hasExecute) {
56
+ return client;
57
+ }
58
+ if (client[INSTRUMENTED_FLAG]) {
59
+ return client;
60
+ }
61
+ const {
62
+ tracerName = DEFAULT_TRACER_NAME,
63
+ dbSystem = DEFAULT_DB_SYSTEM,
64
+ dbName,
65
+ captureQueryText = true,
66
+ maxQueryTextLength = 1e3,
67
+ peerName,
68
+ peerPort
69
+ } = config ?? {};
70
+ const tracer = api.trace.getTracer(tracerName);
71
+ const originalMethod = hasQuery ? client.query : client.execute;
72
+ if (!originalMethod) {
73
+ return client;
74
+ }
75
+ const instrumentedMethod = function instrumented(...incomingArgs) {
76
+ const args = [...incomingArgs];
77
+ let callback;
78
+ if (typeof args.at(-1) === "function") {
79
+ callback = args.pop();
80
+ }
81
+ const queryText = extractQueryText(args[0]);
82
+ const operation = queryText ? extractOperation(queryText) : void 0;
83
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
84
+ const span = tracer.startSpan(spanName, { kind: api.SpanKind.CLIENT });
85
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);
86
+ if (operation) {
87
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
88
+ }
89
+ if (dbName) {
90
+ span.setAttribute(SEMATTRS_DB_NAME, dbName);
91
+ }
92
+ if (captureQueryText && queryText !== void 0) {
93
+ const sanitized = sanitizeQueryText(queryText, maxQueryTextLength);
94
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
95
+ }
96
+ if (peerName) {
97
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);
98
+ }
99
+ if (peerPort) {
100
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);
101
+ }
102
+ if (callback) {
103
+ return traceHelpers.runWithSpan(span, () => {
104
+ const wrappedCallback = (err, result) => {
105
+ traceHelpers.finalizeSpan(span, err);
106
+ if (callback) {
107
+ callback(err, result);
108
+ }
109
+ };
110
+ try {
111
+ return Reflect.apply(originalMethod, this, [
112
+ ...args,
113
+ wrappedCallback
114
+ ]);
115
+ } catch (error) {
116
+ traceHelpers.finalizeSpan(span, error);
117
+ throw error;
118
+ }
119
+ });
120
+ }
121
+ return traceHelpers.runWithSpan(span, () => {
122
+ try {
123
+ const result = originalMethod.apply(this, args);
124
+ return Promise.resolve(result).then((value) => {
125
+ traceHelpers.finalizeSpan(span);
126
+ return value;
127
+ }).catch((error) => {
128
+ traceHelpers.finalizeSpan(span, error);
129
+ throw error;
130
+ });
131
+ } catch (error) {
132
+ traceHelpers.finalizeSpan(span, error);
133
+ throw error;
134
+ }
135
+ });
136
+ };
137
+ client[INSTRUMENTED_FLAG] = true;
138
+ if (hasQuery) {
139
+ client.query = instrumentedMethod;
140
+ } else {
141
+ client.execute = instrumentedMethod;
142
+ }
143
+ return client;
144
+ }
145
+ function instrumentDrizzleClient(db, config) {
146
+ if (!db) {
147
+ return db;
148
+ }
149
+ if (db[INSTRUMENTED_FLAG]) {
150
+ return db;
151
+ }
152
+ const {
153
+ tracerName = DEFAULT_TRACER_NAME,
154
+ dbSystem = DEFAULT_DB_SYSTEM,
155
+ dbName,
156
+ captureQueryText = true,
157
+ maxQueryTextLength = 1e3,
158
+ peerName,
159
+ peerPort
160
+ } = config ?? {};
161
+ const tracer = api.trace.getTracer(tracerName);
162
+ let instrumented = false;
163
+ if (db.session && !instrumented) {
164
+ const session = db.session;
165
+ if (typeof session.prepareQuery === "function" && !session[INSTRUMENTED_FLAG]) {
166
+ const originalPrepareQuery = session.prepareQuery;
167
+ session.prepareQuery = function(...args) {
168
+ const prepared = originalPrepareQuery.apply(this, args);
169
+ if (prepared && typeof prepared.execute === "function") {
170
+ const originalPreparedExecute = prepared.execute;
171
+ prepared.execute = function(...executeArgs) {
172
+ const queryObj = args[0];
173
+ const queryText = queryObj?.sql || queryObj?.queryString || extractQueryText(queryObj);
174
+ const operation = queryText ? extractOperation(queryText) : void 0;
175
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
176
+ const span = tracer.startSpan(spanName, { kind: api.SpanKind.CLIENT });
177
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);
178
+ if (operation) {
179
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
180
+ }
181
+ if (dbName) {
182
+ span.setAttribute(SEMATTRS_DB_NAME, dbName);
183
+ }
184
+ if (captureQueryText && queryText !== void 0) {
185
+ const sanitized = sanitizeQueryText(
186
+ queryText,
187
+ maxQueryTextLength
188
+ );
189
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
190
+ }
191
+ if (peerName) {
192
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);
193
+ }
194
+ if (peerPort) {
195
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);
196
+ }
197
+ return traceHelpers.runWithSpan(span, () => {
198
+ try {
199
+ const result = originalPreparedExecute.apply(this, executeArgs);
200
+ return Promise.resolve(result).then((value) => {
201
+ traceHelpers.finalizeSpan(span);
202
+ return value;
203
+ }).catch((error) => {
204
+ traceHelpers.finalizeSpan(span, error);
205
+ throw error;
206
+ });
207
+ } catch (error) {
208
+ traceHelpers.finalizeSpan(span, error);
209
+ throw error;
210
+ }
211
+ });
212
+ };
213
+ }
214
+ return prepared;
215
+ };
216
+ session[INSTRUMENTED_FLAG] = true;
217
+ instrumented = true;
218
+ }
219
+ if (typeof session.query === "function" && !session[INSTRUMENTED_FLAG + "_query"]) {
220
+ const originalQuery = session.query;
221
+ session.query = function(queryString, params) {
222
+ const operation = queryString ? extractOperation(queryString) : void 0;
223
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
224
+ const span = tracer.startSpan(spanName, { kind: api.SpanKind.CLIENT });
225
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);
226
+ if (operation) {
227
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
228
+ }
229
+ if (dbName) {
230
+ span.setAttribute(SEMATTRS_DB_NAME, dbName);
231
+ }
232
+ if (captureQueryText && queryString !== void 0) {
233
+ const sanitized = sanitizeQueryText(queryString, maxQueryTextLength);
234
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
235
+ }
236
+ if (peerName) {
237
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);
238
+ }
239
+ if (peerPort) {
240
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);
241
+ }
242
+ return traceHelpers.runWithSpan(span, () => {
243
+ try {
244
+ const result = Reflect.apply(originalQuery, this, [
245
+ queryString,
246
+ params
247
+ ]);
248
+ return Promise.resolve(result).then((value) => {
249
+ traceHelpers.finalizeSpan(span);
250
+ return value;
251
+ }).catch((error) => {
252
+ traceHelpers.finalizeSpan(span, error);
253
+ throw error;
254
+ });
255
+ } catch (error) {
256
+ traceHelpers.finalizeSpan(span, error);
257
+ throw error;
258
+ }
259
+ });
260
+ };
261
+ session[INSTRUMENTED_FLAG + "_query"] = true;
262
+ instrumented = true;
263
+ }
264
+ if (typeof session.transaction === "function" && !session[INSTRUMENTED_FLAG + "_transaction"]) {
265
+ const originalTransaction = session.transaction;
266
+ session.transaction = function(transactionCallback, ...restArgs) {
267
+ const wrappedCallback = async function(tx) {
268
+ if (tx && (tx.session || tx._?.session || tx)) {
269
+ const txSession = tx.session || tx._?.session || tx;
270
+ if (typeof tx.execute === "function" && !tx[INSTRUMENTED_FLAG + "_execute"]) {
271
+ const originalTxExecute = tx.execute;
272
+ tx.execute = function(...executeArgs) {
273
+ const queryText = extractQueryText(executeArgs[0]);
274
+ const operation = queryText ? extractOperation(queryText) : void 0;
275
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
276
+ const span = tracer.startSpan(spanName, {
277
+ kind: api.SpanKind.CLIENT
278
+ });
279
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);
280
+ span.setAttribute("db.transaction", true);
281
+ if (operation) {
282
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
283
+ }
284
+ if (dbName) {
285
+ span.setAttribute(SEMATTRS_DB_NAME, dbName);
286
+ }
287
+ if (captureQueryText && queryText !== void 0) {
288
+ const sanitized = sanitizeQueryText(
289
+ queryText,
290
+ maxQueryTextLength
291
+ );
292
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
293
+ }
294
+ if (peerName) {
295
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);
296
+ }
297
+ if (peerPort) {
298
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);
299
+ }
300
+ return traceHelpers.runWithSpan(span, () => {
301
+ try {
302
+ const result = originalTxExecute.apply(this, executeArgs);
303
+ return Promise.resolve(result).then((value) => {
304
+ traceHelpers.finalizeSpan(span);
305
+ return value;
306
+ }).catch((error) => {
307
+ traceHelpers.finalizeSpan(span, error);
308
+ throw error;
309
+ });
310
+ } catch (error) {
311
+ traceHelpers.finalizeSpan(span, error);
312
+ throw error;
313
+ }
314
+ });
315
+ };
316
+ tx[INSTRUMENTED_FLAG + "_execute"] = true;
317
+ }
318
+ if (typeof txSession.prepareQuery === "function" && !txSession[INSTRUMENTED_FLAG + "_tx"]) {
319
+ const originalTxPrepareQuery = txSession.prepareQuery;
320
+ txSession.prepareQuery = function(...prepareArgs) {
321
+ const prepared = originalTxPrepareQuery.apply(
322
+ this,
323
+ prepareArgs
324
+ );
325
+ if (prepared && typeof prepared.execute === "function") {
326
+ const originalPreparedExecute = prepared.execute;
327
+ prepared.execute = function(...executeArgs) {
328
+ const queryObj = prepareArgs[0];
329
+ const queryText = queryObj?.sql || queryObj?.queryString || extractQueryText(queryObj);
330
+ const operation = queryText ? extractOperation(queryText) : void 0;
331
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
332
+ const span = tracer.startSpan(spanName, {
333
+ kind: api.SpanKind.CLIENT
334
+ });
335
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);
336
+ span.setAttribute("db.transaction", true);
337
+ if (operation) {
338
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
339
+ }
340
+ if (dbName) {
341
+ span.setAttribute(SEMATTRS_DB_NAME, dbName);
342
+ }
343
+ if (captureQueryText && queryText !== void 0) {
344
+ const sanitized = sanitizeQueryText(
345
+ queryText,
346
+ maxQueryTextLength
347
+ );
348
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
349
+ }
350
+ if (peerName) {
351
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);
352
+ }
353
+ if (peerPort) {
354
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);
355
+ }
356
+ return traceHelpers.runWithSpan(span, () => {
357
+ try {
358
+ const result = originalPreparedExecute.apply(
359
+ this,
360
+ executeArgs
361
+ );
362
+ return Promise.resolve(result).then((value) => {
363
+ traceHelpers.finalizeSpan(span);
364
+ return value;
365
+ }).catch((error) => {
366
+ traceHelpers.finalizeSpan(span, error);
367
+ throw error;
368
+ });
369
+ } catch (error) {
370
+ traceHelpers.finalizeSpan(span, error);
371
+ throw error;
372
+ }
373
+ });
374
+ };
375
+ }
376
+ return prepared;
377
+ };
378
+ txSession[INSTRUMENTED_FLAG + "_tx"] = true;
379
+ }
380
+ }
381
+ return transactionCallback(tx);
382
+ };
383
+ return Reflect.apply(originalTransaction, this, [
384
+ wrappedCallback,
385
+ ...restArgs
386
+ ]);
387
+ };
388
+ session[INSTRUMENTED_FLAG + "_transaction"] = true;
389
+ instrumented = true;
390
+ }
391
+ }
392
+ if (db.$client && !instrumented) {
393
+ const client = db.$client;
394
+ if (typeof client.query === "function" || typeof client.execute === "function") {
395
+ instrumentDrizzle(client, config);
396
+ instrumented = true;
397
+ }
398
+ }
399
+ if (db._ && db._.session && typeof db._.session.execute === "function" && !instrumented) {
400
+ const session = db._.session;
401
+ if (session[INSTRUMENTED_FLAG]) {
402
+ return db;
403
+ }
404
+ const {
405
+ tracerName: tracerName2 = DEFAULT_TRACER_NAME,
406
+ dbSystem: dbSystem2 = DEFAULT_DB_SYSTEM,
407
+ dbName: dbName2,
408
+ captureQueryText: captureQueryText2 = true,
409
+ maxQueryTextLength: maxQueryTextLength2 = 1e3,
410
+ peerName: peerName2,
411
+ peerPort: peerPort2
412
+ } = config ?? {};
413
+ const tracer2 = api.trace.getTracer(tracerName2);
414
+ const originalExecute = session.execute;
415
+ if (!originalExecute) {
416
+ return db;
417
+ }
418
+ const instrumentedExecute = function instrumented2(...args) {
419
+ const queryText = extractQueryText(args[0]);
420
+ const operation = queryText ? extractOperation(queryText) : void 0;
421
+ const spanName = operation ? `drizzle.${operation.toLowerCase()}` : "drizzle.query";
422
+ const span = tracer2.startSpan(spanName, { kind: api.SpanKind.CLIENT });
423
+ span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem2);
424
+ if (operation) {
425
+ span.setAttribute(SEMATTRS_DB_OPERATION, operation);
426
+ }
427
+ if (dbName2) {
428
+ span.setAttribute(SEMATTRS_DB_NAME, dbName2);
429
+ }
430
+ if (captureQueryText2 && queryText !== void 0) {
431
+ const sanitized = sanitizeQueryText(queryText, maxQueryTextLength2);
432
+ span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);
433
+ }
434
+ if (peerName2) {
435
+ span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName2);
436
+ }
437
+ if (peerPort2) {
438
+ span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort2);
439
+ }
440
+ return traceHelpers.runWithSpan(span, () => {
441
+ try {
442
+ const result = originalExecute.apply(this, args);
443
+ return Promise.resolve(result).then((value) => {
444
+ traceHelpers.finalizeSpan(span);
445
+ return value;
446
+ }).catch((error) => {
447
+ traceHelpers.finalizeSpan(span, error);
448
+ throw error;
449
+ });
450
+ } catch (error) {
451
+ traceHelpers.finalizeSpan(span, error);
452
+ throw error;
453
+ }
454
+ });
455
+ };
456
+ session[INSTRUMENTED_FLAG] = true;
457
+ session.execute = instrumentedExecute;
458
+ instrumented = true;
459
+ }
460
+ if (instrumented) {
461
+ db[INSTRUMENTED_FLAG] = true;
462
+ }
463
+ return db;
464
+ }
465
+
466
+ exports.instrumentDrizzle = instrumentDrizzle;
467
+ exports.instrumentDrizzleClient = instrumentDrizzleClient;
468
+ //# sourceMappingURL=drizzle.cjs.map
469
+ //# sourceMappingURL=drizzle.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/common/constants.ts","../src/drizzle/index.ts"],"names":["trace","SpanKind","runWithSpan","finalizeSpan","tracerName","dbSystem","dbName","captureQueryText","maxQueryTextLength","peerName","peerPort","tracer","instrumented"],"mappings":"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","file":"drizzle.cjs","sourcesContent":["/**\n * OpenTelemetry semantic conventions for database operations.\n * These constants are shared across all plugins.\n */\n\n// Common database attributes\nexport const SEMATTRS_DB_SYSTEM = 'db.system' as const;\nexport const SEMATTRS_DB_OPERATION = 'db.operation' as const;\nexport const SEMATTRS_DB_STATEMENT = 'db.statement' as const;\nexport const SEMATTRS_DB_NAME = 'db.name' as const;\n\n// MongoDB-specific attributes\nexport const SEMATTRS_DB_MONGODB_COLLECTION = 'db.mongodb.collection' as const;\n\n// Network attributes\nexport const SEMATTRS_NET_PEER_NAME = 'net.peer.name' as const;\nexport const SEMATTRS_NET_PEER_PORT = 'net.peer.port' as const;\n","/* eslint-disable @typescript-eslint/no-explicit-any */\n// Note: `any` types are necessary for dynamic instrumentation patterns\n// where we need to wrap arbitrary methods and preserve their signatures\nimport { SpanKind, trace } from '@opentelemetry/api';\nimport {\n SEMATTRS_DB_SYSTEM,\n SEMATTRS_DB_OPERATION,\n SEMATTRS_DB_STATEMENT,\n SEMATTRS_DB_NAME,\n SEMATTRS_NET_PEER_NAME,\n SEMATTRS_NET_PEER_PORT,\n} from '../common/constants';\nimport { runWithSpan, finalizeSpan } from 'autotel/trace-helpers';\n\nconst DEFAULT_TRACER_NAME = 'autotel-plugins/drizzle';\nconst DEFAULT_DB_SYSTEM = 'postgresql';\nconst INSTRUMENTED_FLAG = '__autotelDrizzleInstrumented' as const;\n\ntype QueryCallback = (error: unknown, result: unknown) => void;\n\ntype QueryFunction = (...args: any[]) => any;\n\ninterface DrizzleClientLike {\n query?: QueryFunction;\n execute?: QueryFunction;\n [INSTRUMENTED_FLAG]?: true;\n [key: string]: any; // Allow other properties\n}\n\n/**\n * Configuration options for Drizzle instrumentation.\n */\nexport interface InstrumentDrizzleConfig {\n /**\n * Custom tracer name. Defaults to \"autotel-plugins/drizzle\".\n */\n tracerName?: string;\n\n /**\n * Database system identifier (e.g., \"postgresql\", \"mysql\", \"sqlite\").\n * Defaults to \"postgresql\".\n */\n dbSystem?: string;\n\n /**\n * Database name to include in spans.\n */\n dbName?: string;\n\n /**\n * Whether to capture full SQL query text in spans.\n * Defaults to true.\n */\n captureQueryText?: boolean;\n\n /**\n * Maximum length for captured query text. Queries longer than this\n * will be truncated. Defaults to 1000 characters.\n */\n maxQueryTextLength?: number;\n\n /**\n * Remote hostname or IP address of the database server.\n * Example: \"db.example.com\" or \"192.168.1.100\"\n */\n peerName?: string;\n\n /**\n * Remote port number of the database server.\n * Example: 5432 for PostgreSQL, 3306 for MySQL\n */\n peerPort?: number;\n}\n\n/**\n * Extracts SQL query text from various query argument formats.\n */\nfunction extractQueryText(queryArg: unknown): string | undefined {\n if (typeof queryArg === 'string') {\n return queryArg;\n }\n if (queryArg && typeof queryArg === 'object') {\n // Generic SQL object format (used by LibSQL, MySQL, and others)\n if (typeof (queryArg as { sql?: unknown }).sql === 'string') {\n return (queryArg as { sql: string }).sql;\n }\n // PostgreSQL-style query object\n if (typeof (queryArg as { text?: unknown }).text === 'string') {\n return (queryArg as { text: string }).text;\n }\n // Drizzle SQL object\n if (\n typeof (queryArg as { queryChunks?: unknown }).queryChunks === 'object'\n ) {\n // Drizzle query objects may have complex structure, try to extract meaningful info\n const drizzleQuery = queryArg as Record<string, unknown>;\n if (typeof drizzleQuery.sql === 'string') {\n return drizzleQuery.sql;\n }\n }\n }\n return undefined;\n}\n\n/**\n * Sanitizes and truncates query text for safe inclusion in spans.\n */\nfunction sanitizeQueryText(queryText: string, maxLength: number): string {\n if (queryText.length <= maxLength) {\n return queryText;\n }\n return `${queryText.slice(0, Math.max(0, maxLength))}...`;\n}\n\n/**\n * Extracts the SQL operation (SELECT, INSERT, etc.) from query text.\n */\nfunction extractOperation(queryText: string): string | undefined {\n const trimmed = queryText.trimStart();\n const match = /^(?<op>\\w+)/u.exec(trimmed);\n return match?.groups?.op?.toUpperCase();\n}\n\n/**\n * Instruments a database connection pool/client with OpenTelemetry tracing.\n *\n * This function wraps the connection's `query` and `execute` methods to create spans for each database\n * operation.\n * The instrumentation is idempotent - calling it multiple times on the same connection will only\n * instrument it once.\n *\n * @typeParam TClient - The type of the database connection pool or client\n * @param client - The database connection pool or client to instrument\n * @param config - Optional configuration for instrumentation behavior\n * @returns The instrumented pool/client (same instance, modified in place)\n *\n * @example\n * ```typescript\n * // PostgreSQL with node-postgres\n * import { drizzle } from 'drizzle-orm/node-postgres';\n * import { Pool } from 'pg';\n * import { instrumentDrizzle } from 'autotel-plugins/drizzle';\n *\n * const pool = new Pool({ connectionString: process.env.DATABASE_URL });\n * const instrumentedPool = instrumentDrizzle(pool, {\n * dbSystem: 'postgresql',\n * dbName: 'myapp',\n * peerName: 'db.example.com',\n * peerPort: 5432,\n * });\n * const db = drizzle({ client: instrumentedPool });\n * ```\n *\n * @example\n * ```typescript\n * // MySQL with mysql2\n * import { drizzle } from 'drizzle-orm/mysql2';\n * import mysql from 'mysql2/promise';\n * import { instrumentDrizzle } from 'autotel-plugins/drizzle';\n *\n * const connection = await mysql.createConnection({\n * host: 'localhost',\n * user: 'root',\n * database: 'mydb',\n * });\n * const instrumentedConnection = instrumentDrizzle(connection, { dbSystem: 'mysql' });\n * const db = drizzle({ client: instrumentedConnection });\n * ```\n *\n * @example\n * ```typescript\n * // SQLite with better-sqlite3\n * import { drizzle } from 'drizzle-orm/better-sqlite3';\n * import Database from 'better-sqlite3';\n * import { instrumentDrizzle } from 'autotel-plugins/drizzle';\n *\n * const sqlite = new Database('sqlite.db');\n * const instrumentedSqlite = instrumentDrizzle(sqlite, { dbSystem: 'sqlite' });\n * const db = drizzle({ client: instrumentedSqlite });\n * ```\n *\n * @example\n * ```typescript\n * // LibSQL/Turso\n * import { drizzle } from 'drizzle-orm/libsql';\n * import { createClient } from '@libsql/client';\n * import { instrumentDrizzle } from 'autotel-plugins/drizzle';\n *\n * const client = createClient({\n * url: process.env.DATABASE_URL!,\n * authToken: process.env.DATABASE_AUTH_TOKEN,\n * });\n * const instrumentedClient = instrumentDrizzle(client, { dbSystem: 'sqlite' });\n * const db = drizzle({ client: instrumentedClient });\n * ```\n */\nexport function instrumentDrizzle<TClient extends DrizzleClientLike>(\n client: TClient,\n config?: InstrumentDrizzleConfig,\n): TClient {\n if (!client) {\n return client;\n }\n\n // Check if client has query or execute method\n const hasQuery = typeof client.query === 'function';\n const hasExecute = typeof client.execute === 'function';\n\n if (!hasQuery && !hasExecute) {\n return client;\n }\n\n if (client[INSTRUMENTED_FLAG]) {\n return client;\n }\n\n const {\n tracerName = DEFAULT_TRACER_NAME,\n dbSystem = DEFAULT_DB_SYSTEM,\n dbName,\n captureQueryText = true,\n maxQueryTextLength = 1000,\n peerName,\n peerPort,\n } = config ?? {};\n\n const tracer = trace.getTracer(tracerName);\n\n // Store the original method (query or execute)\n const originalMethod = hasQuery ? client.query : client.execute;\n\n if (!originalMethod) {\n return client;\n }\n\n const instrumentedMethod: QueryFunction = function instrumented(\n this: any,\n ...incomingArgs: any[]\n ) {\n const args = [...incomingArgs];\n let callback: QueryCallback | undefined;\n\n // Detect callback pattern\n if (typeof args.at(-1) === 'function') {\n callback = args.pop() as QueryCallback;\n }\n\n // Extract query information\n const queryText = extractQueryText(args[0]);\n const operation = queryText ? extractOperation(queryText) : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, { kind: SpanKind.CLIENT });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryText !== undefined) {\n const sanitized = sanitizeQueryText(queryText, maxQueryTextLength);\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Callback-based pattern\n if (callback) {\n return runWithSpan(span, () => {\n const wrappedCallback: QueryCallback = (err, result) => {\n finalizeSpan(span, err);\n if (callback) {\n callback(err, result);\n }\n };\n\n try {\n return Reflect.apply(originalMethod, this, [\n ...args,\n wrappedCallback,\n ]);\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n }\n\n // Promise-based pattern\n return runWithSpan(span, () => {\n try {\n const result = originalMethod.apply(this, args);\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n\n client[INSTRUMENTED_FLAG] = true;\n\n // Replace the original method with the instrumented one\n if (hasQuery) {\n client.query = instrumentedMethod;\n } else {\n client.execute = instrumentedMethod;\n }\n\n return client;\n}\n\n/**\n * Interface for Drizzle database instances with minimal type requirements.\n */\ninterface DrizzleDbLike {\n $client?: DrizzleClientLike | any; // Allow any client type\n execute?: QueryFunction; // Direct execute method on db\n transaction?: QueryFunction; // Transaction method on db\n _?: {\n session?: {\n execute?: QueryFunction;\n [INSTRUMENTED_FLAG]?: true;\n [key: string]: any;\n };\n [key: string]: any;\n };\n [INSTRUMENTED_FLAG]?: true;\n [key: string]: any; // Allow other properties\n}\n\n/**\n * Instruments a Drizzle database instance with OpenTelemetry tracing.\n *\n * This function instruments the database at the session level, automatically tracing all database\n * operations including query builders, direct SQL execution, and transactions.\n *\n * The instrumentation is idempotent - calling it multiple times on the same\n * database will only instrument it once.\n *\n * @typeParam TDb - The type of the Drizzle database instance\n * @param db - The Drizzle database instance to instrument\n * @param config - Optional configuration for instrumentation behavior\n * @returns The instrumented database instance (same instance, modified in place)\n *\n * @example\n * ```typescript\n * // PostgreSQL with postgres.js\n * import { drizzle } from 'drizzle-orm/postgres-js';\n * import postgres from 'postgres';\n * import { instrumentDrizzleClient } from 'autotel-plugins/drizzle';\n *\n * // Using connection string\n * const db = drizzle(process.env.DATABASE_URL!);\n * instrumentDrizzleClient(db, { dbSystem: 'postgresql' });\n *\n * // Or with a client instance\n * const queryClient = postgres(process.env.DATABASE_URL!);\n * const db = drizzle({ client: queryClient });\n * instrumentDrizzleClient(db, { dbSystem: 'postgresql' });\n * ```\n *\n * @example\n * ```typescript\n * // PostgreSQL with node-postgres (pg)\n * import { drizzle } from 'drizzle-orm/node-postgres';\n * import { Pool } from 'pg';\n * import { instrumentDrizzleClient } from 'autotel-plugins/drizzle';\n *\n * // Using connection string\n * const db = drizzle(process.env.DATABASE_URL!);\n * instrumentDrizzleClient(db, { dbSystem: 'postgresql' });\n *\n * // Or with a pool\n * const pool = new Pool({ connectionString: process.env.DATABASE_URL });\n * const db = drizzle({ client: pool });\n * instrumentDrizzleClient(db, {\n * dbSystem: 'postgresql',\n * dbName: 'myapp',\n * peerName: 'db.example.com',\n * peerPort: 5432,\n * });\n * ```\n */\nexport function instrumentDrizzleClient<TDb extends DrizzleDbLike>(\n db: TDb,\n config?: InstrumentDrizzleConfig,\n): TDb {\n if (!db) {\n return db;\n }\n\n // Check if already instrumented\n if (db[INSTRUMENTED_FLAG]) {\n return db;\n }\n\n const {\n tracerName = DEFAULT_TRACER_NAME,\n dbSystem = DEFAULT_DB_SYSTEM,\n dbName,\n captureQueryText = true,\n maxQueryTextLength = 1000,\n peerName,\n peerPort,\n } = config ?? {};\n\n const tracer = trace.getTracer(tracerName);\n let instrumented = false;\n\n // First priority: Instrument the session directly\n // This is where all queries actually go through\n if ((db as any).session && !instrumented) {\n const session = (db as any).session;\n\n // Check if session has prepareQuery method (used by select/insert/update/delete)\n if (\n typeof session.prepareQuery === 'function' &&\n !session[INSTRUMENTED_FLAG]\n ) {\n const originalPrepareQuery = session.prepareQuery;\n\n session.prepareQuery = function (...args: any[]) {\n const prepared = originalPrepareQuery.apply(this, args);\n\n // Wrap the prepared query's execute method\n if (prepared && typeof prepared.execute === 'function') {\n const originalPreparedExecute = prepared.execute;\n\n prepared.execute = function (this: any, ...executeArgs: any[]) {\n // Extract query information from the query object\n const queryObj = args[0]; // The query object passed to prepareQuery\n const queryText =\n queryObj?.sql ||\n queryObj?.queryString ||\n extractQueryText(queryObj);\n const operation = queryText\n ? extractOperation(queryText)\n : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, { kind: SpanKind.CLIENT });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryText !== undefined) {\n const sanitized = sanitizeQueryText(\n queryText,\n maxQueryTextLength,\n );\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Execute the prepared query\n return runWithSpan(span, () => {\n try {\n const result = originalPreparedExecute.apply(this, executeArgs);\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n }\n\n return prepared;\n };\n\n session[INSTRUMENTED_FLAG] = true;\n instrumented = true;\n }\n\n // Also instrument direct query method if exists\n if (\n typeof session.query === 'function' &&\n !session[INSTRUMENTED_FLAG + '_query']\n ) {\n const originalQuery = session.query;\n\n session.query = function (this: any, queryString: string, params: any[]) {\n const operation = queryString\n ? extractOperation(queryString)\n : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, { kind: SpanKind.CLIENT });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryString !== undefined) {\n const sanitized = sanitizeQueryText(queryString, maxQueryTextLength);\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Execute the query\n return runWithSpan(span, () => {\n try {\n const result = Reflect.apply(originalQuery, this, [\n queryString,\n params,\n ]);\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n\n session[INSTRUMENTED_FLAG + '_query'] = true;\n instrumented = true;\n }\n\n // Instrument transaction method to ensure transaction sessions are also instrumented\n if (\n typeof session.transaction === 'function' &&\n !session[INSTRUMENTED_FLAG + '_transaction']\n ) {\n const originalTransaction = session.transaction;\n\n session.transaction = function (\n this: any,\n transactionCallback: any,\n ...restArgs: any[]\n ) {\n // Wrap the transaction callback to instrument the tx object\n const wrappedCallback = async function (tx: any) {\n // Instrument the transaction's session if it has one\n if (tx && (tx.session || tx._?.session || tx)) {\n const txSession = tx.session || tx._?.session || tx;\n\n // Instrument tx.execute if it exists\n if (\n typeof tx.execute === 'function' &&\n !tx[INSTRUMENTED_FLAG + '_execute']\n ) {\n const originalTxExecute = tx.execute;\n\n tx.execute = function (this: any, ...executeArgs: any[]) {\n const queryText = extractQueryText(executeArgs[0]);\n const operation = queryText\n ? extractOperation(queryText)\n : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, {\n kind: SpanKind.CLIENT,\n });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n span.setAttribute('db.transaction', true);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryText !== undefined) {\n const sanitized = sanitizeQueryText(\n queryText,\n maxQueryTextLength,\n );\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Execute the query\n return runWithSpan(span, () => {\n try {\n const result = originalTxExecute.apply(this, executeArgs);\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n\n tx[INSTRUMENTED_FLAG + '_execute'] = true;\n }\n\n // Also instrument txSession.prepareQuery if it exists\n if (\n typeof txSession.prepareQuery === 'function' &&\n !txSession[INSTRUMENTED_FLAG + '_tx']\n ) {\n const originalTxPrepareQuery = txSession.prepareQuery;\n\n txSession.prepareQuery = function (...prepareArgs: any[]) {\n const prepared = originalTxPrepareQuery.apply(\n this,\n prepareArgs,\n );\n\n // Wrap the prepared query's execute method\n if (prepared && typeof prepared.execute === 'function') {\n const originalPreparedExecute = prepared.execute;\n\n prepared.execute = function (\n this: any,\n ...executeArgs: any[]\n ) {\n // Extract query information from the query object\n const queryObj = prepareArgs[0]; // The query object passed to prepareQuery\n const queryText =\n queryObj?.sql ||\n queryObj?.queryString ||\n extractQueryText(queryObj);\n const operation = queryText\n ? extractOperation(queryText)\n : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, {\n kind: SpanKind.CLIENT,\n });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n span.setAttribute('db.transaction', true);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryText !== undefined) {\n const sanitized = sanitizeQueryText(\n queryText,\n maxQueryTextLength,\n );\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Execute the prepared query\n return runWithSpan(span, () => {\n try {\n const result = originalPreparedExecute.apply(\n this,\n executeArgs,\n );\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n }\n\n return prepared;\n };\n\n txSession[INSTRUMENTED_FLAG + '_tx'] = true;\n }\n }\n\n // Call the original callback with the instrumented tx\n return transactionCallback(tx);\n };\n\n // Call the original transaction with the wrapped callback\n return Reflect.apply(originalTransaction, this, [\n wrappedCallback,\n ...restArgs,\n ]);\n };\n\n session[INSTRUMENTED_FLAG + '_transaction'] = true;\n instrumented = true;\n }\n }\n\n if (db.$client && !instrumented) {\n const client = db.$client;\n // Check if client has query or execute function\n if (\n typeof client.query === 'function' ||\n typeof client.execute === 'function'\n ) {\n instrumentDrizzle(client, config);\n instrumented = true;\n }\n }\n\n // Third priority: Try to instrument via session.execute as fallback\n if (\n db._ &&\n db._.session &&\n typeof db._.session.execute === 'function' &&\n !instrumented\n ) {\n const session = db._.session;\n\n // Check if already instrumented\n if (session[INSTRUMENTED_FLAG]) {\n return db;\n }\n\n const {\n tracerName = DEFAULT_TRACER_NAME,\n dbSystem = DEFAULT_DB_SYSTEM,\n dbName,\n captureQueryText = true,\n maxQueryTextLength = 1000,\n peerName,\n peerPort,\n } = config ?? {};\n\n const tracer = trace.getTracer(tracerName);\n const originalExecute = session.execute;\n\n if (!originalExecute) {\n return db;\n }\n\n const instrumentedExecute: QueryFunction = function instrumented(\n this: any,\n ...args: any[]\n ) {\n // Extract query information\n const queryText = extractQueryText(args[0]);\n const operation = queryText ? extractOperation(queryText) : undefined;\n const spanName = operation\n ? `drizzle.${operation.toLowerCase()}`\n : 'drizzle.query';\n\n // Start span\n const span = tracer.startSpan(spanName, { kind: SpanKind.CLIENT });\n span.setAttribute(SEMATTRS_DB_SYSTEM, dbSystem);\n\n if (operation) {\n span.setAttribute(SEMATTRS_DB_OPERATION, operation);\n }\n\n if (dbName) {\n span.setAttribute(SEMATTRS_DB_NAME, dbName);\n }\n\n if (captureQueryText && queryText !== undefined) {\n const sanitized = sanitizeQueryText(queryText, maxQueryTextLength);\n span.setAttribute(SEMATTRS_DB_STATEMENT, sanitized);\n }\n\n if (peerName) {\n span.setAttribute(SEMATTRS_NET_PEER_NAME, peerName);\n }\n\n if (peerPort) {\n span.setAttribute(SEMATTRS_NET_PEER_PORT, peerPort);\n }\n\n // Promise-based pattern (session.execute is typically promise-based)\n return runWithSpan(span, () => {\n try {\n const result = originalExecute.apply(this, args);\n return Promise.resolve(result)\n .then((value) => {\n finalizeSpan(span);\n return value;\n })\n .catch((error) => {\n finalizeSpan(span, error);\n throw error;\n });\n } catch (error) {\n finalizeSpan(span, error);\n throw error;\n }\n });\n };\n\n session[INSTRUMENTED_FLAG] = true;\n session.execute = instrumentedExecute;\n instrumented = true;\n }\n\n // Mark the db as instrumented if we instrumented anything\n if (instrumented) {\n db[INSTRUMENTED_FLAG] = true;\n }\n\n return db;\n}\n"]}