From ec7a0d8059c82b728489c23af5c9adf013aa2b75 Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Tue, 24 Dec 2024 19:00:25 -0800 Subject: [PATCH 1/5] [Instrumentation.AWS]: always add context propagation data to requests --- .../AWSTracingPipelineHandler.cs | 41 ++++++++++++------- 1 file changed, 27 insertions(+), 14 deletions(-) diff --git a/src/OpenTelemetry.Instrumentation.AWS/Implementation/AWSTracingPipelineHandler.cs b/src/OpenTelemetry.Instrumentation.AWS/Implementation/AWSTracingPipelineHandler.cs index ef28d3cc5e..c2a9ee300e 100644 --- a/src/OpenTelemetry.Instrumentation.AWS/Implementation/AWSTracingPipelineHandler.cs +++ b/src/OpenTelemetry.Instrumentation.AWS/Implementation/AWSTracingPipelineHandler.cs @@ -46,6 +46,22 @@ public override async Task InvokeAsync(IExecutionContext executionContext) return ret; } + private static void AddPropagationDataToRequest(Activity activity, IRequestContext requestContext) + { + var service = requestContext.ServiceMetaData.ServiceId; + + if (AWSServiceType.IsSqsService(service)) + { + SqsRequestContextHelper.AddAttributes( + requestContext, AWSMessagingUtils.InjectIntoDictionary(new PropagationContext(activity.Context, Baggage.Current))); + } + else if (AWSServiceType.IsSnsService(service)) + { + SnsRequestContextHelper.AddAttributes( + requestContext, AWSMessagingUtils.InjectIntoDictionary(new PropagationContext(activity.Context, Baggage.Current))); + } + } + #if NET [System.Diagnostics.CodeAnalysis.UnconditionalSuppressMessage( "Trimming", @@ -167,16 +183,6 @@ private void AddRequestSpecificInformation(Activity activity, IRequestContext re { this.awsSemanticConventions.TagBuilder.SetTagAttributeDbSystemToDynamoDb(activity); } - else if (AWSServiceType.IsSqsService(service)) - { - SqsRequestContextHelper.AddAttributes( - requestContext, AWSMessagingUtils.InjectIntoDictionary(new PropagationContext(activity.Context, Baggage.Current))); - } - else if (AWSServiceType.IsSnsService(service)) - { - SnsRequestContextHelper.AddAttributes( - requestContext, AWSMessagingUtils.InjectIntoDictionary(new PropagationContext(activity.Context, Baggage.Current))); - } else if (AWSServiceType.IsBedrockRuntimeService(service)) { this.awsSemanticConventions.TagBuilder.SetTagAttributeGenAiSystemToBedrock(activity); @@ -202,14 +208,21 @@ private void ProcessEndRequest(Activity? activity, IExecutionContext executionCo var currentActivity = Activity.Current; - if (currentActivity == null - || !currentActivity.Source.Name.StartsWith(TelemetryConstants.TelemetryScopePrefix, StringComparison.Ordinal) - || !currentActivity.IsAllDataRequested) + if (currentActivity == null) { return null; } - this.AddRequestSpecificInformation(currentActivity, executionContext.RequestContext); + if (currentActivity.IsAllDataRequested + && currentActivity.Source.Name.StartsWith(TelemetryConstants.TelemetryScopePrefix, StringComparison.Ordinal)) + { + this.AddRequestSpecificInformation(currentActivity, executionContext.RequestContext); + } + + // Context propagation should always happen regardless of sampling decision (which affects Activity.IsAllDataRequested and Activity.Source). + // Otherwise, downstream services can make inconsistent sampling decisions and create incomplete traces. + AddPropagationDataToRequest(currentActivity, executionContext.RequestContext); + return currentActivity; } } From c922146def77e261895f7a57bccad5f061a8439c Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Tue, 24 Dec 2024 19:07:31 -0800 Subject: [PATCH 2/5] changelog --- src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md b/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md index 11d55af4be..0ff4530a50 100644 --- a/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md +++ b/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md @@ -2,6 +2,10 @@ ## Unreleased +* Context propagation data is always added to SQS and SNS requests regardless of sampling decision. + This enables downstream services to make consistent sampling decisions and prevents incomplete traces. + ([#2447](https://github.com/open-telemetry/opentelemetry-dotnet-contrib/pull/2447)) + ## 1.10.0-beta.3 Released 2024-Dec-20 From 189367ddbbc2860967e09d6380ae39241e8cf4e4 Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Tue, 31 Dec 2024 17:44:05 -0800 Subject: [PATCH 3/5] add test --- .../TestAWSClientInstrumentation.cs | 50 +++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs b/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs index 487ededdb6..b9a71f6247 100644 --- a/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs +++ b/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs @@ -232,6 +232,56 @@ public async Task TestSQSSendMessageSuccessful() Assert.Equal(requestId, Utils.GetTagValue(awssdk_activity, "aws.request_id")); } + [Fact] +#if NETFRAMEWORK + public void TestSQSSendMessageSuccessfulNotSampled() +#else + public async Task TestSQSSendMessageSuccessfulNotSampled() +#endif + { + var exportedItems = new List(); + + var parent = new Activity("parent").Start(); + var requestId = @"fakerequ-esti-dfak-ereq-uestidfakere"; + + SendMessageRequest send_msg_req; + + using (Sdk.CreateTracerProviderBuilder() + .AddXRayTraceId() + .SetSampler(new AlwaysOffSampler()) + .AddAWSInstrumentation(o => + { + o.SemanticConventionVersion = SemanticConventionVersion.Latest; + }) + .AddInMemoryExporter(exportedItems) + .Build()) + { + var sqs = new AmazonSQSClient(new AnonymousAWSCredentials(), RegionEndpoint.USEast1); + var dummyResponse = "{}"; + CustomResponses.SetResponse(sqs, dummyResponse, requestId, true); + send_msg_req = new SendMessageRequest + { + QueueUrl = "https://sqs.us-east-1.amazonaws.com/123456789/MyTestQueue", + MessageBody = "Hello from OT", + }; + send_msg_req.MessageAttributes.Add("Custom", new MessageAttributeValue { StringValue = "Value", DataType = "String" }); +#if NETFRAMEWORK + sqs.SendMessage(send_msg_req); +#else + await sqs.SendMessageAsync(send_msg_req); +#endif + } + + Assert.Empty(exportedItems); + Assert.Equal(2, send_msg_req.MessageAttributes.Count); + Assert.Contains( + send_msg_req.MessageAttributes, + kv => kv.Key == "traceparent" && kv.Value.StringValue.EndsWith("00", StringComparison.Ordinal)); + Assert.Contains( + send_msg_req.MessageAttributes, + kv => kv.Key == "Custom" && kv.Value.StringValue == "Value"); + } + [Fact] #if NETFRAMEWORK public void TestBedrockGetGuardrailSuccessful() From fc15a27c739c11870f70551efbdf0304cf0bbc41 Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Tue, 31 Dec 2024 17:58:27 -0800 Subject: [PATCH 4/5] check traceparent of both sampled & not sampled SQS requests --- .../TestAWSClientInstrumentation.cs | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs b/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs index b9a71f6247..1e975c5568 100644 --- a/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs +++ b/test/OpenTelemetry.Instrumentation.AWS.Tests/TestAWSClientInstrumentation.cs @@ -185,9 +185,9 @@ public async Task TestDDBScanUnsuccessful() [Fact] #if NETFRAMEWORK - public void TestSQSSendMessageSuccessful() + public void TestSQSSendMessageSuccessfulSampled() #else - public async Task TestSQSSendMessageSuccessful() + public async Task TestSQSSendMessageSuccessfulSampled() #endif { var exportedItems = new List(); @@ -195,6 +195,8 @@ public async Task TestSQSSendMessageSuccessful() var parent = new Activity("parent").Start(); var requestId = @"fakerequ-esti-dfak-ereq-uestidfakere"; + SendMessageRequest send_msg_req; + using (Sdk.CreateTracerProviderBuilder() .AddXRayTraceId() .SetSampler(new AlwaysOnSampler()) @@ -208,7 +210,7 @@ public async Task TestSQSSendMessageSuccessful() var sqs = new AmazonSQSClient(new AnonymousAWSCredentials(), RegionEndpoint.USEast1); var dummyResponse = "{}"; CustomResponses.SetResponse(sqs, dummyResponse, requestId, true); - var send_msg_req = new SendMessageRequest + send_msg_req = new SendMessageRequest { QueueUrl = "https://sqs.us-east-1.amazonaws.com/123456789/MyTestQueue", MessageBody = "Hello from OT", @@ -230,6 +232,14 @@ public async Task TestSQSSendMessageSuccessful() Assert.Equal(ActivityStatusCode.Unset, awssdk_activity.Status); Assert.Equal(requestId, Utils.GetTagValue(awssdk_activity, "aws.request_id")); + + Assert.Equal(2, send_msg_req.MessageAttributes.Count); + Assert.Contains( + send_msg_req.MessageAttributes, + kv => kv.Key == "traceparent" && kv.Value.StringValue == $"00-{awssdk_activity.TraceId}-{awssdk_activity.SpanId}-01"); + Assert.Contains( + send_msg_req.MessageAttributes, + kv => kv.Key == "Custom" && kv.Value.StringValue == "Value"); } [Fact] @@ -273,10 +283,11 @@ public async Task TestSQSSendMessageSuccessfulNotSampled() } Assert.Empty(exportedItems); + Assert.Equal(2, send_msg_req.MessageAttributes.Count); Assert.Contains( send_msg_req.MessageAttributes, - kv => kv.Key == "traceparent" && kv.Value.StringValue.EndsWith("00", StringComparison.Ordinal)); + kv => kv.Key == "traceparent" && kv.Value.StringValue == $"00-{parent.TraceId}-{parent.SpanId}-00"); Assert.Contains( send_msg_req.MessageAttributes, kv => kv.Key == "Custom" && kv.Value.StringValue == "Value"); From 8b17b8ae0e79c5a2ae081b7e08a4b5720d2515e6 Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Sun, 5 Jan 2025 19:21:36 -0800 Subject: [PATCH 5/5] changelog line wrap --- src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md b/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md index 0ff4530a50..faa7a59e30 100644 --- a/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md +++ b/src/OpenTelemetry.Instrumentation.AWS/CHANGELOG.md @@ -2,8 +2,9 @@ ## Unreleased -* Context propagation data is always added to SQS and SNS requests regardless of sampling decision. - This enables downstream services to make consistent sampling decisions and prevents incomplete traces. +* Context propagation data is always added to SQS and SNS requests regardless of + sampling decision. This enables downstream services to make consistent sampling + decisions and prevents incomplete traces. ([#2447](https://github.com/open-telemetry/opentelemetry-dotnet-contrib/pull/2447)) ## 1.10.0-beta.3