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(opentelemetry-instrumentation-aws-sdk): add missing spec-defined DynamoDB attributes #1524

Merged
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,10 @@ import {
} from '../types';

export class DynamodbServiceExtension implements ServiceExtension {
toArray<T>(values: T | T[]): T[] {
return Array.isArray(values) ? values : [values];
}

requestPreSpanHook(normalizedRequest: NormalizedRequest): RequestMetadata {
const spanKind: SpanKind = SpanKind.CLIENT;
let spanName: string | undefined;
Expand All @@ -41,12 +45,140 @@ export class DynamodbServiceExtension implements ServiceExtension {
),
};

if (operation === 'BatchGetItem') {
// normalizedRequest.commandInput.RequestItems) is undefined when no table names are returned
// keys in this object are the table names
if (normalizedRequest.commandInput?.TableName) {
// Necessary for commands with only 1 table name (example: CreateTable). Attribute is TableName not keys of RequestItems
// single table name returned for operations like CreateTable
spanAttributes[SemanticAttributes.AWS_DYNAMODB_TABLE_NAMES] = [
normalizedRequest.commandInput.TableName,
];
} else if (normalizedRequest.commandInput?.RequestItems) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_TABLE_NAMES] = Object.keys(
normalizedRequest.commandInput.RequestItems
);
}

if (operation === 'CreateTable' || operation === 'UpdateTable') {
// only check for ProvisionedThroughput since ReadCapacityUnits and WriteCapacity units are required attributes
if (normalizedRequest.commandInput?.ProvisionedThroughput) {
spanAttributes[
SemanticAttributes.AWS_DYNAMODB_PROVISIONED_READ_CAPACITY
] =
normalizedRequest.commandInput.ProvisionedThroughput.ReadCapacityUnits;
spanAttributes[
SemanticAttributes.AWS_DYNAMODB_PROVISIONED_WRITE_CAPACITY
] =
normalizedRequest.commandInput.ProvisionedThroughput.WriteCapacityUnits;
}
}

if (
operation === 'GetItem' ||
operation === 'Scan' ||
operation === 'Query'
) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_CONSISTENT_READ] =
normalizedRequest.commandInput.ConsistentRead;
}

if (operation === 'Query' || operation === 'Scan') {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_PROJECTION] =
normalizedRequest.commandInput.ProjectionExpression;
carolabadeer marked this conversation as resolved.
Show resolved Hide resolved
}

if (operation === 'CreateTable') {
if (normalizedRequest.commandInput?.GlobalSecondaryIndexes) {
spanAttributes[
SemanticAttributes.AWS_DYNAMODB_GLOBAL_SECONDARY_INDEXES
] = this.toArray(
normalizedRequest.commandInput.GlobalSecondaryIndexes
).map((x: { [DictionaryKey: string]: any }) => JSON.stringify(x));
}

if (normalizedRequest.commandInput?.LocalSecondaryIndexes) {
spanAttributes[
SemanticAttributes.AWS_DYNAMODB_LOCAL_SECONDARY_INDEXES
] = this.toArray(
normalizedRequest.commandInput.LocalSecondaryIndexes
).map((x: { [DictionaryKey: string]: any }) => JSON.stringify(x));
}
}

if (
operation === 'ListTables' ||
operation === 'Query' ||
operation === 'Scan'
) {
if (normalizedRequest.commandInput?.Limit) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_LIMIT] =
normalizedRequest.commandInput.Limit;
}
}

if (operation === 'ListTables') {
if (normalizedRequest.commandInput?.ExclusiveStartTableName) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_EXCLUSIVE_START_TABLE] =
normalizedRequest.commandInput.ExclusiveStartTableName;
}
}

if (operation === 'Query') {
if (normalizedRequest.commandInput?.ScanIndexForward) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_SCAN_FORWARD] =
normalizedRequest.commandInput.ScanIndexForward;
}

if (normalizedRequest.commandInput?.IndexName) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_INDEX_NAME] =
normalizedRequest.commandInput.IndexName;
}

if (normalizedRequest.commandInput?.Select) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_SELECT] =
normalizedRequest.commandInput.Select;
}
}

if (operation === 'Scan') {
if (normalizedRequest.commandInput?.Segment) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_SEGMENT] =
normalizedRequest.commandInput?.Segment;
}

if (normalizedRequest.commandInput?.TotalSegments) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_TOTAL_SEGMENTS] =
normalizedRequest.commandInput?.TotalSegments;
}

if (normalizedRequest.commandInput?.IndexName) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_INDEX_NAME] =
normalizedRequest.commandInput.IndexName;
}

if (normalizedRequest.commandInput?.Select) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_SELECT] =
normalizedRequest.commandInput.Select;
}
}

if (operation === 'UpdateTable') {
if (normalizedRequest.commandInput?.AttributeDefinitions) {
spanAttributes[SemanticAttributes.AWS_DYNAMODB_ATTRIBUTE_DEFINITIONS] =
this.toArray(normalizedRequest.commandInput.AttributeDefinitions).map(
(x: { [DictionaryKey: string]: any }) => JSON.stringify(x)
);
}

if (normalizedRequest.commandInput?.GlobalSecondaryIndexUpdates) {
spanAttributes[
SemanticAttributes.AWS_DYNAMODB_GLOBAL_SECONDARY_INDEX_UPDATES
] = this.toArray(
normalizedRequest.commandInput.GlobalSecondaryIndexUpdates
).map((x: { [DictionaryKey: string]: any }) => JSON.stringify(x));
}
}

return {
isIncoming,
spanAttributes,
Expand All @@ -73,5 +205,43 @@ export class DynamodbServiceExtension implements ServiceExtension {
);
}
}

if (
operation === 'BatchWriteItem' ||
operation === 'CreateTable' ||
operation === 'DeleteItem' ||
operation === 'PutItem' ||
operation === 'UpdateItem'
) {
span.setAttribute(
SemanticAttributes.AWS_DYNAMODB_ITEM_COLLECTION_METRICS,
response.data.ItemCollectionMetrics
carolabadeer marked this conversation as resolved.
Show resolved Hide resolved
);
}

if (operation === 'ListTables') {
if (response.data?.TableNames) {
span.setAttribute(
SemanticAttributes.AWS_DYNAMODB_TABLE_COUNT,
response.data?.TableNames.length
);
}
}

if (operation === 'Scan') {
if (response.data?.Count) {
span.setAttribute(
SemanticAttributes.AWS_DYNAMODB_COUNT,
response.data?.Count
);
}

if (response.data?.ScannedCount) {
span.setAttribute(
SemanticAttributes.AWS_DYNAMODB_SCANNED_COUNT,
response.data?.ScannedCount
);
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,14 @@ describe('DynamoDB', () => {
ExpressionAttributeValues: {
':v': 'val1',
},
ProjectionExpression: 'id',
ScanIndexForward: true,
ConsistentRead: true,
IndexName: 'name_to_group',
Limit: 10,
Select: 'ALL_ATTRIBUTES',
};

dynamodb.query(
params,
(err: AWSError, data: AWS.DynamoDB.DocumentClient.QueryOutput) => {
Expand All @@ -81,6 +88,102 @@ describe('DynamoDB', () => {
);
expect(attrs[SemanticAttributes.DB_NAME]).toStrictEqual('test-table');
expect(attrs[SemanticAttributes.DB_OPERATION]).toStrictEqual('Query');
expect(
carolabadeer marked this conversation as resolved.
Show resolved Hide resolved
attrs[SemanticAttributes.AWS_DYNAMODB_SCAN_FORWARD]
).toStrictEqual(true);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_CONSISTENT_READ]
).toStrictEqual(true);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_INDEX_NAME]
).toStrictEqual('name_to_group');
expect(attrs[SemanticAttributes.AWS_DYNAMODB_SELECT]).toStrictEqual(
'ALL_ATTRIBUTES'
);
expect(attrs[SemanticAttributes.AWS_DYNAMODB_LIMIT]).toStrictEqual(
10
);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_TABLE_NAMES]
).toStrictEqual(['test-table']);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_PROJECTION]
).toStrictEqual('id');
expect(
JSON.parse(attrs[SemanticAttributes.DB_STATEMENT] as string)
).toEqual(params);
expect(err).toBeFalsy();
done();
}
);
});
});

describe('Scan', () => {
beforeEach(() =>
mockV2AwsSend(responseMockSuccess, {
ConsumedCapacity: {
TableName: 'test-table',
CapacityUnits: 0.5,
Table: { CapacityUnits: 0.5 },
},
Count: 10,
ScannedCount: 50,
} as AWS.DynamoDB.Types.ScanOutput)
);

it('should populate specific Scan attributes', done => {
const dynamodb = new AWS.DynamoDB.DocumentClient();
const params = {
TableName: 'test-table',
Item: { key1: 'val1' },
ProjectionExpression: 'id',
ConsistentRead: true,
Segment: 10,
TotalSegments: 100,
IndexName: 'index_name',
Limit: 10,
Select: 'ALL_ATTRIBUTES',
};

dynamodb.scan(
params,
(err: AWSError, data: AWS.DynamoDB.DocumentClient.ScanOutput) => {
const spans = getTestSpans();
expect(spans.length).toStrictEqual(1);
const attrs = spans[0].attributes;
expect(attrs[SemanticAttributes.DB_SYSTEM]).toStrictEqual(
DbSystemValues.DYNAMODB
);
expect(attrs[SemanticAttributes.DB_NAME]).toStrictEqual('test-table');
expect(attrs[SemanticAttributes.DB_OPERATION]).toStrictEqual('Scan');
expect(attrs[SemanticAttributes.AWS_DYNAMODB_SEGMENT]).toStrictEqual(
10
);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_TOTAL_SEGMENTS]
).toStrictEqual(100);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_INDEX_NAME]
).toStrictEqual('index_name');
expect(attrs[SemanticAttributes.AWS_DYNAMODB_SELECT]).toStrictEqual(
'ALL_ATTRIBUTES'
);
expect(attrs[SemanticAttributes.AWS_DYNAMODB_COUNT]).toStrictEqual(
10
);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_SCANNED_COUNT]
).toStrictEqual(50);
expect(attrs[SemanticAttributes.AWS_DYNAMODB_LIMIT]).toStrictEqual(
10
);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_TABLE_NAMES]
).toStrictEqual(['test-table']);
expect(
attrs[SemanticAttributes.AWS_DYNAMODB_PROJECTION]
).toStrictEqual('id');
expect(
JSON.parse(attrs[SemanticAttributes.DB_STATEMENT] as string)
).toEqual(params);
Expand Down