Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(mongodb): add db statement serializer config #626

Merged
merged 12 commits into from
Aug 31, 2021
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ import {
CommandResult,
} from './types';
import { VERSION } from './version';
import { DbStatementSerializer } from './types';

const supportedVersions = ['>=3.3 <4'];

Expand Down Expand Up @@ -153,12 +154,13 @@ export class MongoDBInstrumentation extends InstrumentationBase<
kind: SpanKind.CLIENT,
}
);

instrumentation._populateAttributes(
span,
ns,
server,
// eslint-disable-next-line @typescript-eslint/no-explicit-any
operationName !== 'insert' ? (ops[0] as any) : undefined
dyladan marked this conversation as resolved.
Show resolved Hide resolved
ops[0] as any
);
const patchedCallback = instrumentation._patchEnd(span, resultHandler);
// handle when options is the callback to send the correct number of args
Expand Down Expand Up @@ -408,15 +410,35 @@ export class MongoDBInstrumentation extends InstrumentationBase<

// capture parameters within the query as well if enhancedDatabaseReporting is enabled.
const commandObj = command.query ?? command.q ?? command;
const query =
this._config?.enhancedDatabaseReporting === true
? commandObj
: Object.keys(commandObj).reduce((obj, key) => {
obj[key] = '?';
return obj;
}, {} as { [key: string]: unknown });
const dbStatementSerializer: DbStatementSerializer =
this._config.dbStatementSerializer ||
this._defaultDbStatementSerializer.bind(this);

if (typeof dbStatementSerializer === 'function') {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you should do this checking a bit earlier for example

const dbStatementSerializer: DbStatementSerializer =
      typeof this._config.dbStatementSerializer === 'function' ? this._config.dbStatementSerializer :
      this._defaultDbStatementSerializer.bind(this);

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

safeExecuteInTheMiddle(
() => {
const query = dbStatementSerializer(commandObj);
span.setAttribute(SemanticAttributes.DB_STATEMENT, query);
},
err => {
if (err) {
this._diag.error('Error running dbStatementSerializer hook', err);
}
},
true
);
}
}

span.setAttribute(SemanticAttributes.DB_STATEMENT, JSON.stringify(query));
private _defaultDbStatementSerializer(commandObj: Record<string, unknown>) {
const enhancedDbReporting = !!this._config?.enhancedDatabaseReporting;
const resultObj = enhancedDbReporting
? commandObj
: Object.keys(commandObj).reduce((obj, key) => {
obj[key] = '?';
return obj;
}, {} as { [key: string]: unknown });
return JSON.stringify(resultObj);
}

/**
Expand Down
13 changes: 13 additions & 0 deletions plugins/node/opentelemetry-instrumentation-mongodb/src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,14 @@ export interface MongoDBInstrumentationExecutionResponseHook {
(span: Span, responseInfo: MongoResponseHookInformation): void;
}

/**
* Function that can be used to serialize db.statement tag
* @param cmd - MongoDB command object
*
* @returns serialized string that will be used as the db.statement attribute.
*/
export type DbStatementSerializer = (cmd: Record<string, unknown>) => string;

export interface MongoDBInstrumentationConfig extends InstrumentationConfig {
/**
* If true, additional information about query parameters and
Expand All @@ -36,6 +44,11 @@ export interface MongoDBInstrumentationConfig extends InstrumentationConfig {
* @default undefined
*/
responseHook?: MongoDBInstrumentationExecutionResponseHook;

/**
* Custom serializer function for the db.statement tag
*/
dbStatementSerializer?: DbStatementSerializer;
nozik marked this conversation as resolved.
Show resolved Hide resolved
}

export type Func<T> = (...args: unknown[]) => T;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ instrumentation.disable();

import * as mongodb from 'mongodb';
import { assertSpans, accessCollection } from './utils';
import { SemanticAttributes } from '@opentelemetry/semantic-conventions';

describe('MongoDBInstrumentation', () => {
function create(config: MongoDBInstrumentationConfig = {}) {
Expand Down Expand Up @@ -249,6 +250,39 @@ describe('MongoDBInstrumentation', () => {
});
});

describe('when collecting insert data', () => {
beforeEach(() => {
memoryExporter.reset();
create({
enhancedDatabaseReporting: true,
dbStatementSerializer: (commandObj: Record<string, unknown>) => {
return JSON.stringify(commandObj);
},
});
});

it('should collect insert data when configured to do so', done => {
const key = 'key';
const value = 'value';
const object = { [key]: value };
const span = provider.getTracer('default').startSpan('insertRootSpan');
context.with(trace.setSpan(context.active(), span), () => {
collection.insertOne(object).then(() => {
span.end();
const spans = memoryExporter.getFinishedSpans();
const operationName = 'mongodb.insert';
assertSpans(spans, operationName, SpanKind.CLIENT, false, true);
const mongoSpan = spans.find(s => s.name === operationName);
const dbStatement = JSON.parse(
mongoSpan!.attributes[SemanticAttributes.DB_STATEMENT] as string
);
assert.strictEqual(dbStatement[key], value);
done();
});
});
});
});

describe('when specifying a responseHook configuration', () => {
const dataAttributeName = 'mongodb_data';
beforeEach(() => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -86,9 +86,9 @@ export function assertSpans(
assert.strictEqual(mongoSpan.status.code, SpanStatusCode.UNSET);

if (isEnhancedDatabaseReportingEnabled) {
const dbStatement = mongoSpan.attributes[
SemanticAttributes.DB_STATEMENT
] as any;
const dbStatement = JSON.parse(
mongoSpan.attributes[SemanticAttributes.DB_STATEMENT] as string
);
for (const key in dbStatement) {
assert.notStrictEqual(dbStatement[key], '?');
}
Expand Down