From 4434a8c5a869c80ca68a0ce6720da1125905b789 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Thu, 26 Sep 2024 14:03:36 -0700 Subject: [PATCH 01/28] First pass at instrumentation --- .../Attributes/AttributeDefinitionService.cs | 15 +++ .../Core/Transactions/NoOpTransaction.cs | 5 + .../Agent/Core/Transactions/Transaction.cs | 12 +++ .../Api/ITransaction.cs | 2 + .../Helpers/AwsSdk.cs | 60 ++++++++++++ .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 23 +++++ .../AwsSdk/LambdaInvokeRequestHandler.cs | 97 +++++++++++++++++++ .../Helpers/AwsSdkHelperTests.cs | 36 +++++++ 8 files changed, 250 insertions(+) create mode 100644 src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs create mode 100644 src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs create mode 100644 tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs diff --git a/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs b/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs index c51ea26b61..9f1a7151b8 100644 --- a/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs +++ b/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs @@ -126,6 +126,7 @@ public interface IAttributeDefinitions AttributeDefinition GetLambdaAttribute(string name); AttributeDefinition GetFaasAttribute(string name); + AttributeDefinition GetCloudSdkAttribute(string name); AttributeDefinition GetRequestParameterAttribute(string paramName); @@ -190,6 +191,7 @@ public AttributeDefinitions(IAttributeFilter attribFilter) private readonly ConcurrentDictionary> _requestHeadersAttributes = new ConcurrentDictionary>(); private readonly ConcurrentDictionary> _lambdaAttributes = new ConcurrentDictionary>(); private readonly ConcurrentDictionary> _faasAttributes = new(); + private readonly ConcurrentDictionary> _cloudSdkAttributes = new(); private readonly ConcurrentDictionary> _typeAttributes = new ConcurrentDictionary>(); @@ -281,6 +283,19 @@ public AttributeDefinition GetFaasAttribute(string name) } + private AttributeDefinition CreateCloudSdkAttribute(string attribName) + { + return AttributeDefinitionBuilder + .Create(attribName, AttributeClassification.AgentAttributes) + .AppliesTo(AttributeDestinations.TransactionTrace) + .AppliesTo(AttributeDestinations.SpanEvent) + .Build(_attribFilter); + } + + public AttributeDefinition GetCloudSdkAttribute(string name) + { + return _cloudSdkAttributes.GetOrAdd(name, CreateCloudSdkAttribute); + } public AttributeDefinition GetCustomAttributeForTransaction(string name) { return _trxCustomAttributes.GetOrAdd(name, CreateCustomAttributeForTransaction); diff --git a/src/Agent/NewRelic/Agent/Core/Transactions/NoOpTransaction.cs b/src/Agent/NewRelic/Agent/Core/Transactions/NoOpTransaction.cs index 404cbac7f4..91c3723e52 100644 --- a/src/Agent/NewRelic/Agent/Core/Transactions/NoOpTransaction.cs +++ b/src/Agent/NewRelic/Agent/Core/Transactions/NoOpTransaction.cs @@ -331,5 +331,10 @@ public void AddFaasAttribute(string name, object value) { return; } + + public void AddCloudSdkAttribute(string name, object value) + { + return; + } } } diff --git a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs index 9cbe293901..3706e795d0 100644 --- a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs +++ b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs @@ -1393,5 +1393,17 @@ public void AddFaasAttribute(string name, object value) var faasAttrib = _attribDefs.GetFaasAttribute(name); TransactionMetadata.UserAndRequestAttributes.TrySetValue(faasAttrib, value); } + + public void AddCloudSdkAttribute(string name, object value) + { + if (string.IsNullOrWhiteSpace(name)) + { + Log.Debug($"AddCloudSdkAttribute - Unable to set Cloud value on transaction because the key is null/empty"); + return; + } + + var cloudAttrib = _attribDefs.GetCloudSdkAttribute(name); + TransactionMetadata.UserAndRequestAttributes.TrySetValue(cloudAttrib, value); + } } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs index ea0f22e041..333ea16e6b 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs @@ -311,5 +311,7 @@ ISegment StartMessageBrokerSegment(MethodCall methodCall, MessageBrokerDestinati void AddLambdaAttribute(string name, object value); void AddFaasAttribute(string name, object value); + + void AddCloudSdkAttribute(string name, object value); } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs new file mode 100644 index 0000000000..f473533221 --- /dev/null +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs @@ -0,0 +1,60 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using NewRelic.Agent.Api; + +namespace NewRelic.Agent.Extensions.Helpers +{ + public static class AwsSdkHelpers + { + public static string ConstructArn(IAgent agent, string invocationName, string region, string accountId) + { + if (invocationName.StartsWith("arn:")) + { + if (invocationName.StartsWith("arn:aws:lambda:")) + { + return invocationName; + } + agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); + return null; + } + var segments = invocationName.Split(':'); + string functionName; + + if ((segments.Length == 1) || (segments.Length == 2)) + { + // 'myfunction' or 'myfunction:alias' + // Need account ID to reconstruct ARN + if (string.IsNullOrEmpty(accountId)) + { + agent?.Logger.Debug($"Need account ID in order to resolve function '{invocationName}'"); + return null; + } + functionName = invocationName; + } + else if (segments.Length == 3) + { + // 123456789012:function:my-function' + accountId = segments[0]; + functionName = segments[2]; + } + else if (segments.Length == 4) + { + // 123456789012:function:my-function:myalias + accountId = segments[0]; + functionName = $"{segments[2]}:{segments[3]}"; + } + else + { + agent?.Logger.Debug($"Unable to parse function name '{invocationName}'."); + return null; + } + if (string.IsNullOrEmpty(region)) + { + agent?.Logger.Debug($"Need region in order to resolve function '{invocationName}'"); + return null; + } + return $"arn:aws:lambda:{region}:{accountId}:function:{functionName}"; + } + } +} diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index bc772687c1..f0cf737f9d 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -1,7 +1,9 @@ // Copyright 2020 New Relic, Inc. All rights reserved. // SPDX-License-Identifier: Apache-2.0 +using System; using System.Collections.Generic; +using System.Threading.Tasks; using NewRelic.Agent.Api; using NewRelic.Agent.Extensions.Providers.Wrapper; @@ -19,6 +21,23 @@ public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) return new CanWrapResponse(WrapperName.Equals(methodInfo.RequestedWrapperName)); } + private string GetRegion(IAgent agent, dynamic requestContext) + { + try + { + var clientconfig = requestContext.ClientConfig; + var regionEndpoint = clientconfig.RegionEndpoint; + var systemName = regionEndpoint.SystemName; + return systemName; + } + catch (Exception e) + { + agent.Logger.Debug(e, $"AwsSdkPipelineWrapper: Unable to get region from requestContext."); + } + + return ""; + } + public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction) { // Get the IExecutionContext (the only parameter) @@ -54,6 +73,10 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins { return SQSRequestHandler.HandleSQSRequest(instrumentedMethodCall, agent, transaction, request, isAsync, executionContext); } + else if (requestType == "Amazon.Lambda.Model.InvokeRequest") + { + return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, GetRegion(agent, requestContext)); + } if (_unsupportedRequestTypes.Add(requestType)) // log once per unsupported request type agent.Logger.Debug($"AwsSdkPipelineWrapper: Unsupported request type: {requestType}. Returning NoOp delegate."); diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs new file mode 100644 index 0000000000..66405f3eca --- /dev/null +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -0,0 +1,97 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System.Collections.Concurrent; +using System; +using System.Threading.Tasks; +using NewRelic.Agent.Api; +using NewRelic.Agent.Extensions.Providers.Wrapper; +using NewRelic.Reflection; +using NewRelic.Agent.Extensions.Helpers; + +namespace NewRelic.Providers.Wrapper.AwsSdk +{ + internal static class LambdaInvokeRequestHandler + { + private static ConcurrentDictionary> _getResultFromGenericTask = new(); + + private static object GetTaskResult(object task) + { + if (((Task)task).IsFaulted) + { + return null; + } + + var getResponse = _getResultFromGenericTask.GetOrAdd(task.GetType(), t => VisibilityBypasser.Instance.GeneratePropertyAccessor(t, "Result")); + return getResponse(task); + } + + private static void SetRequestIdIfAvailable(IAgent agent, ITransaction transaction, dynamic response) + { + try + { + dynamic metadata = response.ResponseMetadata; + string requestId = metadata.RequestId; + transaction.AddCloudSdkAttribute("aws.requestId", requestId); + } + catch (Exception e) + { + agent.Logger.Debug(e, "Unable to get RequestId from response metadata."); + } + } + + public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, string region) + { + string functionName = request.FunctionName; + string qualifier = request.Qualifier; + if (!string.IsNullOrEmpty(qualifier) && !functionName.EndsWith(qualifier)) + { + functionName = $"{functionName}:{qualifier}"; + } + string arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); + var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); + + transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); + transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); + transaction.AddCloudSdkAttribute("aws.region", region); + + + if (!string.IsNullOrEmpty(arn)) + { + transaction.AddCloudSdkAttribute("cloud.resource_id", arn); + } + + if (isAsync) + { + return Delegates.GetAsyncDelegateFor(agent, segment, true, InvokeTryProcessResponse, TaskContinuationOptions.ExecuteSynchronously); + + void InvokeTryProcessResponse(Task responseTask) + { + try + { + if (responseTask.Status == TaskStatus.Faulted) + { + transaction.NoticeError(responseTask.Exception); + } + SetRequestIdIfAvailable(agent, transaction, GetTaskResult(responseTask)); + } + finally + { + segment?.End(); + } + } + } + else + { + return Delegates.GetDelegateFor( + onFailure: ex => segment.End(ex), + onSuccess: response => + { + SetRequestIdIfAvailable(agent, transaction, response); + segment.End(); + } + ); + } + } + } +} diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs new file mode 100644 index 0000000000..322210b186 --- /dev/null +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -0,0 +1,36 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using NUnit.Framework; +using NewRelic.Agent.Extensions.Helpers; + +namespace Agent.Extensions.Tests.Helpers +{ + public class AwsSdkHelperTests + { + [Test] + [TestCase("myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("myfunction", "us-west-2", "", null)] + [TestCase("myfunction", "", "123456789012", null)] + [TestCase("myfunction:alias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction:alias")] + [TestCase("myfunction:alias", "us-west-2", "", null)] + [TestCase("123456789012:function:my-function", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function")] + [TestCase("123456789012:function:my-function:myalias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function:myalias")] + [TestCase("123456789012:function:my-function:myalias:extra", "us-west-2", "123456789012", null)] + [TestCase("123456789012:function:my-function:myalias:extra:lots:of:extra:way:too:many", "us-west-2", "123456789012", null)] + [TestCase("arn:aws:", "us-west-2", "123456789012", null)] + [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "us-west-2", "", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("myfunction", "us-east-1", "987654321098", "arn:aws:lambda:us-east-1:987654321098:function:myfunction")] + [TestCase("myfunction:prod", "eu-west-1", "111122223333", "arn:aws:lambda:eu-west-1:111122223333:function:myfunction:prod")] + [TestCase("my-function", "ap-southeast-1", "444455556666", "arn:aws:lambda:ap-southeast-1:444455556666:function:my-function")] + [TestCase("my-function:beta", "ca-central-1", "777788889999", "arn:aws:lambda:ca-central-1:777788889999:function:my-function:beta")] + [TestCase("arn:aws:lambda:eu-central-1:222233334444:function:myfunction", "eu-central-1", "222233334444", "arn:aws:lambda:eu-central-1:222233334444:function:myfunction")] + public void ConstructArn(string name, string region, string accountId, string arn) + { + var constructedArn = AwsSdkHelpers.ConstructArn(null, name, region, accountId); + Assert.That(constructedArn, Is.EqualTo(arn), "Did not get expected ARN"); + } + } +} From 5475b4935ed5d696bdb4d799088a8470e1c29cd7 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 30 Sep 2024 09:13:11 -0700 Subject: [PATCH 02/28] First pass at integration tests --- .../AwsSdk/InvokeLambdaTests.cs | 126 ++++++++++++++++++ .../MFALatestPackages.csproj | 5 +- .../MultiFunctionApplicationHelpers.csproj | 8 ++ .../AwsSdk/InvokeLambdaExerciser.cs | 93 +++++++++++++ 4 files changed, 231 insertions(+), 1 deletion(-) create mode 100644 tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs create mode 100644 tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs diff --git a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs new file mode 100644 index 0000000000..3c959da12f --- /dev/null +++ b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs @@ -0,0 +1,126 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System; +using System.Collections.Generic; +using System.Drawing; +using System.Linq; +using NewRelic.Agent.IntegrationTestHelpers; +using NewRelic.Agent.IntegrationTestHelpers.RemoteServiceFixtures; +using Xunit; +using Xunit.Abstractions; + +namespace NewRelic.Agent.IntegrationTests.AwsSdk +{ + public abstract class InvokeLambdaTestBase : NewRelicIntegrationTest + where TFixture : ConsoleDynamicMethodFixture + { + private readonly TFixture _fixture; + private readonly string _function; + private readonly string _qualifier; + private readonly string _arn; + private readonly bool _isAsync; + + public InvokeLambdaTestBase(TFixture fixture, ITestOutputHelper output, bool async, string function, string qualifier, string arn) : base(fixture) + { + _function = function; + _qualifier = qualifier; + _arn = arn; + _isAsync = async; + + _fixture = fixture; + _fixture.SetTimeout(TimeSpan.FromMinutes(20)); + + _fixture.TestLogger = output; + _fixture.AddActions( + setupConfiguration: () => + { + new NewRelicConfigModifier(fixture.DestinationNewRelicConfigFilePath) + .ForceTransactionTraces() + .SetLogLevel("finest"); + }, + exerciseApplication: () => + { + _fixture.AgentLog.WaitForLogLines(AgentLogBase.TransactionTransformCompletedLogLineRegex, TimeSpan.FromMinutes(20), 2); + } + ); + + if (async) + { + _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaAsync {_function}:{_qualifier} \"fakepayload\""); + _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaAsyncWithQualifier {_function} {_qualifier} \"fakepayload\""); + } + else + { + _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync {_function} \"fakepayload\""); + } + _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync fakefunction fakepayload"); + + _fixture.Initialize(); + } + + [Fact] + public void InvokeLambda() + { + var metrics = _fixture.AgentLog.GetMetrics().ToList(); + + var expectedMetrics = new List + { + }; + + Assertions.MetricsExist(expectedMetrics, metrics); + + var transactions = _fixture.AgentLog.GetTransactionEvents(); + + Assert.NotNull(transactions); + + var spans = _fixture.AgentLog.GetSpanEvents() + .Where(e => e.AgentAttributes.ContainsKey("cloud.resource_id")) + .ToList(); + + Assert.Equal(_isAsync ? 2 : 1, spans.Count); + + foreach (var span in spans) + { + Assert.Equal(_arn, span.AgentAttributes["cloud.resource_id"]); + Assert.Equal("aws_lambda", span.AgentAttributes["cloud.platform"]); + Assert.Equal("InvokeRequest", span.AgentAttributes["aws.operation"]); + Assert.Equal("us-west-2", span.AgentAttributes["aws.region"]); + } + } + } + [NetFrameworkTest] + public class InvokeLambdaTest_Sync_FW462 : InvokeLambdaTestBase + { + public InvokeLambdaTest_Sync_FW462(ConsoleDynamicMethodFixtureFW462 fixture, ITestOutputHelper output) + : base(fixture, output, false, "342444490463:NotARealFunction", null, "arn:aws:lambda:us-west-2:342444490463:function:NotARealFunction") + { + } + } + [NetFrameworkTest] + public class InvokeLambdaTest_Sync_FWLatest : InvokeLambdaTestBase + { + public InvokeLambdaTest_Sync_FWLatest(ConsoleDynamicMethodFixtureFWLatest fixture, ITestOutputHelper output) + : base(fixture, output, false, "342444490463:NotARealFunction", null, "arn:aws:lambda:us-west-2:342444490463:function:NotARealFunction") + { + } + } + [NetCoreTest] + public class InvokeLambdaTest_Async_CoreOldest : InvokeLambdaTestBase + { + public InvokeLambdaTest_Async_CoreOldest(ConsoleDynamicMethodFixtureCoreOldest fixture, ITestOutputHelper output) + : base(fixture, output, true, "342444490463:NotARealFunction", "NotARealAlias", "arn:aws:lambda:us-west-2:342444490463:function:NotARealFunction:NotARealAlias") + { + } + } + + [NetCoreTest] + public class InvokeLambdaTest_Async_CoreLatest : InvokeLambdaTestBase + { + public InvokeLambdaTest_Async_CoreLatest(ConsoleDynamicMethodFixtureCoreLatest fixture, ITestOutputHelper output) + : base(fixture, output, true, "342444490463:NotARealFunction", "NotARealAlias", "arn:aws:lambda:us-west-2:342444490463:function:NotARealFunction:NotARealAlias") + { + } + } + +} diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj index fa488c6a6f..edf37575ac 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj @@ -7,7 +7,10 @@ - + + + + diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj index 13c4870720..57f498b5e1 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj @@ -272,6 +272,14 @@ + + + + + + + + diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs new file mode 100644 index 0000000000..e02532ab8e --- /dev/null +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs @@ -0,0 +1,93 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System.IO; +using System.Security.AccessControl; +using System.Threading.Tasks; +using Amazon; +using NewRelic.Agent.IntegrationTests.Shared.ReflectionHelpers; +using NewRelic.Api.Agent; + +namespace MultiFunctionApplicationHelpers.NetStandardLibraries.AwsSdk +{ + [Library] + public class InvokeLambdaExerciser + { + [LibraryMethod] + [Transaction] + public void InvokeLambdaSync(string function, string payload) + { +#if NETFRAMEWORK + var client = new Amazon.Lambda.AmazonLambdaClient(RegionEndpoint.USWest2); + var request = new Amazon.Lambda.Model.InvokeRequest + { + FunctionName = function, + Payload = payload + }; + + // Note that we aren't invoking a lambda that exists! This is guaranteed to fail, but all we care + // about is that the agent is able to instrument the call. + try + { + var response = client.Invoke(request); + } + catch + { + } +#endif + } + + [LibraryMethod] + [Transaction] + public async Task InvokeLambdaAsync(string function, string payload) + { + var client = new Amazon.Lambda.AmazonLambdaClient(RegionEndpoint.USWest2); + var request = new Amazon.Lambda.Model.InvokeRequest + { + FunctionName = function, + Payload = payload + }; + + // Note that we aren't invoking a lambda that exists! This is guaranteed to fail, but all we care + // about is that the agent is able to instrument the call. + try + { + var response = await client.InvokeAsync(request); + MemoryStream stream = response.Payload; + string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); + return returnValue; + } + catch + { + } + return null; + } + + [LibraryMethod] + [Transaction] + public async Task InvokeLambdaAsyncWithQualifier(string function, string qualifier, string payload) + { + var client = new Amazon.Lambda.AmazonLambdaClient(RegionEndpoint.USWest2); + var request = new Amazon.Lambda.Model.InvokeRequest + { + FunctionName = function, + Qualifier = qualifier, + Payload = payload + }; + + // Note that we aren't invoking a lambda that exists! This is guaranteed to fail, but all we care + // about is that the agent is able to instrument the call. + try + { + var response = await client.InvokeAsync(request); + MemoryStream stream = response.Payload; + string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); + return returnValue; + } + catch + { + } + return null; + } + } +} From 3d213e6ed5ce3b0c37a46663a4bd0dd38f867661 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 8 Oct 2024 08:32:18 -0700 Subject: [PATCH 03/28] Update ARN parsing logic --- .../Helpers/AwsSdk.cs | 93 +++++++++++++------ 1 file changed, 65 insertions(+), 28 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs index f473533221..4a66b9079e 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs @@ -1,59 +1,96 @@ // Copyright 2020 New Relic, Inc. All rights reserved. // SPDX-License-Identifier: Apache-2.0 +using System.Collections.Generic; +using System.Linq; +using System.Text.RegularExpressions; using NewRelic.Agent.Api; namespace NewRelic.Agent.Extensions.Helpers { public static class AwsSdkHelpers { + private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); + private static bool LooksLikeARegion(string text) => RegionRegex.IsMatch(text); + private static bool LooksLikeAnAccountId(string text) => (text.Length == 12) && text.All(c => c >= '0' && c <= '9'); + // Only log ARNs we can't parse once + private static HashSet BadInvocations = new HashSet(); + + // This is the full regex pattern for an ARN: + // (arn:(aws[a-zA-Z-]*)?:lambda:)?([a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}:)?(\d{12}:)?(function:)?([a-zA-Z0-9-_\.]+)(:(\$LATEST|[a-zA-Z0-9-_]+))? + + // If it's a full ARN, it has to start with 'arn:' + // A partial ARN can contain up to 5 segments separated by ':' + // 1. Region + // 2. Account ID + // 3. 'function' (fixed string) + // 4. Function name + // 5. Alias or version + // Only the function name is required, the reset are all optional. e.g. you could have region and function name and nothing else + + // Note that this will not catch functions where the name also looks like a region or account ID public static string ConstructArn(IAgent agent, string invocationName, string region, string accountId) { if (invocationName.StartsWith("arn:")) { - if (invocationName.StartsWith("arn:aws:lambda:")) - { - return invocationName; - } - agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); - return null; + return invocationName; } var segments = invocationName.Split(':'); - string functionName; + string functionName = null; + string alias = null; - if ((segments.Length == 1) || (segments.Length == 2)) + foreach (var segment in segments) { - // 'myfunction' or 'myfunction:alias' - // Need account ID to reconstruct ARN - if (string.IsNullOrEmpty(accountId)) + if (LooksLikeARegion(segment) && string.IsNullOrEmpty(region)) { - agent?.Logger.Debug($"Need account ID in order to resolve function '{invocationName}'"); + region = segment; + } + else if (LooksLikeAnAccountId(segment) && string.IsNullOrEmpty(accountId)) + { + accountId = segment; + } + else if (segment == "function") + { + continue; + } + else if (functionName == null) + { + functionName = segment; + } + else if (alias == null) + { + alias = segment; + } + else + { + if (BadInvocations.Add(invocationName)) + { + agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); + } return null; } - functionName = invocationName; - } - else if (segments.Length == 3) - { - // 123456789012:function:my-function' - accountId = segments[0]; - functionName = segments[2]; } - else if (segments.Length == 4) - { - // 123456789012:function:my-function:myalias - accountId = segments[0]; - functionName = $"{segments[2]}:{segments[3]}"; - } - else + + if (string.IsNullOrEmpty(accountId)) { - agent?.Logger.Debug($"Unable to parse function name '{invocationName}'."); + if (BadInvocations.Add(invocationName)) + { + agent?.Logger.Debug($"Need account ID in order to resolve function '{invocationName}'"); + } return null; } if (string.IsNullOrEmpty(region)) { - agent?.Logger.Debug($"Need region in order to resolve function '{invocationName}'"); + if (BadInvocations.Add(invocationName)) + { + agent?.Logger.Debug($"Need region in order to resolve function '{invocationName}'"); + } return null; } + if (!string.IsNullOrEmpty(alias)) + { + return $"arn:aws:lambda:{region}:{accountId}:function:{functionName}:{alias}"; + } return $"arn:aws:lambda:{region}:{accountId}:function:{functionName}"; } } From 7c33292808f731bb7bdf26f76ef1f9eaebd5a990 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 8 Oct 2024 11:34:46 -0700 Subject: [PATCH 04/28] Updating ARN parsing logic --- .../Helpers/AwsSdk.cs | 43 ++++++++++++++++--- .../Helpers/AwsSdkHelperTests.cs | 16 ++++++- 2 files changed, 52 insertions(+), 7 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs index 4a66b9079e..716e18b5b9 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs @@ -27,8 +27,6 @@ public static class AwsSdkHelpers // 4. Function name // 5. Alias or version // Only the function name is required, the reset are all optional. e.g. you could have region and function name and nothing else - - // Note that this will not catch functions where the name also looks like a region or account ID public static string ConstructArn(IAgent agent, string invocationName, string region, string accountId) { if (invocationName.StartsWith("arn:")) @@ -38,16 +36,33 @@ public static string ConstructArn(IAgent agent, string invocationName, string re var segments = invocationName.Split(':'); string functionName = null; string alias = null; + string fallback = null; foreach (var segment in segments) { - if (LooksLikeARegion(segment) && string.IsNullOrEmpty(region)) + if (LooksLikeARegion(segment)) { - region = segment; + if (string.IsNullOrEmpty(region)) + { + region = segment; + } + else + { + fallback = segment; + } + continue; } - else if (LooksLikeAnAccountId(segment) && string.IsNullOrEmpty(accountId)) + else if (LooksLikeAnAccountId(segment)) { - accountId = segment; + if (string.IsNullOrEmpty(accountId)) + { + accountId = segment; + } + else + { + fallback = segment; + } + continue; } else if (segment == "function") { @@ -71,6 +86,22 @@ public static string ConstructArn(IAgent agent, string invocationName, string re } } + if (string.IsNullOrEmpty(functionName)) + { + if (!string.IsNullOrEmpty(fallback)) + { + functionName = fallback; + } + else + { + if (BadInvocations.Add(invocationName)) + { + agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); + } + return null; + } + } + if (string.IsNullOrEmpty(accountId)) { if (BadInvocations.Add(invocationName)) diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index 322210b186..1382e0d093 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -18,7 +18,7 @@ public class AwsSdkHelperTests [TestCase("123456789012:function:my-function:myalias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function:myalias")] [TestCase("123456789012:function:my-function:myalias:extra", "us-west-2", "123456789012", null)] [TestCase("123456789012:function:my-function:myalias:extra:lots:of:extra:way:too:many", "us-west-2", "123456789012", null)] - [TestCase("arn:aws:", "us-west-2", "123456789012", null)] + [TestCase("arn:aws:", "us-west-2", "123456789012", "arn:aws:")] [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "us-west-2", "", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] [TestCase("arn:aws:lambda:us-west-2:123456789012:function:myfunction", "", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] @@ -27,6 +27,20 @@ public class AwsSdkHelperTests [TestCase("my-function", "ap-southeast-1", "444455556666", "arn:aws:lambda:ap-southeast-1:444455556666:function:my-function")] [TestCase("my-function:beta", "ca-central-1", "777788889999", "arn:aws:lambda:ca-central-1:777788889999:function:my-function:beta")] [TestCase("arn:aws:lambda:eu-central-1:222233334444:function:myfunction", "eu-central-1", "222233334444", "arn:aws:lambda:eu-central-1:222233334444:function:myfunction")] + [TestCase("us-west-2:myfunction", null, "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("us-west-2:myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] + [TestCase("us-west-2:myfunction", "us-west-2", "", null)] + [TestCase("us-west-2:myfunction:alias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction:alias")] + [TestCase("us-west-2:myfunction:alias", "us-west-2", "", null)] + [TestCase("123456789012:my-function", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function")] + [TestCase("123456789012:my-function:myalias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function:myalias")] + [TestCase("123456789012:my-function:myalias:extra", "us-west-2", "123456789012", null)] + [TestCase("123456789012:my-function:myalias:extra:lots:of:extra:way:too:many", "us-west-2", "123456789012", null)] + [TestCase("eu-west-1:us-west-2", "eu-west-1", "123456789012", "arn:aws:lambda:eu-west-1:123456789012:function:us-west-2")] + // Edge cases: functions that look like account IDs or region names + [TestCase("123456789012:444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] + [TestCase("444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] + [TestCase("us-west-2", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:us-west-2")] public void ConstructArn(string name, string region, string accountId, string arn) { var constructedArn = AwsSdkHelpers.ConstructArn(null, name, region, accountId); From c15328cb0ad56f25a253af843ba6d848aeff7bc0 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Wed, 9 Oct 2024 14:34:04 -0700 Subject: [PATCH 05/28] Fixes to integration tests --- .../AwsSdk/InvokeLambdaTests.cs | 36 +++++++++++++------ .../AwsSdk/InvokeLambdaExerciser.cs | 3 ++ 2 files changed, 28 insertions(+), 11 deletions(-) diff --git a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs index 3c959da12f..cb243612b7 100644 --- a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs +++ b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs @@ -41,7 +41,7 @@ public InvokeLambdaTestBase(TFixture fixture, ITestOutputHelper output, bool asy }, exerciseApplication: () => { - _fixture.AgentLog.WaitForLogLines(AgentLogBase.TransactionTransformCompletedLogLineRegex, TimeSpan.FromMinutes(20), 2); + _fixture.AgentLog.WaitForLogLines(AgentLogBase.TransactionTransformCompletedLogLineRegex, TimeSpan.FromMinutes(20),2); } ); @@ -53,9 +53,10 @@ public InvokeLambdaTestBase(TFixture fixture, ITestOutputHelper output, bool asy else { _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync {_function} \"fakepayload\""); - } - _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync fakefunction fakepayload"); + // This will fail without an ARN because there's no account ID + _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync fakefunction fakepayload"); + } _fixture.Initialize(); } @@ -66,27 +67,40 @@ public void InvokeLambda() var expectedMetrics = new List { + new Assertions.ExpectedMetric {metricName = @"DotNet/InvokeRequest", CallCountAllHarvests = 2}, }; - Assertions.MetricsExist(expectedMetrics, metrics); var transactions = _fixture.AgentLog.GetTransactionEvents(); + Assert.Equal(2, transactions.Count()); - Assert.NotNull(transactions); + foreach (var transaction in transactions) + { + Assert.StartsWith("OtherTransaction/Custom/MultiFunctionApplicationHelpers.NetStandardLibraries.AwsSdk.InvokeLambdaExerciser/InvokeLambda", transaction.IntrinsicAttributes["name"].ToString()); + } - var spans = _fixture.AgentLog.GetSpanEvents() - .Where(e => e.AgentAttributes.ContainsKey("cloud.resource_id")) + var allSpans = _fixture.AgentLog.GetSpanEvents() + .Where(e => e.AgentAttributes.ContainsKey("cloud.platform")) .ToList(); + Assert.Equal(2, allSpans.Count); - Assert.Equal(_isAsync ? 2 : 1, spans.Count); - - foreach (var span in spans) + foreach (var span in allSpans) { - Assert.Equal(_arn, span.AgentAttributes["cloud.resource_id"]); Assert.Equal("aws_lambda", span.AgentAttributes["cloud.platform"]); Assert.Equal("InvokeRequest", span.AgentAttributes["aws.operation"]); Assert.Equal("us-west-2", span.AgentAttributes["aws.region"]); } + + // There should be one fewer span in this list, because there's one where there wasn't + // enough info to create an ARN + var spansWithArn = _fixture.AgentLog.GetSpanEvents() + .Where(e => e.AgentAttributes.ContainsKey("cloud.resource_id")) + .ToList(); + Assert.Equal(_isAsync ? 2 : 1, spansWithArn.Count); + foreach (var span in spansWithArn) + { + Assert.Equal(_arn, span.AgentAttributes["cloud.resource_id"]); + } } } [NetFrameworkTest] diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs index e02532ab8e..574707aa0f 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs @@ -1,6 +1,7 @@ // Copyright 2020 New Relic, Inc. All rights reserved. // SPDX-License-Identifier: Apache-2.0 +using System; using System.IO; using System.Security.AccessControl; using System.Threading.Tasks; @@ -34,6 +35,8 @@ public void InvokeLambdaSync(string function, string payload) catch { } +#else + throw new Exception($"Synchronous calls are only supported on .NET Framework!"); #endif } From 8728cffd19cd7d140ae2381c35b8fa853fc2eb8b Mon Sep 17 00:00:00 2001 From: chynesNR Date: Wed, 9 Oct 2024 16:43:34 -0700 Subject: [PATCH 06/28] Forgot to add tests to workflow --- .github/workflows/all_solutions.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/all_solutions.yml b/.github/workflows/all_solutions.yml index 91b5ffcc09..5a3e975486 100644 --- a/.github/workflows/all_solutions.yml +++ b/.github/workflows/all_solutions.yml @@ -232,6 +232,7 @@ jobs: AwsLambda.Sns, AwsLambda.Sqs, AwsLambda.WebRequest, + AwsSdk, AzureFunction, BasicInstrumentation, CatInbound, From cdb0bf381cdcdf8f3abd60c2398903b4905c8547 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 14 Oct 2024 14:58:36 -0700 Subject: [PATCH 07/28] Don't generate a segment for the HTTP request --- .../Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index 66405f3eca..d4bd94e0f6 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -5,6 +5,7 @@ using System; using System.Threading.Tasks; using NewRelic.Agent.Api; +using NewRelic.Agent.Api.Experimental; using NewRelic.Agent.Extensions.Providers.Wrapper; using NewRelic.Reflection; using NewRelic.Agent.Extensions.Helpers; @@ -50,6 +51,7 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC } string arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); + segment.GetExperimentalApi().MakeLeaf(); transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); From 10a5e4bab1341a6627f5485728c717261649a384 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 14 Oct 2024 15:08:23 -0700 Subject: [PATCH 08/28] Fix some typos --- .../NewRelic.Agent.Extensions/Helpers/AwsSdk.cs | 2 +- .../IntegrationTests/AwsSdk/InvokeLambdaTests.cs | 10 +++++----- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs index 716e18b5b9..f9408211d1 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs @@ -26,7 +26,7 @@ public static class AwsSdkHelpers // 3. 'function' (fixed string) // 4. Function name // 5. Alias or version - // Only the function name is required, the reset are all optional. e.g. you could have region and function name and nothing else + // Only the function name is required, the rest are all optional. e.g. you could have region and function name and nothing else public static string ConstructArn(IAgent agent, string invocationName, string region, string accountId) { if (invocationName.StartsWith("arn:")) diff --git a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs index cb243612b7..464a4128d2 100644 --- a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs +++ b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs @@ -21,15 +21,15 @@ public abstract class InvokeLambdaTestBase : NewRelicIntegrationTest { - _fixture.AgentLog.WaitForLogLines(AgentLogBase.TransactionTransformCompletedLogLineRegex, TimeSpan.FromMinutes(20),2); + _fixture.AgentLog.WaitForLogLines(AgentLogBase.TransactionTransformCompletedLogLineRegex, TimeSpan.FromMinutes(2),2); } ); - if (async) + if (useAsync) { _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaAsync {_function}:{_qualifier} \"fakepayload\""); _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaAsyncWithQualifier {_function} {_qualifier} \"fakepayload\""); From db2202ca4a23c84a7fa6d360998ca588e5ec4cd0 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 15 Oct 2024 10:04:44 -0700 Subject: [PATCH 09/28] Cache constructed ARNs --- .../Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index d4bd94e0f6..2beaef823d 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -15,6 +15,7 @@ namespace NewRelic.Providers.Wrapper.AwsSdk internal static class LambdaInvokeRequestHandler { private static ConcurrentDictionary> _getResultFromGenericTask = new(); + private static ConcurrentDictionary _arnCache = new ConcurrentDictionary(); private static object GetTaskResult(object task) { @@ -49,7 +50,12 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC { functionName = $"{functionName}:{qualifier}"; } - string arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); + string arn; + if (!_arnCache.TryGetValue(functionName, out arn)) + { + arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); + _arnCache.TryAdd(functionName, arn); + } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); segment.GetExperimentalApi().MakeLeaf(); From a14adbbb7813116d4d1191a422d87128afc36851 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 15 Oct 2024 14:55:09 -0700 Subject: [PATCH 10/28] PR feedback --- .../Agent/Core/Transactions/Transaction.cs | 6 ++-- .../Collections/ConcurrentHashSet.cs | 8 ++++- .../Helpers/AwsSdk.cs | 32 +++++++++---------- .../Wrapper/AwsLambda/HandlerMethodWrapper.cs | 6 ++-- .../AwsSdk/LambdaInvokeRequestHandler.cs | 25 +++++++++++---- .../MFALatestPackages.csproj | 4 +-- .../MultiFunctionApplicationHelpers.csproj | 6 ++-- .../Collections/ConcurrentHashSet.cs | 17 ++++++++++ 8 files changed, 68 insertions(+), 36 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs index 3706e795d0..501fa9a2dd 100644 --- a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs +++ b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs @@ -1374,7 +1374,7 @@ public void AddLambdaAttribute(string name, object value) { if (string.IsNullOrWhiteSpace(name)) { - Log.Debug($"AddLambdaAttribute - Unable to set Lambda value on transaction because the key is null/empty"); + Log.Debug($"AddLambdaAttribute - Name cannot be null/empty"); return; } @@ -1386,7 +1386,7 @@ public void AddFaasAttribute(string name, object value) { if (string.IsNullOrWhiteSpace(name)) { - Log.Debug($"AddFaasAttribute - Unable to set FaaS value on transaction because the key is null/empty"); + Log.Debug($"AddFaasAttribute - Name cannot be null/empty"); return; } @@ -1398,7 +1398,7 @@ public void AddCloudSdkAttribute(string name, object value) { if (string.IsNullOrWhiteSpace(name)) { - Log.Debug($"AddCloudSdkAttribute - Unable to set Cloud value on transaction because the key is null/empty"); + Log.Debug($"AddCloudSdkAttribute - Name cannot be null/empty"); return; } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Collections/ConcurrentHashSet.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Collections/ConcurrentHashSet.cs index 28e8133cf2..145d4391be 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Collections/ConcurrentHashSet.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Collections/ConcurrentHashSet.cs @@ -47,7 +47,13 @@ public void Add(T item) _hashSet.Add(item); } } - + public bool TryAdd(T item) + { + using (_writeLock()) + { + return _hashSet.Add(item); + } + } public void Clear() { using (_writeLock()) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs index f9408211d1..036bd02f09 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs @@ -5,6 +5,7 @@ using System.Linq; using System.Text.RegularExpressions; using NewRelic.Agent.Api; +using NewRelic.Agent.Extensions.Collections; namespace NewRelic.Agent.Extensions.Helpers { @@ -14,7 +15,16 @@ public static class AwsSdkHelpers private static bool LooksLikeARegion(string text) => RegionRegex.IsMatch(text); private static bool LooksLikeAnAccountId(string text) => (text.Length == 12) && text.All(c => c >= '0' && c <= '9'); // Only log ARNs we can't parse once - private static HashSet BadInvocations = new HashSet(); + private static ConcurrentHashSet BadInvocations = new ConcurrentHashSet(); + private const int MAX_BAD_INVOCATIONS = 25; + + private static void ReportBadInvocation(IAgent agent, string invocationName) + { + if ((agent?.Logger.IsDebugEnabled == true) && (BadInvocations.Count < MAX_BAD_INVOCATIONS) && BadInvocations.TryAdd(invocationName)) + { + agent.Logger.Debug($"Unable to parse function name '{invocationName}'"); + } + } // This is the full regex pattern for an ARN: // (arn:(aws[a-zA-Z-]*)?:lambda:)?([a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}:)?(\d{12}:)?(function:)?([a-zA-Z0-9-_\.]+)(:(\$LATEST|[a-zA-Z0-9-_]+))? @@ -78,10 +88,7 @@ public static string ConstructArn(IAgent agent, string invocationName, string re } else { - if (BadInvocations.Add(invocationName)) - { - agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); - } + ReportBadInvocation(agent, invocationName); return null; } } @@ -94,28 +101,19 @@ public static string ConstructArn(IAgent agent, string invocationName, string re } else { - if (BadInvocations.Add(invocationName)) - { - agent?.Logger.Debug($"Unable to parse function name '{invocationName}'"); - } + ReportBadInvocation(agent, invocationName); return null; } } if (string.IsNullOrEmpty(accountId)) { - if (BadInvocations.Add(invocationName)) - { - agent?.Logger.Debug($"Need account ID in order to resolve function '{invocationName}'"); - } + ReportBadInvocation(agent, invocationName); return null; } if (string.IsNullOrEmpty(region)) { - if (BadInvocations.Add(invocationName)) - { - agent?.Logger.Debug($"Need region in order to resolve function '{invocationName}'"); - } + ReportBadInvocation(agent, invocationName); return null; } if (!string.IsNullOrEmpty(alias)) diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsLambda/HandlerMethodWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsLambda/HandlerMethodWrapper.cs index 1fdc696c79..ac2a32c79b 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsLambda/HandlerMethodWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsLambda/HandlerMethodWrapper.cs @@ -193,10 +193,9 @@ private void InitLambdaData(InstrumentedMethodCall instrumentedMethodCall, IAgen { agent.Logger.Log(Agent.Extensions.Logging.Level.Debug, $"Supported Event Type found: {_functionDetails.EventType}"); } - else if (!_unsupportedInputTypes.Contains(name)) + else if (_unsupportedInputTypes.TryAdd(name)) { agent.Logger.Log(Agent.Extensions.Logging.Level.Warn, $"Unsupported input object type: {name}. Unable to provide additional instrumentation."); - _unsupportedInputTypes.Add(name); } } @@ -344,10 +343,9 @@ private void CaptureResponseData(ITransaction transaction, object response, IAge || (_functionDetails.EventType == AwsLambdaEventType.ApplicationLoadBalancerRequest && responseType != "Amazon.Lambda.ApplicationLoadBalancerEvents.ApplicationLoadBalancerResponse")) { - if (!_unexpectedResponseTypes.Contains(responseType)) + if (_unexpectedResponseTypes.TryAdd(responseType)) { agent.Logger.Log(Agent.Extensions.Logging.Level.Warn, $"Unexpected response type {responseType} for request event type {_functionDetails.EventType}. Not capturing any response data."); - _unexpectedResponseTypes.Add(responseType); } return; diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index 2beaef823d..f2795dfd09 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -14,8 +14,9 @@ namespace NewRelic.Providers.Wrapper.AwsSdk { internal static class LambdaInvokeRequestHandler { - private static ConcurrentDictionary> _getResultFromGenericTask = new(); + private static Func _getResultFromGenericTask; private static ConcurrentDictionary _arnCache = new ConcurrentDictionary(); + private static bool _reportMissingRequestId = true; private static object GetTaskResult(object task) { @@ -24,7 +25,7 @@ private static object GetTaskResult(object task) return null; } - var getResponse = _getResultFromGenericTask.GetOrAdd(task.GetType(), t => VisibilityBypasser.Instance.GeneratePropertyAccessor(t, "Result")); + var getResponse = _getResultFromGenericTask ??= VisibilityBypasser.Instance.GeneratePropertyAccessor(task.GetType(), "Result"); return getResponse(task); } @@ -38,7 +39,11 @@ private static void SetRequestIdIfAvailable(IAgent agent, ITransaction transacti } catch (Exception e) { - agent.Logger.Debug(e, "Unable to get RequestId from response metadata."); + if (_reportMissingRequestId) + { + agent.Logger.Debug(e, "Unable to get RequestId from response metadata."); + _reportMissingRequestId = false; + } } } @@ -51,10 +56,18 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC functionName = $"{functionName}:{qualifier}"; } string arn; - if (!_arnCache.TryGetValue(functionName, out arn)) + if (functionName.StartsWith("arn:")) { - arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); - _arnCache.TryAdd(functionName, arn); + arn = functionName; + } + else + { + string key = $"{region}:{functionName}"; + if (!_arnCache.TryGetValue(key, out arn)) + { + arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); + _arnCache.TryAdd(key, arn); + } } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); segment.GetExperimentalApi().MakeLeaf(); diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj index b4473527e2..02f30315c9 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj @@ -9,8 +9,8 @@ - - + + diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj index 604e2e7804..b57276e597 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj @@ -277,10 +277,10 @@ - + - - + + diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs index 85a69a634d..3245e2e9e4 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs @@ -96,6 +96,23 @@ public void ConcurrentHashSet_IsThreadSafe() tasks.ForEach(task => task.Wait()); } + [Test] + [TestCase(new[] { 1, 2, 3 }, new[] { true, true, true })] + [TestCase(new[] { 1, 2, 2, 3 }, new[] { true, true, false, true })] + [TestCase(new[] { 4, 4, 4, 4 }, new[] { true, false, false, false })] + [TestCase(new[] { 5, 6, 7, 8, 9 }, new[] { true, true, true, true, true })] + [TestCase(new[] { 10, 10, 11, 11, 12 }, new[] { true, false, true, false, true })] + public void ConcurrentHashSet_TryAdd(int[] values, bool[] results) + { + for (var i = 0; i < values.Length; i++) + { + var value = values[i]; + var result = results[i]; + + Assert.That(_concurrentHashSet.TryAdd(value), Is.EqualTo(result)); + } + } + private static void ExerciseFullApi(ConcurrentHashSet hashSet, int[] numbersToAdd) { dynamic _; From 28a5ebe3da60907abbc07e97a989e89bfb5e3e65 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Wed, 16 Oct 2024 09:10:03 -0700 Subject: [PATCH 11/28] Fix unit test --- .../Collections/ConcurrentHashSet.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs index 3245e2e9e4..2f4e589362 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Collections/ConcurrentHashSet.cs @@ -104,12 +104,13 @@ public void ConcurrentHashSet_IsThreadSafe() [TestCase(new[] { 10, 10, 11, 11, 12 }, new[] { true, false, true, false, true })] public void ConcurrentHashSet_TryAdd(int[] values, bool[] results) { + ConcurrentHashSet set = new(); for (var i = 0; i < values.Length; i++) { var value = values[i]; var result = results[i]; - Assert.That(_concurrentHashSet.TryAdd(value), Is.EqualTo(result)); + Assert.That(set.TryAdd(value), Is.EqualTo(result)); } } From 434018102c5417004da9d7ccba7214bdcbbdbedb Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 28 Oct 2024 12:46:20 -0700 Subject: [PATCH 12/28] First pass at incorporating account ID --- .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 23 ++++++++++++++++++- .../AwsSdk/LambdaInvokeRequestHandler.cs | 6 ++--- 2 files changed, 25 insertions(+), 4 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index 3fa75fd2dd..84e2178c0f 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; +using System.Linq; using System.Threading.Tasks; using NewRelic.Agent.Api; using NewRelic.Agent.Extensions.Providers.Wrapper; @@ -12,6 +13,7 @@ namespace NewRelic.Providers.Wrapper.AwsSdk public class AwsSdkPipelineWrapper : IWrapper { public bool IsTransactionRequired => true; + private string _accountId = null; private const string WrapperName = "AwsSdkPipelineWrapper"; private static HashSet _unsupportedRequestTypes = new(); @@ -38,6 +40,24 @@ private string GetRegion(IAgent agent, dynamic requestContext) return ""; } + private string GetAccountId(IAgent agent) + { + if (_accountId != null) + { + return _accountId; + } + _accountId = agent.Configuration.AwsAccountId; + if (_accountId == null) + { + _accountId = ""; + } + else if ((_accountId.Length != 12) || _accountId.Any(c => (c < '0') || (c > '9'))) + { + agent.Logger.Warn("Supplied AWS Account Id appears to be invalid: {0}", _accountId); + } + return _accountId; + } + public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction) { // Get the IExecutionContext (the only parameter) @@ -67,6 +87,7 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins } dynamic request = requestContext.OriginalRequest; string requestType = request.GetType().FullName; + string accountId = agent.Configuration.AwsAccountId; if (requestType.StartsWith("Amazon.SQS")) { @@ -74,7 +95,7 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins } else if (requestType == "Amazon.Lambda.Model.InvokeRequest") { - return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, GetRegion(agent, requestContext)); + return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, GetRegion(agent, requestContext), accountId); } if (_unsupportedRequestTypes.Add(requestType)) // log once per unsupported request type diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index f2795dfd09..1a0afe865f 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -47,7 +47,7 @@ private static void SetRequestIdIfAvailable(IAgent agent, ITransaction transacti } } - public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, string region) + public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, string region, string accountId) { string functionName = request.FunctionName; string qualifier = request.Qualifier; @@ -65,12 +65,12 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC string key = $"{region}:{functionName}"; if (!_arnCache.TryGetValue(key, out arn)) { - arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, ""); + arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, accountId); _arnCache.TryAdd(key, arn); } } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); - segment.GetExperimentalApi().MakeLeaf(); + //segment.GetExperimentalApi().MakeLeaf(); transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); From 2d1e5c010663c411a368b38c62749dff372fd217 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 11 Nov 2024 14:29:27 -0800 Subject: [PATCH 13/28] First pass at ArnBuilder --- .../AwsSdk/ArnBuilder.cs | 52 +++++++++++++++++++ .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 37 +++++++------ .../AwsSdk/LambdaInvokeRequestHandler.cs | 9 ++-- 3 files changed, 75 insertions(+), 23 deletions(-) create mode 100644 src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs new file mode 100644 index 0000000000..1a64f68fdb --- /dev/null +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -0,0 +1,52 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +namespace NewRelic.Agent.Extensions.AwsSdk +{ + public class ArnBuilder + { + private string _partition; + public string Partition + { + private set + { + _partition = value; + } + get => _partition; + } + + private string _region; + public string Region + { + private set + { + _region = value; + } + get => _region; + } + private string _accountId; + public string AccountId + { + private set + { + _accountId = value; + } + get => _accountId; + } + public ArnBuilder(string partition, string region, string accountId) + { + _partition = partition; + _region = region; + _accountId = accountId; + } + + public string Build(string service, string resource) + { + if (string.IsNullOrEmpty(_partition) || string.IsNullOrEmpty(_region) || string.IsNullOrEmpty(_accountId)) + { + return null; + } + return "arn:" + _partition + ":" + service + ":" + _region + ":" + _accountId + ":" + resource; + } + } +} diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index 84e2178c0f..6108f1bbf9 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -6,6 +6,7 @@ using System.Linq; using System.Threading.Tasks; using NewRelic.Agent.Api; +using NewRelic.Agent.Extensions.AwsSdk; using NewRelic.Agent.Extensions.Providers.Wrapper; namespace NewRelic.Providers.Wrapper.AwsSdk @@ -13,7 +14,7 @@ namespace NewRelic.Providers.Wrapper.AwsSdk public class AwsSdkPipelineWrapper : IWrapper { public bool IsTransactionRequired => true; - private string _accountId = null; + private ArnBuilder _arnBuilder = null; private const string WrapperName = "AwsSdkPipelineWrapper"; private static HashSet _unsupportedRequestTypes = new(); @@ -23,39 +24,37 @@ public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) return new CanWrapResponse(WrapperName.Equals(methodInfo.RequestedWrapperName)); } - private string GetRegion(IAgent agent, dynamic requestContext) + private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) { + string + if (_arnBuilder != null) + { + return _arnBuilder; + } try { var clientconfig = requestContext.ClientConfig; var regionEndpoint = clientconfig.RegionEndpoint; var systemName = regionEndpoint.SystemName; - return systemName; + var partition = regionEndpoint.PartitionName; + _arnBuilder = new ArnBuilder(partition, systemName, GetAccountId(agent)); } catch (Exception e) { - agent.Logger.Debug(e, $"AwsSdkPipelineWrapper: Unable to get region from requestContext."); + agent.Logger.Debug(e, $"AwsSdkPipelineWrapper: Unable to get required ARN components from requestContext."); } - return ""; + return _arnBuilder; } private string GetAccountId(IAgent agent) { - if (_accountId != null) - { - return _accountId; - } - _accountId = agent.Configuration.AwsAccountId; - if (_accountId == null) - { - _accountId = ""; - } - else if ((_accountId.Length != 12) || _accountId.Any(c => (c < '0') || (c > '9'))) + string accountId = agent.Configuration.AwsAccountId; + if ((accountId.Length != 12) || accountId.Any(c => (c < '0') || (c > '9'))) { - agent.Logger.Warn("Supplied AWS Account Id appears to be invalid: {0}", _accountId); + agent.Logger.Warn("Supplied AWS Account Id appears to be invalid"); } - return _accountId; + return accountId; } public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction) @@ -87,7 +86,7 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins } dynamic request = requestContext.OriginalRequest; string requestType = request.GetType().FullName; - string accountId = agent.Configuration.AwsAccountId; + ArnBuilder builder = CreateArnBuilder(agent, requestContext); if (requestType.StartsWith("Amazon.SQS")) { @@ -95,7 +94,7 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins } else if (requestType == "Amazon.Lambda.Model.InvokeRequest") { - return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, GetRegion(agent, requestContext), accountId); + return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, builder); } if (_unsupportedRequestTypes.Add(requestType)) // log once per unsupported request type diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index 1a0afe865f..cff7d864c4 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -9,6 +9,7 @@ using NewRelic.Agent.Extensions.Providers.Wrapper; using NewRelic.Reflection; using NewRelic.Agent.Extensions.Helpers; +using NewRelic.Agent.Extensions.AwsSdk; namespace NewRelic.Providers.Wrapper.AwsSdk { @@ -47,7 +48,7 @@ private static void SetRequestIdIfAvailable(IAgent agent, ITransaction transacti } } - public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, string region, string accountId) + public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, ArnBuilder builder) { string functionName = request.FunctionName; string qualifier = request.Qualifier; @@ -62,10 +63,10 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC } else { - string key = $"{region}:{functionName}"; + string key = $"{builder.Region}:{functionName}"; if (!_arnCache.TryGetValue(key, out arn)) { - arn = AwsSdkHelpers.ConstructArn(agent, functionName, region, accountId); + arn = builder.Build("function", functionName); _arnCache.TryAdd(key, arn); } } @@ -74,7 +75,7 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); - transaction.AddCloudSdkAttribute("aws.region", region); + transaction.AddCloudSdkAttribute("aws.region", builder.Region); if (!string.IsNullOrEmpty(arn)) From eb4a73e154e38ee49bf33dba1cbc97bd54ffc8fa Mon Sep 17 00:00:00 2001 From: chynesNR Date: Thu, 14 Nov 2024 16:12:56 -0800 Subject: [PATCH 14/28] Another pass at generic ARN construction --- .../AwsSdk/ArnBuilder.cs | 133 +++++++++++++++++- .../Helpers/AwsSdk.cs | 126 ----------------- .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 8 +- .../AwsSdk/LambdaInvokeRequestHandler.cs | 15 +- .../Helpers/AwsSdkHelperTests.cs | 28 +++- 5 files changed, 168 insertions(+), 142 deletions(-) delete mode 100644 src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index 1a64f68fdb..0073b1a7c6 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -1,6 +1,10 @@ // Copyright 2020 New Relic, Inc. All rights reserved. // SPDX-License-Identifier: Apache-2.0 +using System.Linq; +using System.Text.RegularExpressions; +using NewRelic.Agent.Extensions.Collections; + namespace NewRelic.Agent.Extensions.AwsSdk { public class ArnBuilder @@ -40,13 +44,134 @@ public ArnBuilder(string partition, string region, string accountId) _accountId = accountId; } - public string Build(string service, string resource) + public string Build(string service, string resource) => ConstructArn(_partition, service, _region, _accountId, resource); + + // This is the full regex pattern for a Lambda ARN: + // (arn:(aws[a-zA-Z-]*)?:lambda:)?([a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}:)?(\d{12}:)?(function:)?([a-zA-Z0-9-_\.]+)(:(\$LATEST|[a-zA-Z0-9-_]+))? + + // If it's a full ARN, it has to start with 'arn:' + // A partial ARN can contain up to 5 segments separated by ':' + // 1. Region + // 2. Account ID + // 3. 'function' (fixed string) + // 4. Function name + // 5. Alias or version + // Only the function name is required, the rest are all optional. e.g. you could have region and function name and nothing else + public string BuildFromPartialLambdaArn(string invocationName) + { + if (invocationName.StartsWith("arn:")) + { + return invocationName; + } + var segments = invocationName.Split(':'); + string functionName = null; + string alias = null; + string fallback = null; + string region = null; + string accountId = null; + + // If there's only one string, assume it's the function name + if (segments.Length == 1) + { + functionName = segments[0]; + } + else + { + // All we should need is the function name, but if we find a region or account ID, we'll use it + // since it should be more accurate + foreach (var segment in segments) + { + // A string that looks like a region or account ID could also be the function name + // Assume it's the former, unless we never find a function name + if (LooksLikeARegion(segment)) + { + if (string.IsNullOrEmpty(region)) + { + region = segment; + } + else + { + fallback = segment; + } + continue; + } + else if (LooksLikeAnAccountId(segment)) + { + if (string.IsNullOrEmpty(accountId)) + { + accountId = segment; + } + else + { + fallback = segment; + } + continue; + } + else if (segment == "function") + { + continue; + } + else if (functionName == null) + { + functionName = segment; + } + else if (alias == null) + { + alias = segment; + } + else + { + return null; + } + } + } + + if (string.IsNullOrEmpty(functionName)) + { + if (!string.IsNullOrEmpty(fallback)) + { + functionName = fallback; + } + else + { + return null; + } + } + + accountId = !string.IsNullOrEmpty(accountId) ? accountId : _accountId; + if (string.IsNullOrEmpty(accountId)) + { + return null; + } + + region = !string.IsNullOrEmpty(region) ? region : _region; + if (string.IsNullOrEmpty(region)) + { + return null; + } + + + if (!string.IsNullOrEmpty(alias)) + { + return ConstructArn(_partition, "lambda", region, accountId, $"function:{functionName}:{alias}"); + + } + return ConstructArn(_partition, "lambda", region, accountId, $"function:{functionName}"); + } + + private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); + private static bool LooksLikeARegion(string text) => RegionRegex.IsMatch(text); + private static bool LooksLikeAnAccountId(string text) => (text.Length == 12) && text.All(c => c >= '0' && c <= '9'); + + private string ConstructArn(string partition, string service, string region, string accountId, string resource) { - if (string.IsNullOrEmpty(_partition) || string.IsNullOrEmpty(_region) || string.IsNullOrEmpty(_accountId)) + if (string.IsNullOrEmpty(partition) || string.IsNullOrEmpty(region) || string.IsNullOrEmpty(accountId) + || string.IsNullOrEmpty(service) || string.IsNullOrEmpty(resource)) { return null; } - return "arn:" + _partition + ":" + service + ":" + _region + ":" + _accountId + ":" + resource; - } + return "arn:" + partition + ":" + service + ":" + region + ":" + accountId + ":" + resource; + } + } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs deleted file mode 100644 index 036bd02f09..0000000000 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Helpers/AwsSdk.cs +++ /dev/null @@ -1,126 +0,0 @@ -// Copyright 2020 New Relic, Inc. All rights reserved. -// SPDX-License-Identifier: Apache-2.0 - -using System.Collections.Generic; -using System.Linq; -using System.Text.RegularExpressions; -using NewRelic.Agent.Api; -using NewRelic.Agent.Extensions.Collections; - -namespace NewRelic.Agent.Extensions.Helpers -{ - public static class AwsSdkHelpers - { - private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); - private static bool LooksLikeARegion(string text) => RegionRegex.IsMatch(text); - private static bool LooksLikeAnAccountId(string text) => (text.Length == 12) && text.All(c => c >= '0' && c <= '9'); - // Only log ARNs we can't parse once - private static ConcurrentHashSet BadInvocations = new ConcurrentHashSet(); - private const int MAX_BAD_INVOCATIONS = 25; - - private static void ReportBadInvocation(IAgent agent, string invocationName) - { - if ((agent?.Logger.IsDebugEnabled == true) && (BadInvocations.Count < MAX_BAD_INVOCATIONS) && BadInvocations.TryAdd(invocationName)) - { - agent.Logger.Debug($"Unable to parse function name '{invocationName}'"); - } - } - - // This is the full regex pattern for an ARN: - // (arn:(aws[a-zA-Z-]*)?:lambda:)?([a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}:)?(\d{12}:)?(function:)?([a-zA-Z0-9-_\.]+)(:(\$LATEST|[a-zA-Z0-9-_]+))? - - // If it's a full ARN, it has to start with 'arn:' - // A partial ARN can contain up to 5 segments separated by ':' - // 1. Region - // 2. Account ID - // 3. 'function' (fixed string) - // 4. Function name - // 5. Alias or version - // Only the function name is required, the rest are all optional. e.g. you could have region and function name and nothing else - public static string ConstructArn(IAgent agent, string invocationName, string region, string accountId) - { - if (invocationName.StartsWith("arn:")) - { - return invocationName; - } - var segments = invocationName.Split(':'); - string functionName = null; - string alias = null; - string fallback = null; - - foreach (var segment in segments) - { - if (LooksLikeARegion(segment)) - { - if (string.IsNullOrEmpty(region)) - { - region = segment; - } - else - { - fallback = segment; - } - continue; - } - else if (LooksLikeAnAccountId(segment)) - { - if (string.IsNullOrEmpty(accountId)) - { - accountId = segment; - } - else - { - fallback = segment; - } - continue; - } - else if (segment == "function") - { - continue; - } - else if (functionName == null) - { - functionName = segment; - } - else if (alias == null) - { - alias = segment; - } - else - { - ReportBadInvocation(agent, invocationName); - return null; - } - } - - if (string.IsNullOrEmpty(functionName)) - { - if (!string.IsNullOrEmpty(fallback)) - { - functionName = fallback; - } - else - { - ReportBadInvocation(agent, invocationName); - return null; - } - } - - if (string.IsNullOrEmpty(accountId)) - { - ReportBadInvocation(agent, invocationName); - return null; - } - if (string.IsNullOrEmpty(region)) - { - ReportBadInvocation(agent, invocationName); - return null; - } - if (!string.IsNullOrEmpty(alias)) - { - return $"arn:aws:lambda:{region}:{accountId}:function:{functionName}:{alias}"; - } - return $"arn:aws:lambda:{region}:{accountId}:function:{functionName}"; - } - } -} diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index 6108f1bbf9..f5f3e11327 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -26,7 +26,6 @@ public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) { - string if (_arnBuilder != null) { return _arnBuilder; @@ -50,9 +49,12 @@ private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) private string GetAccountId(IAgent agent) { string accountId = agent.Configuration.AwsAccountId; - if ((accountId.Length != 12) || accountId.Any(c => (c < '0') || (c > '9'))) + if (accountId != null) { - agent.Logger.Warn("Supplied AWS Account Id appears to be invalid"); + if ((accountId.Length != 12) || accountId.Any(c => (c < '0') || (c > '9'))) + { + agent.Logger.Warn("Supplied AWS Account Id appears to be invalid"); + } } return accountId; } diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index cff7d864c4..ee0c97d687 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -18,6 +18,7 @@ internal static class LambdaInvokeRequestHandler private static Func _getResultFromGenericTask; private static ConcurrentDictionary _arnCache = new ConcurrentDictionary(); private static bool _reportMissingRequestId = true; + private static bool _reportBadInvocationName = true; private static object GetTaskResult(object task) { @@ -63,15 +64,14 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC } else { - string key = $"{builder.Region}:{functionName}"; - if (!_arnCache.TryGetValue(key, out arn)) + if (!_arnCache.TryGetValue(functionName, out arn)) { - arn = builder.Build("function", functionName); - _arnCache.TryAdd(key, arn); + arn = builder.BuildFromPartialLambdaArn(functionName); + _arnCache.TryAdd(functionName, arn); } } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); - //segment.GetExperimentalApi().MakeLeaf(); + //segment.GetExperimentalApi().MakeLeaf(); // TODO: Leaf-or-not decision has not yet been finalized transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); @@ -82,6 +82,11 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC { transaction.AddCloudSdkAttribute("cloud.resource_id", arn); } + else if (_reportBadInvocationName) + { + agent.Logger.Debug($"Unable to parse lambda invocation named '{functionName}''"); + _reportBadInvocationName = false; + } if (isAsync) { diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index 1382e0d093..b95edf7ce9 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -2,11 +2,11 @@ // SPDX-License-Identifier: Apache-2.0 using NUnit.Framework; -using NewRelic.Agent.Extensions.Helpers; +using NewRelic.Agent.Extensions.AwsSdk; namespace Agent.Extensions.Tests.Helpers { - public class AwsSdkHelperTests + public class AwsSdkArnBuilderTests { [Test] [TestCase("myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] @@ -41,10 +41,30 @@ public class AwsSdkHelperTests [TestCase("123456789012:444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] [TestCase("444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] [TestCase("us-west-2", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:us-west-2")] - public void ConstructArn(string name, string region, string accountId, string arn) + public void ConstructLambdaArn(string name, string region, string accountId, string arn) { - var constructedArn = AwsSdkHelpers.ConstructArn(null, name, region, accountId); + var arnBuilder = new ArnBuilder("aws", region, accountId); + var constructedArn = arnBuilder.BuildFromPartialLambdaArn(name); Assert.That(constructedArn, Is.EqualTo(arn), "Did not get expected ARN"); } + + [Test] + [TestCase("aws", "s3", "us-west-2", "123456789012", "bucket_name", "arn:aws:s3:us-west-2:123456789012:bucket_name")] + [TestCase("aws", "dynamodb", "us-east-1", "987654321098", "table_name", "arn:aws:dynamodb:us-east-1:987654321098:table_name")] + [TestCase("aws", "dynamodb", "us-east-1", "987654321098", "table/tabletName", "arn:aws:dynamodb:us-east-1:987654321098:table/tabletName")] + [TestCase("aws", "ec2", "eu-west-1", "111122223333", "instance_id", "arn:aws:ec2:eu-west-1:111122223333:instance_id")] + [TestCase("aws", "sqs", "ap-southeast-1", "444455556666", "queue_name", "arn:aws:sqs:ap-southeast-1:444455556666:queue_name")] + [TestCase("aws", "sns", "ca-central-1", "777788889999", "topic_name", "arn:aws:sns:ca-central-1:777788889999:topic_name")] + [TestCase("aws-cn", "lambda", "cn-north-1", "222233334444", "function_name", "arn:aws-cn:lambda:cn-north-1:222233334444:function_name")] + [TestCase("aws-us-gov", "iam", "us-gov-west-1", "555566667777", "role_name", "arn:aws-us-gov:iam:us-gov-west-1:555566667777:role_name")] + [TestCase("aws", "rds", "sa-east-1", "888899990000", "db_instance", "arn:aws:rds:sa-east-1:888899990000:db_instance")] + [TestCase("aws", "s3", "", "123456789012", "bucket_name", null)] + [TestCase("aws", "s3", "us-west-2", "", "bucket_name", null)] + public void ConstructGenericArn(string partition, string service, string region, string accountId, string resource, string expectedArn) + { + var arnBuilder = new ArnBuilder(partition, region, accountId); + var constructedArn = arnBuilder.Build(service, resource); + Assert.That(constructedArn, Is.EqualTo(expectedArn), "Did not get expected ARN"); + } } } From 35dad1c58b295ac01451db1c99127f01b4eef77e Mon Sep 17 00:00:00 2001 From: chynesNR Date: Fri, 15 Nov 2024 12:56:22 -0800 Subject: [PATCH 15/28] Increasing code coverage --- .../AwsSdk/ArnBuilder.cs | 46 +++++-------------- .../Helpers/AwsSdkHelperTests.cs | 1 + 2 files changed, 12 insertions(+), 35 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index 0073b1a7c6..751d7f1220 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -9,42 +9,18 @@ namespace NewRelic.Agent.Extensions.AwsSdk { public class ArnBuilder { - private string _partition; - public string Partition - { - private set - { - _partition = value; - } - get => _partition; - } + public readonly string Partition; + public readonly string Region; + public readonly string AccountId; - private string _region; - public string Region - { - private set - { - _region = value; - } - get => _region; - } - private string _accountId; - public string AccountId - { - private set - { - _accountId = value; - } - get => _accountId; - } public ArnBuilder(string partition, string region, string accountId) { - _partition = partition; - _region = region; - _accountId = accountId; + Partition = partition; + Region = region; + AccountId = accountId; } - public string Build(string service, string resource) => ConstructArn(_partition, service, _region, _accountId, resource); + public string Build(string service, string resource) => ConstructArn(Partition, service, Region, AccountId, resource); // This is the full regex pattern for a Lambda ARN: // (arn:(aws[a-zA-Z-]*)?:lambda:)?([a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}:)?(\d{12}:)?(function:)?([a-zA-Z0-9-_\.]+)(:(\$LATEST|[a-zA-Z0-9-_]+))? @@ -138,13 +114,13 @@ public string BuildFromPartialLambdaArn(string invocationName) } } - accountId = !string.IsNullOrEmpty(accountId) ? accountId : _accountId; + accountId = !string.IsNullOrEmpty(accountId) ? accountId : AccountId; if (string.IsNullOrEmpty(accountId)) { return null; } - region = !string.IsNullOrEmpty(region) ? region : _region; + region = !string.IsNullOrEmpty(region) ? region : Region; if (string.IsNullOrEmpty(region)) { return null; @@ -153,10 +129,10 @@ public string BuildFromPartialLambdaArn(string invocationName) if (!string.IsNullOrEmpty(alias)) { - return ConstructArn(_partition, "lambda", region, accountId, $"function:{functionName}:{alias}"); + return ConstructArn(Partition, "lambda", region, accountId, $"function:{functionName}:{alias}"); } - return ConstructArn(_partition, "lambda", region, accountId, $"function:{functionName}"); + return ConstructArn(Partition, "lambda", region, accountId, $"function:{functionName}"); } private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index b95edf7ce9..4995e9a8a6 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -37,6 +37,7 @@ public class AwsSdkArnBuilderTests [TestCase("123456789012:my-function:myalias:extra", "us-west-2", "123456789012", null)] [TestCase("123456789012:my-function:myalias:extra:lots:of:extra:way:too:many", "us-west-2", "123456789012", null)] [TestCase("eu-west-1:us-west-2", "eu-west-1", "123456789012", "arn:aws:lambda:eu-west-1:123456789012:function:us-west-2")] + [TestCase("", "us-west-2", "123456789012", null)] // Edge cases: functions that look like account IDs or region names [TestCase("123456789012:444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] [TestCase("444455556666", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:444455556666")] From d81184990fdc9196de71ad47f8b089e7665b19e8 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Fri, 15 Nov 2024 16:15:34 -0800 Subject: [PATCH 16/28] Add supportability metric --- .../Agent/Core/AgentHealth/AgentHealthReporter.cs | 9 ++++++++- src/Agent/NewRelic/Agent/Core/Metrics/MetricNames.cs | 1 + .../AgentHealth/AgentHealthReporterTests.cs | 8 ++++++++ 3 files changed, 17 insertions(+), 1 deletion(-) diff --git a/src/Agent/NewRelic/Agent/Core/AgentHealth/AgentHealthReporter.cs b/src/Agent/NewRelic/Agent/Core/AgentHealth/AgentHealthReporter.cs index 53fce6b123..85f4f4edaf 100644 --- a/src/Agent/NewRelic/Agent/Core/AgentHealth/AgentHealthReporter.cs +++ b/src/Agent/NewRelic/Agent/Core/AgentHealth/AgentHealthReporter.cs @@ -684,6 +684,7 @@ private void CollectOneTimeMetrics() ReportIfLoggingDisabled(); ReportIfInstrumentationIsDisabled(); ReportIfGCSamplerV2IsEnabled(); + ReportIfAwsAccountIdProvided(); } public void CollectMetrics() @@ -846,8 +847,14 @@ private void ReportIfGCSamplerV2IsEnabled() { ReportSupportabilityCountMetric(MetricNames.SupportabilityGCSamplerV2Enabled); } - } + private void ReportIfAwsAccountIdProvided() + { + if (!string.IsNullOrEmpty(_configuration.AwsAccountId)) + { + ReportSupportabilityCountMetric(MetricNames.SupportabilityAwsAccountIdProvided); + } + } } } diff --git a/src/Agent/NewRelic/Agent/Core/Metrics/MetricNames.cs b/src/Agent/NewRelic/Agent/Core/Metrics/MetricNames.cs index 2ea94148d3..b28ca695ec 100644 --- a/src/Agent/NewRelic/Agent/Core/Metrics/MetricNames.cs +++ b/src/Agent/NewRelic/Agent/Core/Metrics/MetricNames.cs @@ -838,6 +838,7 @@ public static string GetSupportabilityInstallType(string installType) public const string SupportabilityIgnoredInstrumentation = SupportabilityDotnetPs + "IgnoredInstrumentation"; public const string SupportabilityGCSamplerV2Enabled = SupportabilityDotnetPs + "GCSamplerV2/Enabled"; + public const string SupportabilityAwsAccountIdProvided = SupportabilityDotnetPs + "AwsAccountId/Config"; #endregion Supportability diff --git a/tests/Agent/UnitTests/Core.UnitTest/AgentHealth/AgentHealthReporterTests.cs b/tests/Agent/UnitTests/Core.UnitTest/AgentHealth/AgentHealthReporterTests.cs index b25163cee6..76acfea007 100644 --- a/tests/Agent/UnitTests/Core.UnitTest/AgentHealth/AgentHealthReporterTests.cs +++ b/tests/Agent/UnitTests/Core.UnitTest/AgentHealth/AgentHealthReporterTests.cs @@ -61,6 +61,7 @@ private IConfiguration GetDefaultConfiguration() Mock.Arrange(() => configuration.LoggingEnabled).Returns(() => _enableLogging); Mock.Arrange(() => configuration.IgnoredInstrumentation).Returns(() => _ignoredInstrumentation); Mock.Arrange(() => configuration.GCSamplerV2Enabled).Returns(true); + Mock.Arrange(() => configuration.AwsAccountId).Returns("123456789012"); return configuration; } @@ -531,5 +532,12 @@ public void GCSamplerV2EnabledSupportabiliityMetricPresent() _agentHealthReporter.CollectMetrics(); Assert.That(_publishedMetrics.Any(x => x.MetricNameModel.Name == "Supportability/Dotnet/GCSamplerV2/Enabled"), Is.True); } + + [Test] + public void AwsAccountIdSupportabiliityMetricPresent() + { + _agentHealthReporter.CollectMetrics(); + Assert.That(_publishedMetrics.Any(x => x.MetricNameModel.Name == "Supportability/Dotnet/AwsAccountId/Config"), Is.True); + } } } From d47b5a49c802dfa5b6c29ea9851ff73036fab08f Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 25 Nov 2024 12:37:21 -0800 Subject: [PATCH 17/28] Some cleanup and better logging --- .../AwsSdk/ArnBuilder.cs | 6 ++-- .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 30 ++++++++++++------- .../AwsSdk/LambdaInvokeRequestHandler.cs | 7 +++-- 3 files changed, 26 insertions(+), 17 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index 751d7f1220..08301de978 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -46,7 +46,7 @@ public string BuildFromPartialLambdaArn(string invocationName) string region = null; string accountId = null; - // If there's only one string, assume it's the function name + // If there's only one segment, assume it's the function name if (segments.Length == 1) { functionName = segments[0]; @@ -126,11 +126,9 @@ public string BuildFromPartialLambdaArn(string invocationName) return null; } - if (!string.IsNullOrEmpty(alias)) { - return ConstructArn(Partition, "lambda", region, accountId, $"function:{functionName}:{alias}"); - + functionName += $":{alias}"; } return ConstructArn(Partition, "lambda", region, accountId, $"function:{functionName}"); } diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index f5f3e11327..4e4c3c62b8 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -14,10 +14,11 @@ namespace NewRelic.Providers.Wrapper.AwsSdk public class AwsSdkPipelineWrapper : IWrapper { public bool IsTransactionRequired => true; - private ArnBuilder _arnBuilder = null; private const string WrapperName = "AwsSdkPipelineWrapper"; private static HashSet _unsupportedRequestTypes = new(); + private static bool _reportBadAccountId = true; + private static bool _reportBadArnBuilder = false; public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) { @@ -26,24 +27,27 @@ public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) { - if (_arnBuilder != null) - { - return _arnBuilder; - } + string partition = ""; + string systemName = ""; + string accountId = ""; try { + accountId = GetAccountId(agent); var clientconfig = requestContext.ClientConfig; var regionEndpoint = clientconfig.RegionEndpoint; - var systemName = regionEndpoint.SystemName; - var partition = regionEndpoint.PartitionName; - _arnBuilder = new ArnBuilder(partition, systemName, GetAccountId(agent)); + systemName = regionEndpoint.SystemName; + partition = regionEndpoint.PartitionName; } catch (Exception e) { - agent.Logger.Debug(e, $"AwsSdkPipelineWrapper: Unable to get required ARN components from requestContext."); + if (_reportBadArnBuilder) + { + agent.Logger.Debug(e, $"AwsSdkPipelineWrapper: Unable to get required ARN components from requestContext."); + _reportBadArnBuilder = false; + } } - return _arnBuilder; + return new ArnBuilder(partition, systemName, accountId); ; } private string GetAccountId(IAgent agent) @@ -53,7 +57,11 @@ private string GetAccountId(IAgent agent) { if ((accountId.Length != 12) || accountId.Any(c => (c < '0') || (c > '9'))) { - agent.Logger.Warn("Supplied AWS Account Id appears to be invalid"); + if (_reportBadAccountId) + { + agent.Logger.Warn("Supplied AWS Account ID appears to be invalid"); + _reportBadAccountId = false; + } } } return accountId; diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index ee0c97d687..e52b526e48 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -19,6 +19,7 @@ internal static class LambdaInvokeRequestHandler private static ConcurrentDictionary _arnCache = new ConcurrentDictionary(); private static bool _reportMissingRequestId = true; private static bool _reportBadInvocationName = true; + private const int MAX_CACHE_SIZE = 25; // Shouldn't ever get this big, but just in case private static object GetTaskResult(object task) { @@ -67,11 +68,13 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC if (!_arnCache.TryGetValue(functionName, out arn)) { arn = builder.BuildFromPartialLambdaArn(functionName); - _arnCache.TryAdd(functionName, arn); + if (_arnCache.Count < MAX_CACHE_SIZE) + { + _arnCache.TryAdd(functionName, arn); + } } } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); - //segment.GetExperimentalApi().MakeLeaf(); // TODO: Leaf-or-not decision has not yet been finalized transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); From f229fc1e78d0600a4534afeadf4e577fab116bae Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 25 Nov 2024 14:36:20 -0800 Subject: [PATCH 18/28] More logging improvements --- .../AwsSdk/ArnBuilder.cs | 15 ++++++++++++--- .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 6 +++--- .../Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs | 2 +- .../Helpers/AwsSdkHelperTests.cs | 12 ++++++++++++ 4 files changed, 28 insertions(+), 7 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index 08301de978..bb0830c1c5 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -15,9 +15,9 @@ public class ArnBuilder public ArnBuilder(string partition, string region, string accountId) { - Partition = partition; - Region = region; - AccountId = accountId; + Partition = partition ?? ""; + Region = region ?? ""; + AccountId = accountId ?? ""; } public string Build(string service, string resource) => ConstructArn(Partition, service, Region, AccountId, resource); @@ -133,6 +133,15 @@ public string BuildFromPartialLambdaArn(string invocationName) return ConstructArn(Partition, "lambda", region, accountId, $"function:{functionName}"); } + public override string ToString() + { + string partition = string.IsNullOrEmpty(Partition) ? "[Missing]" : Partition; + string region = string.IsNullOrEmpty(Region) ? "[Missing]" : Region; + string accountId = string.IsNullOrEmpty(AccountId) ? "[Missing]" : "[Present]"; + + return $"Partition: {partition}, Region: {region}, AccountId: {accountId}"; + } + private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); private static bool LooksLikeARegion(string text) => RegionRegex.IsMatch(text); private static bool LooksLikeAnAccountId(string text) => (text.Length == 12) && text.All(c => c >= '0' && c <= '9'); diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index 6d100ad09f..0d3d2c80aa 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -28,9 +28,9 @@ public CanWrapResponse CanWrap(InstrumentedMethodInfo methodInfo) private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) { - string partition = ""; - string systemName = ""; - string accountId = ""; + string partition = null; + string systemName = null; + string accountId = null; try { accountId = GetAccountId(agent); diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs index e52b526e48..cb41437d2c 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs @@ -87,7 +87,7 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC } else if (_reportBadInvocationName) { - agent.Logger.Debug($"Unable to parse lambda invocation named '{functionName}''"); + agent.Logger.Debug($"Unable to resolve Lambda invocation named '{functionName}' [{builder.ToString()}]"); _reportBadInvocationName = false; } diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index 4995e9a8a6..c342e4fbf7 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -67,5 +67,17 @@ public void ConstructGenericArn(string partition, string service, string region, var constructedArn = arnBuilder.Build(service, resource); Assert.That(constructedArn, Is.EqualTo(expectedArn), "Did not get expected ARN"); } + + [Test] + [TestCase("aws", "us-west-2", "123456789012", "Partition: aws, Region: us-west-2, AccountId: [Present]")] + [TestCase("aws", "", "123456789012", "Partition: aws, Region: [Missing], AccountId: [Present]")] + [TestCase("aws", "us-west-2", "", "Partition: aws, Region: us-west-2, AccountId: [Missing]")] + [TestCase("aws", "us-west-2", null, "Partition: aws, Region: us-west-2, AccountId: [Missing]")] + [TestCase("", "", "", "Partition: [Missing], Region: [Missing], AccountId: [Missing]")] + public void ArnBuilderToString(string partition, string region, string accountId, string expected) + { + var arnBuilder = new ArnBuilder(partition, region, accountId); + Assert.That(arnBuilder.ToString(), Is.EqualTo(expected), "Did not get expected string"); + } } } From 8af3e951d8bfd22c48dc32084c80a692b8a6af10 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 25 Nov 2024 15:10:56 -0800 Subject: [PATCH 19/28] Fix bad merge --- .../Common/MFALatestPackages/MFALatestPackages.csproj | 3 --- 1 file changed, 3 deletions(-) diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj index 2bbec636b3..624d00e705 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj @@ -12,9 +12,6 @@ - - - From 272ace422678666c035465ed425af31609d9aec5 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Mon, 25 Nov 2024 16:11:38 -0800 Subject: [PATCH 20/28] Fixing more merge errors --- .../Common/MFALatestPackages/MFALatestPackages.csproj | 4 ++-- .../MultiFunctionApplicationHelpers.csproj | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj index 624d00e705..03efc80501 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MFALatestPackages/MFALatestPackages.csproj @@ -10,7 +10,7 @@ - + @@ -34,7 +34,7 @@ - + diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj index 91c80022f4..8b02a93245 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/MultiFunctionApplicationHelpers.csproj @@ -274,7 +274,7 @@ - + From 0b91f6ad8d5ad370de9bf9bf06c181664395a4cd Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 26 Nov 2024 09:44:34 -0800 Subject: [PATCH 21/28] Trying to address CodeQL and CodeCov issues --- .../AwsSdk/ArnBuilder.cs | 4 +-- .../Transactions/TransactionTests.cs | 31 +++++++++++++++++++ .../Helpers/AwsSdkHelperTests.cs | 1 + 3 files changed, 34 insertions(+), 2 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index bb0830c1c5..b067b5b06b 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -137,9 +137,9 @@ public override string ToString() { string partition = string.IsNullOrEmpty(Partition) ? "[Missing]" : Partition; string region = string.IsNullOrEmpty(Region) ? "[Missing]" : Region; - string accountId = string.IsNullOrEmpty(AccountId) ? "[Missing]" : "[Present]"; + string idPresent = string.IsNullOrEmpty(AccountId) ? "[Missing]" : "[Present]"; - return $"Partition: {partition}, Region: {region}, AccountId: {accountId}"; + return $"Partition: {partition}, Region: {region}, AccountId: {idPresent}"; } private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); diff --git a/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs b/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs index 929a38be34..f14db94c45 100644 --- a/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs +++ b/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs @@ -591,4 +591,35 @@ public void HasHttpResponseStatusCode_ReturnsFalse_WhenStatusCodeIsNotSet() // Assert Assert.That(result, Is.False); } + public void AddCloudSdkAttribute_SetAttributeInTransactionMetadata() + { + // Arrange + var key = "TestAttribute"; + var value = "TestValue"; + + // Act + _transaction.AddCloudSdkAttribute(key, value); + + // Assert + var allAttributeValuesDic = _transaction.TransactionMetadata.UserAndRequestAttributes.GetAllAttributeValuesDic(); + + var attributeValue = allAttributeValuesDic[key]; + Assert.That(attributeValue, Is.EqualTo(value)); + } + + [TestCase(" ")] + [TestCase("")] + [TestCase(null)] + public void AddCloudSdkAttribute_DoesNotSetAttribute_WhenKeyIsBad(string key) + { + // Arrange + var value = "TestValue"; + + // Act + _transaction.AddCloudSdkAttribute(key, value); + + // Assert + Assert.That(_transaction.TransactionMetadata.UserAndRequestAttributes.Count, Is.EqualTo(0)); + } + } diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index c342e4fbf7..126be85b63 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -74,6 +74,7 @@ public void ConstructGenericArn(string partition, string service, string region, [TestCase("aws", "us-west-2", "", "Partition: aws, Region: us-west-2, AccountId: [Missing]")] [TestCase("aws", "us-west-2", null, "Partition: aws, Region: us-west-2, AccountId: [Missing]")] [TestCase("", "", "", "Partition: [Missing], Region: [Missing], AccountId: [Missing]")] + [TestCase(null, null, null, "Partition: [Missing], Region: [Missing], AccountId: [Missing]")] public void ArnBuilderToString(string partition, string region, string accountId, string expected) { var arnBuilder = new ArnBuilder(partition, region, accountId); From 17f810f053c7203fd993a9df6cd9e9f923e8e29a Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 26 Nov 2024 13:38:56 -0800 Subject: [PATCH 22/28] Only pass on expected error --- .../NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs index 574707aa0f..371741d585 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs @@ -6,6 +6,7 @@ using System.Security.AccessControl; using System.Threading.Tasks; using Amazon; +using Amazon.Lambda.Model; using NewRelic.Agent.IntegrationTests.Shared.ReflectionHelpers; using NewRelic.Api.Agent; @@ -32,7 +33,7 @@ public void InvokeLambdaSync(string function, string payload) { var response = client.Invoke(request); } - catch + catch (ResourceNotFoundException) { } #else @@ -60,7 +61,7 @@ public async Task InvokeLambdaAsync(string function, string payload) string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); return returnValue; } - catch + catch (ResourceNotFoundException) { } return null; @@ -87,7 +88,7 @@ public async Task InvokeLambdaAsyncWithQualifier(string function, string string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); return returnValue; } - catch + catch (ResourceNotFoundException) { } return null; From 641eafc0f615ebfabc1eebb3f45be47978ae1a28 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 3 Dec 2024 11:03:30 -0800 Subject: [PATCH 23/28] Revert change (expected error is not consistent) --- .../NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs index 371741d585..269e812fcb 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/AwsSdk/InvokeLambdaExerciser.cs @@ -33,7 +33,7 @@ public void InvokeLambdaSync(string function, string payload) { var response = client.Invoke(request); } - catch (ResourceNotFoundException) + catch { } #else @@ -61,7 +61,7 @@ public async Task InvokeLambdaAsync(string function, string payload) string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); return returnValue; } - catch (ResourceNotFoundException) + catch { } return null; @@ -88,7 +88,7 @@ public async Task InvokeLambdaAsyncWithQualifier(string function, string string returnValue = System.Text.Encoding.UTF8.GetString(stream.ToArray()); return returnValue; } - catch (ResourceNotFoundException) + catch { } return null; From 55cc8a457f3715e34b4879ecf55c00f3017ec19f Mon Sep 17 00:00:00 2001 From: Marty T <120425148+tippmar-nr@users.noreply.github.com> Date: Fri, 6 Dec 2024 15:33:47 -0600 Subject: [PATCH 24/28] Feature work/aws account id parsing (#2920) --- .../AwsSdk/ArnBuilder.cs | 12 +-- .../AwsSdk/AwsAccountIdDecoder.cs | 70 ++++++++++++ .../AwsSdk/AmazonServiceClientWrapper.cs | 52 +++++++++ .../Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs | 32 +++--- .../Wrapper/AwsSdk/Instrumentation.xml | 9 ++ .../DynamoDbRequestHandler.cs | 6 +- .../LambdaInvokeRequestHandler.cs | 6 +- .../SQSRequestHandler.cs | 6 +- src/Agent/NewRelic/Home/Home.csproj | 4 +- .../RemoteApplicationFixture.cs | 1 + .../AwsSdk/InvokeLambdaTests.cs | 14 ++- ...ckConfiguration.cs => AwsConfiguration.cs} | 2 +- .../NetStandardLibraries/LLM/BedrockModels.cs | 2 +- .../Helpers/AwsAccountIdDecoderTests.cs | 102 ++++++++++++++++++ .../Helpers/AwsSdkHelperTests.cs | 10 +- 15 files changed, 281 insertions(+), 47 deletions(-) create mode 100644 src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/AwsAccountIdDecoder.cs create mode 100644 src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs rename src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/{ => RequestHandlers}/DynamoDbRequestHandler.cs (87%) rename src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/{ => RequestHandlers}/LambdaInvokeRequestHandler.cs (95%) rename src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/{ => RequestHandlers}/SQSRequestHandler.cs (95%) rename tests/Agent/IntegrationTests/Shared/{AwsBedrockConfiguration.cs => AwsConfiguration.cs} (96%) create mode 100644 tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsAccountIdDecoderTests.cs diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index b067b5b06b..4030057429 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -3,7 +3,6 @@ using System.Linq; using System.Text.RegularExpressions; -using NewRelic.Agent.Extensions.Collections; namespace NewRelic.Agent.Extensions.AwsSdk { @@ -15,8 +14,8 @@ public class ArnBuilder public ArnBuilder(string partition, string region, string accountId) { - Partition = partition ?? ""; - Region = region ?? ""; + Partition = string.IsNullOrEmpty(partition) ? "aws" : partition; + Region = string.IsNullOrEmpty(region) ? "(unknown)" : region; AccountId = accountId ?? ""; } @@ -135,11 +134,9 @@ public string BuildFromPartialLambdaArn(string invocationName) public override string ToString() { - string partition = string.IsNullOrEmpty(Partition) ? "[Missing]" : Partition; - string region = string.IsNullOrEmpty(Region) ? "[Missing]" : Region; string idPresent = string.IsNullOrEmpty(AccountId) ? "[Missing]" : "[Present]"; - - return $"Partition: {partition}, Region: {region}, AccountId: {idPresent}"; + + return $"Partition: {Partition}, Region: {Region}, AccountId: {idPresent}"; } private static Regex RegionRegex = new Regex(@"^[a-z]{2}((-gov)|(-iso([a-z]?)))?-[a-z]+-\d{1}$", RegexOptions.Compiled); @@ -155,6 +152,5 @@ private string ConstructArn(string partition, string service, string region, str } return "arn:" + partition + ":" + service + ":" + region + ":" + accountId + ":" + resource; } - } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/AwsAccountIdDecoder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/AwsAccountIdDecoder.cs new file mode 100644 index 0000000000..a6e3dea2fb --- /dev/null +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/AwsAccountIdDecoder.cs @@ -0,0 +1,70 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System; + +namespace NewRelic.Agent.Extensions.AwsSdk +{ + public static class AwsAccountIdDecoder + { + // magic number + private const long Mask = 0x7FFFFFFFFF80; + + public static string GetAccountId(string awsAccessKeyId) + { + if (string.IsNullOrEmpty(awsAccessKeyId)) + { + throw new ArgumentNullException(nameof(awsAccessKeyId), "AWS Access Key ID cannot be null or empty."); + } + + if (awsAccessKeyId.Length < 14) + { + throw new ArgumentOutOfRangeException(nameof(awsAccessKeyId), "AWS Access Key ID must be at least 14 characters long."); + } + + string accessKeyWithoutPrefix = awsAccessKeyId.Substring(4).ToLowerInvariant(); + long encodedAccount = Base32Decode(accessKeyWithoutPrefix); + + return ((encodedAccount & Mask) >> 7).ToString(); + } + + /// + /// Performs a Base-32 decode of the specified input string. + /// Allowed character range is a-z and 2-7. 'a' being 0 and '7' is 31. + /// + /// public to allow for unit testing + /// + /// The string to be decoded. Must be at least 10 characters. + /// A long containing first 6 bytes of the base 32 decoded data. + /// If src has less than 10 characters. + /// If src contains invalid characters for Base-32 + public static long Base32Decode(string src) + { + if (string.IsNullOrEmpty(src)) + { + throw new ArgumentNullException(nameof(src), "The input string cannot be null or empty."); + } + + if (src.Length < 10) + { + throw new ArgumentException("The input string must be at least 10 characters long.", nameof(src)); + } + + long baseValue = 0; + for (int i = 0; i < 10; i++) + { + baseValue <<= 5; + char c = src[i]; + baseValue += c switch + { + >= 'a' and <= 'z' => c - 'a', + >= '2' and <= '7' => c - '2' + 26, + _ => throw new ArgumentOutOfRangeException(nameof(src), + "The input string must contain only characters in the range a-z and 2-7.") + }; + } + + return baseValue >> 2; + } + } +} diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs new file mode 100644 index 0000000000..615a9ea8e5 --- /dev/null +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs @@ -0,0 +1,52 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System; +using NewRelic.Agent.Api; +using NewRelic.Agent.Extensions.AwsSdk; +using NewRelic.Agent.Extensions.Providers.Wrapper; + +namespace NewRelic.Providers.Wrapper.AwsSdk +{ + public class AmazonServiceClientWrapper : IWrapper + { + /// + /// The AWS account id. + /// Parsed from the access key in the credentials of the client - or fall back to the configuration value if parsing fails. + /// Assumes only a single account id is used in the application. + /// + public static string AwsAccountId { get; private set; } + + public bool IsTransactionRequired => false; + + public CanWrapResponse CanWrap(InstrumentedMethodInfo instrumentedMethodInfo) + { + return new CanWrapResponse(instrumentedMethodInfo.RequestedWrapperName == nameof(AmazonServiceClientWrapper)); + } + + public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction) + { + if (AwsAccountId != null) + return Delegates.NoOp; + + try + { + // get the AWSCredentials parameter + dynamic awsCredentials = instrumentedMethodCall.MethodCall.MethodArguments[0]; + + dynamic immutableCredentials = awsCredentials.GetCredentials(); + string accessKey = immutableCredentials.AccessKey; + + // convert the access key to an account id + AwsAccountId = AwsAccountIdDecoder.GetAccountId(accessKey); + } + catch (Exception e) + { + agent.Logger.Info($"Unable to parse AWS Account ID from AccessKey. Using AccountId from configuration instead. Exception: {e.Message}"); + AwsAccountId = agent.Configuration.AwsAccountId; + } + + return Delegates.NoOp; + } + } +} diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs index 0d3d2c80aa..d8e7f02642 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AwsSdkPipelineWrapper.cs @@ -2,13 +2,12 @@ // SPDX-License-Identifier: Apache-2.0 using System; -using System.Collections.Generic; using System.Linq; -using System.Threading.Tasks; using NewRelic.Agent.Api; using NewRelic.Agent.Extensions.AwsSdk; using NewRelic.Agent.Extensions.Collections; using NewRelic.Agent.Extensions.Providers.Wrapper; +using NewRelic.Providers.Wrapper.AwsSdk.RequestHandlers; namespace NewRelic.Providers.Wrapper.AwsSdk { @@ -34,10 +33,13 @@ private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) try { accountId = GetAccountId(agent); - var clientconfig = requestContext.ClientConfig; - var regionEndpoint = clientconfig.RegionEndpoint; - systemName = regionEndpoint.SystemName; - partition = regionEndpoint.PartitionName; + var clientConfig = requestContext.ClientConfig; + if (clientConfig.RegionEndpoint != null) + { + var regionEndpoint = clientConfig.RegionEndpoint; + systemName = regionEndpoint.SystemName; + partition = regionEndpoint.PartitionName; + } } catch (Exception e) { @@ -48,12 +50,13 @@ private ArnBuilder CreateArnBuilder(IAgent agent, dynamic requestContext) } } - return new ArnBuilder(partition, systemName, accountId); ; + return new ArnBuilder(partition, systemName, accountId); } private string GetAccountId(IAgent agent) { - string accountId = agent.Configuration.AwsAccountId; + string accountId = AmazonServiceClientWrapper.AwsAccountId; + if (accountId != null) { if ((accountId.Length != 12) || accountId.Any(c => (c < '0') || (c > '9'))) @@ -65,6 +68,7 @@ private string GetAccountId(IAgent agent) } } } + return accountId; } @@ -103,17 +107,19 @@ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall ins { return SQSRequestHandler.HandleSQSRequest(instrumentedMethodCall, agent, transaction, request, isAsync, executionContext); } - else if (requestType == "Amazon.Lambda.Model.InvokeRequest") + + if (requestType == "Amazon.Lambda.Model.InvokeRequest") { return LambdaInvokeRequestHandler.HandleInvokeRequest(instrumentedMethodCall, agent, transaction, request, isAsync, builder); - } - else if (requestType.StartsWith("Amazon.DynamoDBv2")) + } + + if (requestType.StartsWith("Amazon.DynamoDBv2")) { - return DynamoDbRequestHandler.HandleDynamoDbRequest(instrumentedMethodCall, agent, transaction, request, isAsync, executionContext); + return DynamoDbRequestHandler.HandleDynamoDbRequest(instrumentedMethodCall, agent, transaction, request, isAsync, builder); } if (!_unsupportedRequestTypes.Contains(requestType)) // log once per unsupported request type - { + { agent.Logger.Debug($"AwsSdkPipelineWrapper: Unsupported request type: {requestType}. Returning NoOp delegate."); _unsupportedRequestTypes.Add(requestType); } diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/Instrumentation.xml b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/Instrumentation.xml index c909ea4cd5..556e0d0d82 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/Instrumentation.xml +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/Instrumentation.xml @@ -13,5 +13,14 @@ SPDX-License-Identifier: Apache-2.0 + + + + + + diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/DynamoDbRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/DynamoDbRequestHandler.cs similarity index 87% rename from src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/DynamoDbRequestHandler.cs rename to src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/DynamoDbRequestHandler.cs index 023ee2281e..4e3b9922aa 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/DynamoDbRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/DynamoDbRequestHandler.cs @@ -4,15 +4,16 @@ using System.Collections.Concurrent; using System.Threading.Tasks; using NewRelic.Agent.Api; +using NewRelic.Agent.Extensions.AwsSdk; using NewRelic.Agent.Extensions.Parsing; using NewRelic.Agent.Extensions.Providers.Wrapper; -namespace NewRelic.Providers.Wrapper.AwsSdk +namespace NewRelic.Providers.Wrapper.AwsSdk.RequestHandlers { internal static class DynamoDbRequestHandler { - private static ConcurrentDictionary _operationNameCache = new ConcurrentDictionary(); + private static ConcurrentDictionary _operationNameCache = new ConcurrentDictionary(); public static AfterWrappedMethodDelegate HandleDynamoDbRequest(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction, dynamic request, bool isAsync, dynamic executionContext) { @@ -30,6 +31,7 @@ public static AfterWrappedMethodDelegate HandleDynamoDbRequest(InstrumentedMetho model = request.TableName; var segment = transaction.StartDatastoreSegment(instrumentedMethodCall.MethodCall, new ParsedSqlStatement(DatastoreVendor.DynamoDB, model, operation), isLeaf: true); + return isAsync ? Delegates.GetAsyncDelegateFor(agent, segment) : diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs similarity index 95% rename from src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs rename to src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs index cb41437d2c..dc0694023e 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs @@ -5,18 +5,16 @@ using System; using System.Threading.Tasks; using NewRelic.Agent.Api; -using NewRelic.Agent.Api.Experimental; using NewRelic.Agent.Extensions.Providers.Wrapper; using NewRelic.Reflection; -using NewRelic.Agent.Extensions.Helpers; using NewRelic.Agent.Extensions.AwsSdk; -namespace NewRelic.Providers.Wrapper.AwsSdk +namespace NewRelic.Providers.Wrapper.AwsSdk.RequestHandlers { internal static class LambdaInvokeRequestHandler { private static Func _getResultFromGenericTask; - private static ConcurrentDictionary _arnCache = new ConcurrentDictionary(); + private static readonly ConcurrentDictionary _arnCache = new(); private static bool _reportMissingRequestId = true; private static bool _reportBadInvocationName = true; private const int MAX_CACHE_SIZE = 25; // Shouldn't ever get this big, but just in case diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/SQSRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/SQSRequestHandler.cs similarity index 95% rename from src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/SQSRequestHandler.cs rename to src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/SQSRequestHandler.cs index 45f4fecffb..587f4ba69f 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/SQSRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/SQSRequestHandler.cs @@ -11,7 +11,7 @@ using NewRelic.Agent.Extensions.Providers.Wrapper; using NewRelic.Reflection; -namespace NewRelic.Providers.Wrapper.AwsSdk +namespace NewRelic.Providers.Wrapper.AwsSdk.RequestHandlers { internal static class SQSRequestHandler { @@ -48,7 +48,7 @@ public static AfterWrappedMethodDelegate HandleSQSRequest(InstrumentedMethodCall var dtHeaders = agent.GetConfiguredDTHeaders(); string requestQueueUrl = request.QueueUrl; - ISegment segment = SqsHelper.GenerateSegment(transaction, instrumentedMethodCall.MethodCall, requestQueueUrl, action); + var segment = SqsHelper.GenerateSegment(transaction, instrumentedMethodCall.MethodCall, requestQueueUrl, action); switch (action) { case MessageBrokerAction.Produce when requestType == "SendMessageRequest": @@ -117,7 +117,7 @@ public static AfterWrappedMethodDelegate HandleSQSRequest(InstrumentedMethodCall void ProcessResponse(Task responseTask) { - if (!ValidTaskResponse(responseTask) || (segment == null) || action != MessageBrokerAction.Consume) + if (!ValidTaskResponse(responseTask) || segment == null || action != MessageBrokerAction.Consume) return; // taskResult is a ReceiveMessageResponse diff --git a/src/Agent/NewRelic/Home/Home.csproj b/src/Agent/NewRelic/Home/Home.csproj index 58165ba6a3..29cf52ca73 100644 --- a/src/Agent/NewRelic/Home/Home.csproj +++ b/src/Agent/NewRelic/Home/Home.csproj @@ -8,8 +8,8 @@ - - + + diff --git a/tests/Agent/IntegrationTests/IntegrationTestHelpers/RemoteServiceFixtures/RemoteApplicationFixture.cs b/tests/Agent/IntegrationTests/IntegrationTestHelpers/RemoteServiceFixtures/RemoteApplicationFixture.cs index c665fe3052..bfb7db67fe 100644 --- a/tests/Agent/IntegrationTests/IntegrationTestHelpers/RemoteServiceFixtures/RemoteApplicationFixture.cs +++ b/tests/Agent/IntegrationTests/IntegrationTestHelpers/RemoteServiceFixtures/RemoteApplicationFixture.cs @@ -351,6 +351,7 @@ public virtual void Initialize() catch (Exception ex) { TestLogger?.WriteLine("Exception occurred in Initialize: " + ex.ToString()); + AgentLogExpected = false; throw; } finally diff --git a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs index 464a4128d2..4d42f15097 100644 --- a/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs +++ b/tests/Agent/IntegrationTests/IntegrationTests/AwsSdk/InvokeLambdaTests.cs @@ -53,9 +53,6 @@ public InvokeLambdaTestBase(TFixture fixture, ITestOutputHelper output, bool use else { _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync {_function} \"fakepayload\""); - // This will fail without an ARN because there's no account ID - _fixture.AddCommand($"InvokeLambdaExerciser InvokeLambdaSync fakefunction fakepayload"); - } _fixture.Initialize(); } @@ -65,14 +62,15 @@ public void InvokeLambda() { var metrics = _fixture.AgentLog.GetMetrics().ToList(); + var expectedCount = _isAsync ? 2 : 1; var expectedMetrics = new List { - new Assertions.ExpectedMetric {metricName = @"DotNet/InvokeRequest", CallCountAllHarvests = 2}, + new Assertions.ExpectedMetric {metricName = @"DotNet/InvokeRequest", CallCountAllHarvests = expectedCount}, }; Assertions.MetricsExist(expectedMetrics, metrics); - var transactions = _fixture.AgentLog.GetTransactionEvents(); - Assert.Equal(2, transactions.Count()); + var transactions = _fixture.AgentLog.GetTransactionEvents().ToList(); + Assert.Equal(expectedCount, transactions.Count()); foreach (var transaction in transactions) { @@ -82,7 +80,7 @@ public void InvokeLambda() var allSpans = _fixture.AgentLog.GetSpanEvents() .Where(e => e.AgentAttributes.ContainsKey("cloud.platform")) .ToList(); - Assert.Equal(2, allSpans.Count); + Assert.Equal(expectedCount, allSpans.Count); foreach (var span in allSpans) { @@ -96,7 +94,7 @@ public void InvokeLambda() var spansWithArn = _fixture.AgentLog.GetSpanEvents() .Where(e => e.AgentAttributes.ContainsKey("cloud.resource_id")) .ToList(); - Assert.Equal(_isAsync ? 2 : 1, spansWithArn.Count); + Assert.Equal(expectedCount, spansWithArn.Count); foreach (var span in spansWithArn) { Assert.Equal(_arn, span.AgentAttributes["cloud.resource_id"]); diff --git a/tests/Agent/IntegrationTests/Shared/AwsBedrockConfiguration.cs b/tests/Agent/IntegrationTests/Shared/AwsConfiguration.cs similarity index 96% rename from tests/Agent/IntegrationTests/Shared/AwsBedrockConfiguration.cs rename to tests/Agent/IntegrationTests/Shared/AwsConfiguration.cs index fc446cb945..26158c52b0 100644 --- a/tests/Agent/IntegrationTests/Shared/AwsBedrockConfiguration.cs +++ b/tests/Agent/IntegrationTests/Shared/AwsConfiguration.cs @@ -3,7 +3,7 @@ namespace NewRelic.Agent.IntegrationTests.Shared { - public class AwsBedrockConfiguration + public class AwsConfiguration { public static string AwsAccessKeyId { diff --git a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/LLM/BedrockModels.cs b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/LLM/BedrockModels.cs index f7d16f8a9e..d2731d06d5 100644 --- a/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/LLM/BedrockModels.cs +++ b/tests/Agent/IntegrationTests/SharedApplications/Common/MultiFunctionApplicationHelpers/NetStandardLibraries/LLM/BedrockModels.cs @@ -20,7 +20,7 @@ namespace MultiFunctionApplicationHelpers.NetStandardLibraries.LLM internal class BedrockModels { private static readonly AmazonBedrockRuntimeClient _amazonBedrockRuntimeClient = - new AmazonBedrockRuntimeClient(AwsBedrockConfiguration.AwsAccessKeyId, AwsBedrockConfiguration.AwsSecretAccessKey, AwsBedrockConfiguration.AwsRegion.ToRegionId()); + new AmazonBedrockRuntimeClient(AwsConfiguration.AwsAccessKeyId, AwsConfiguration.AwsSecretAccessKey, AwsConfiguration.AwsRegion.ToRegionId()); [MethodImpl(MethodImplOptions.NoInlining)] public static async Task InvokeAmazonEmbedAsync(string prompt, bool generateError) => await InvokeTitanAsync(prompt, true, generateError); diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsAccountIdDecoderTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsAccountIdDecoderTests.cs new file mode 100644 index 0000000000..c884ed0f16 --- /dev/null +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsAccountIdDecoderTests.cs @@ -0,0 +1,102 @@ +// Copyright 2020 New Relic, Inc. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +using System; +using NewRelic.Agent.Extensions.AwsSdk; +using NUnit.Framework; +using Telerik.JustMock; + +namespace Agent.Extensions.Tests.Helpers +{ + [TestFixture] + internal class AwsAccountIdDecoderTests + { + [Test] + public void GetAccountId_ValidAwsAccessKeyId_ReturnsExpectedAccountId() + { + // Arrange + string awsAccessKeyId = "AKIAIOSFODNN7EXAMPLE"; // not a real AWS access key! + string expectedAccountId = "581039954779"; + + // Act + string actualAccountId = AwsAccountIdDecoder.GetAccountId(awsAccessKeyId); + + // Assert + Assert.That(expectedAccountId, Is.EqualTo(actualAccountId)); + } + + [Test] + public void GetAccountId_NullOrEmptyAwsAccessKeyId_ThrowsArgumentNullException() + { + // Arrange + string awsAccessKeyId = null; + + // Act & Assert + var ex = Assert.Throws(() => AwsAccountIdDecoder.GetAccountId(awsAccessKeyId)); + Assert.That(ex.ParamName, Is.EqualTo("awsAccessKeyId")); + } + + [Test] + public void GetAccountId_ShortAwsAccessKeyId_ThrowsArgumentOutOfRangeException() + { + // Arrange + string awsAccessKeyId = "AKIAIOSFODN"; + + // Act & Assert + var ex = Assert.Throws(() => AwsAccountIdDecoder.GetAccountId(awsAccessKeyId)); + Assert.That(ex.ParamName, Is.EqualTo("awsAccessKeyId")); + } + + [Test] + public void Base32Decode_ShortString_ThrowsArgumentException() + { + // Arrange + string shortString = "shortstr"; + + // Act & Assert + var ex = Assert.Throws(() => AwsAccountIdDecoder.Base32Decode(shortString)); + Assert.That(ex.ParamName, Is.EqualTo("src")); + } + + [Test] + public void Base32Decode_InvalidCharacters_ThrowsArgumentOutOfRangeException() + { + // Arrange + string invalidBase32String = "someBogusbase32string"; + + // Act & Assert + var ex = Assert.Throws(() => AwsAccountIdDecoder.Base32Decode(invalidBase32String)); + Assert.That(ex.ParamName, Is.EqualTo("src")); + } + + [Test] + public void Base32Decode_NullOrEmptyString_ThrowsArgumentNullException() + { + // Arrange + string nullString = null; + string emptyString = string.Empty; + + // Act & Assert + var exNull = Assert.Throws(() => AwsAccountIdDecoder.Base32Decode(nullString)); + Assert.That(exNull.ParamName, Is.EqualTo("src")); + + var exEmpty = Assert.Throws(() => AwsAccountIdDecoder.Base32Decode(emptyString)); + Assert.That(exEmpty.ParamName, Is.EqualTo("src")); + } + + [Test] + public void Base32Decode_ValidBase32String_ReturnsDecodedLong() + { + // Arrange + string validBase32String = "iosfodnn7example"; // Example valid Base32 string (10 characters) + long expectedDecodedValue = 74373114211833L; + + // Act + long decodedValue = AwsAccountIdDecoder.Base32Decode(validBase32String); + + // Assert + Assert.That(decodedValue, Is.EqualTo(expectedDecodedValue)); // Adjust expected value based on actual decoding logic + } + + } +} diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs index 126be85b63..e44e2d5722 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/AwsSdkHelperTests.cs @@ -11,7 +11,7 @@ public class AwsSdkArnBuilderTests [Test] [TestCase("myfunction", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction")] [TestCase("myfunction", "us-west-2", "", null)] - [TestCase("myfunction", "", "123456789012", null)] + [TestCase("myfunction", "", "123456789012", "arn:aws:lambda:(unknown):123456789012:function:myfunction")] [TestCase("myfunction:alias", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:myfunction:alias")] [TestCase("myfunction:alias", "us-west-2", "", null)] [TestCase("123456789012:function:my-function", "us-west-2", "123456789012", "arn:aws:lambda:us-west-2:123456789012:function:my-function")] @@ -59,7 +59,7 @@ public void ConstructLambdaArn(string name, string region, string accountId, str [TestCase("aws-cn", "lambda", "cn-north-1", "222233334444", "function_name", "arn:aws-cn:lambda:cn-north-1:222233334444:function_name")] [TestCase("aws-us-gov", "iam", "us-gov-west-1", "555566667777", "role_name", "arn:aws-us-gov:iam:us-gov-west-1:555566667777:role_name")] [TestCase("aws", "rds", "sa-east-1", "888899990000", "db_instance", "arn:aws:rds:sa-east-1:888899990000:db_instance")] - [TestCase("aws", "s3", "", "123456789012", "bucket_name", null)] + [TestCase("aws", "s3", "", "123456789012", "bucket_name", "arn:aws:s3:(unknown):123456789012:bucket_name")] [TestCase("aws", "s3", "us-west-2", "", "bucket_name", null)] public void ConstructGenericArn(string partition, string service, string region, string accountId, string resource, string expectedArn) { @@ -70,11 +70,11 @@ public void ConstructGenericArn(string partition, string service, string region, [Test] [TestCase("aws", "us-west-2", "123456789012", "Partition: aws, Region: us-west-2, AccountId: [Present]")] - [TestCase("aws", "", "123456789012", "Partition: aws, Region: [Missing], AccountId: [Present]")] + [TestCase("aws", "", "123456789012", "Partition: aws, Region: (unknown), AccountId: [Present]")] [TestCase("aws", "us-west-2", "", "Partition: aws, Region: us-west-2, AccountId: [Missing]")] [TestCase("aws", "us-west-2", null, "Partition: aws, Region: us-west-2, AccountId: [Missing]")] - [TestCase("", "", "", "Partition: [Missing], Region: [Missing], AccountId: [Missing]")] - [TestCase(null, null, null, "Partition: [Missing], Region: [Missing], AccountId: [Missing]")] + [TestCase("", "", "", "Partition: aws, Region: (unknown), AccountId: [Missing]")] + [TestCase(null, null, null, "Partition: aws, Region: (unknown), AccountId: [Missing]")] public void ArnBuilderToString(string partition, string region, string accountId, string expected) { var arnBuilder = new ArnBuilder(partition, region, accountId); From 8d41f5485a648db4328909c6c27f5f548b913eef Mon Sep 17 00:00:00 2001 From: Chris Hynes <111462425+chynesNR@users.noreply.github.com> Date: Tue, 10 Dec 2024 11:06:46 -0800 Subject: [PATCH 25/28] Moving cloud SDK attributes to Segments (#2924) --- .../Attributes/AttributeDefinitionService.cs | 1 + .../Agent/Core/Segments/NoOpSegment.cs | 4 +++ .../NewRelic/Agent/Core/Segments/Segment.cs | 13 ++++++++ .../Agent/Core/Transactions/Transaction.cs | 12 ------- .../NewRelic.Agent.Extensions/Api/ISpan.cs | 2 ++ .../Api/ITransaction.cs | 2 -- .../LambdaInvokeRequestHandler.cs | 16 +++++----- .../Transactions/TransactionTests.cs | 31 ------------------- .../Helpers/SqsHelperTests.cs | 4 +++ 9 files changed, 32 insertions(+), 53 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs b/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs index 9f1a7151b8..ab45080fee 100644 --- a/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs +++ b/src/Agent/NewRelic/Agent/Core/Attributes/AttributeDefinitionService.cs @@ -289,6 +289,7 @@ private AttributeDefinition CreateCloudSdkAttribute(string attri .Create(attribName, AttributeClassification.AgentAttributes) .AppliesTo(AttributeDestinations.TransactionTrace) .AppliesTo(AttributeDestinations.SpanEvent) + .WithConvert(x => x) .Build(_attribFilter); } diff --git a/src/Agent/NewRelic/Agent/Core/Segments/NoOpSegment.cs b/src/Agent/NewRelic/Agent/Core/Segments/NoOpSegment.cs index f38e5ddfb6..3b9ad1da87 100644 --- a/src/Agent/NewRelic/Agent/Core/Segments/NoOpSegment.cs +++ b/src/Agent/NewRelic/Agent/Core/Segments/NoOpSegment.cs @@ -61,6 +61,10 @@ public ISpan AddCustomAttribute(string key, object value) { return this; } + public ISpan AddCloudSdkAttribute(string key, object value) + { + return this; + } public ISpan SetName(string name) { diff --git a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs index 7323b80277..462d937395 100644 --- a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs +++ b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs @@ -449,6 +449,19 @@ public ISpan AddCustomAttribute(string key, object value) return this; } + public ISpan AddCloudSdkAttribute(string key, object value) + { + SpanAttributeValueCollection customAttribValues; + lock (_customAttribValuesSyncRoot) + { + customAttribValues = _customAttribValues ?? (_customAttribValues = new SpanAttributeValueCollection()); + } + + AttribDefs.GetCloudSdkAttribute(key).TrySetValue(customAttribValues, value); + + return this; + } + public ISpan SetName(string name) { SegmentNameOverride = name; diff --git a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs index 501fa9a2dd..46f82f827c 100644 --- a/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs +++ b/src/Agent/NewRelic/Agent/Core/Transactions/Transaction.cs @@ -1393,17 +1393,5 @@ public void AddFaasAttribute(string name, object value) var faasAttrib = _attribDefs.GetFaasAttribute(name); TransactionMetadata.UserAndRequestAttributes.TrySetValue(faasAttrib, value); } - - public void AddCloudSdkAttribute(string name, object value) - { - if (string.IsNullOrWhiteSpace(name)) - { - Log.Debug($"AddCloudSdkAttribute - Name cannot be null/empty"); - return; - } - - var cloudAttrib = _attribDefs.GetCloudSdkAttribute(name); - TransactionMetadata.UserAndRequestAttributes.TrySetValue(cloudAttrib, value); - } } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ISpan.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ISpan.cs index 86bc4d6451..6bca5d9ab9 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ISpan.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ISpan.cs @@ -11,6 +11,8 @@ public interface ISpan { ISpan AddCustomAttribute(string key, object value); + ISpan AddCloudSdkAttribute(string key, object value); + ISpan SetName(string name); } } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs index adf31f743d..c8fd50cfaf 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Api/ITransaction.cs @@ -315,7 +315,5 @@ ISegment StartMessageBrokerSegment(MethodCall methodCall, MessageBrokerDestinati void AddLambdaAttribute(string name, object value); void AddFaasAttribute(string name, object value); - - void AddCloudSdkAttribute(string name, object value); } } diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs index dc0694023e..b0bf16a8a7 100644 --- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs +++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/RequestHandlers/LambdaInvokeRequestHandler.cs @@ -30,13 +30,13 @@ private static object GetTaskResult(object task) return getResponse(task); } - private static void SetRequestIdIfAvailable(IAgent agent, ITransaction transaction, dynamic response) + private static void SetRequestIdIfAvailable(IAgent agent, ISegment segment, dynamic response) { try { dynamic metadata = response.ResponseMetadata; string requestId = metadata.RequestId; - transaction.AddCloudSdkAttribute("aws.requestId", requestId); + segment.AddCloudSdkAttribute("aws.requestId", requestId); } catch (Exception e) { @@ -74,14 +74,14 @@ public static AfterWrappedMethodDelegate HandleInvokeRequest(InstrumentedMethodC } var segment = transaction.StartTransactionSegment(instrumentedMethodCall.MethodCall, "InvokeRequest"); - transaction.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); - transaction.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); - transaction.AddCloudSdkAttribute("aws.region", builder.Region); + segment.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); + segment.AddCloudSdkAttribute("aws.operation", "InvokeRequest"); + segment.AddCloudSdkAttribute("aws.region", builder.Region); if (!string.IsNullOrEmpty(arn)) { - transaction.AddCloudSdkAttribute("cloud.resource_id", arn); + segment.AddCloudSdkAttribute("cloud.resource_id", arn); } else if (_reportBadInvocationName) { @@ -101,7 +101,7 @@ void InvokeTryProcessResponse(Task responseTask) { transaction.NoticeError(responseTask.Exception); } - SetRequestIdIfAvailable(agent, transaction, GetTaskResult(responseTask)); + SetRequestIdIfAvailable(agent, segment, GetTaskResult(responseTask)); } finally { @@ -115,7 +115,7 @@ void InvokeTryProcessResponse(Task responseTask) onFailure: ex => segment.End(ex), onSuccess: response => { - SetRequestIdIfAvailable(agent, transaction, response); + SetRequestIdIfAvailable(agent, segment, response); segment.End(); } ); diff --git a/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs b/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs index f14db94c45..929a38be34 100644 --- a/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs +++ b/tests/Agent/UnitTests/Core.UnitTest/Transactions/TransactionTests.cs @@ -591,35 +591,4 @@ public void HasHttpResponseStatusCode_ReturnsFalse_WhenStatusCodeIsNotSet() // Assert Assert.That(result, Is.False); } - public void AddCloudSdkAttribute_SetAttributeInTransactionMetadata() - { - // Arrange - var key = "TestAttribute"; - var value = "TestValue"; - - // Act - _transaction.AddCloudSdkAttribute(key, value); - - // Assert - var allAttributeValuesDic = _transaction.TransactionMetadata.UserAndRequestAttributes.GetAllAttributeValuesDic(); - - var attributeValue = allAttributeValuesDic[key]; - Assert.That(attributeValue, Is.EqualTo(value)); - } - - [TestCase(" ")] - [TestCase("")] - [TestCase(null)] - public void AddCloudSdkAttribute_DoesNotSetAttribute_WhenKeyIsBad(string key) - { - // Arrange - var value = "TestValue"; - - // Act - _transaction.AddCloudSdkAttribute(key, value); - - // Assert - Assert.That(_transaction.TransactionMetadata.UserAndRequestAttributes.Count, Is.EqualTo(0)); - } - } diff --git a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/SqsHelperTests.cs b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/SqsHelperTests.cs index e064d5c351..1e16c5e0f3 100644 --- a/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/SqsHelperTests.cs +++ b/tests/Agent/UnitTests/NewRelic.Agent.Extensions.Tests/Helpers/SqsHelperTests.cs @@ -217,6 +217,10 @@ public ISpan AddCustomAttribute(string key, object value) { throw new NotImplementedException(); } + public ISpan AddCloudSdkAttribute(string key, object value) + { + throw new NotImplementedException(); + } public ISpan SetName(string name) { From da13c620f31eafb9ed2f17d76543207698c04420 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 10 Dec 2024 15:13:27 -0800 Subject: [PATCH 26/28] Some cleanup and code coverage --- src/Agent/NewRelic/Agent/Core/Segments/Segment.cs | 14 +++++++------- .../UnitTests/CompositeTests/AgentApiTests.cs | 15 +++++++++++++-- 2 files changed, 20 insertions(+), 9 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs index 462d937395..0e6d995c26 100644 --- a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs +++ b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs @@ -38,7 +38,7 @@ public class Segment : IInternalSpan, ISegmentDataState public IAttributeDefinitions AttribDefs => _transactionSegmentState.AttribDefs; public string TypeName => MethodCallData.TypeName; - private SpanAttributeValueCollection _customAttribValues; + private SpanAttributeValueCollection _attribValues; public Segment(ITransactionSegmentState transactionSegmentState, MethodCallData methodCallData) { @@ -318,7 +318,7 @@ public TimeSpan ExclusiveDurationOrZero public SpanAttributeValueCollection GetAttributeValues() { - var attribValues = _customAttribValues ?? new SpanAttributeValueCollection(); + var attribValues = _attribValues ?? new SpanAttributeValueCollection(); AttribDefs.Duration.TrySetValue(attribValues, DurationOrZero); AttribDefs.NameForSpan.TrySetValue(attribValues, GetTransactionTraceName()); @@ -434,14 +434,14 @@ public ISegmentExperimental MakeLeaf() return this; } - private readonly object _customAttribValuesSyncRoot = new object(); + private readonly object _attribValuesSyncRoot = new object(); public ISpan AddCustomAttribute(string key, object value) { SpanAttributeValueCollection customAttribValues; - lock (_customAttribValuesSyncRoot) + lock (_attribValuesSyncRoot) { - customAttribValues = _customAttribValues ?? (_customAttribValues = new SpanAttributeValueCollection()); + customAttribValues = _attribValues ?? (_attribValues = new SpanAttributeValueCollection()); } AttribDefs.GetCustomAttributeForSpan(key).TrySetValue(customAttribValues, value); @@ -452,9 +452,9 @@ public ISpan AddCustomAttribute(string key, object value) public ISpan AddCloudSdkAttribute(string key, object value) { SpanAttributeValueCollection customAttribValues; - lock (_customAttribValuesSyncRoot) + lock (_attribValuesSyncRoot) { - customAttribValues = _customAttribValues ?? (_customAttribValues = new SpanAttributeValueCollection()); + customAttribValues = _attribValues ?? (_attribValues = new SpanAttributeValueCollection()); } AttribDefs.GetCloudSdkAttribute(key).TrySetValue(customAttribValues, value); diff --git a/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs b/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs index c0da43590b..206275898b 100644 --- a/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs +++ b/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs @@ -2211,10 +2211,10 @@ public void Test_GetLinkingMetadataReturnsValidValuesIfDTEnabled() #endregion GetLinkingMetadataCATS - #region Span Custom Attributes + #region Span Attributes [Test] - public void SpanCustomAttributes() + public void SpanAttributes() { var agentWrapperApi = _compositeTestAgent.GetAgent(); var dtm1 = DateTime.Now; @@ -2238,6 +2238,9 @@ public void SpanCustomAttributes() segment.AddCustomAttribute("key7", dtm2); segment.AddCustomAttribute("key8", null); segment.AddCustomAttribute("", dtm2); + segment.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); + segment.AddCloudSdkAttribute("aws.region", "us-west-2"); + segment.AddCloudSdkAttribute("cloud.resource_id", "arn:aws:lambda:us-west-2:123456789012:function:myfunction"); var singleStringValue = new StringValues("avalue"); var multiStringValue = new StringValues(new[] { "onevalue", "twovalue", "threevalue" }); @@ -2257,6 +2260,13 @@ public void SpanCustomAttributes() new ExpectedAttribute(){ Key = "key9b", Value = "onevalue,twovalue,threevalue"} }; + var expectedCloudSdkAttributes = new[] + { + new ExpectedAttribute(){ Key = "cloud.platform", Value = "aws_lambda"}, + new ExpectedAttribute(){ Key = "aws.region", Value = "us-west-2"}, + new ExpectedAttribute(){ Key = "cloud.resource_id", Value = "arn:aws:lambda:us-west-2:123456789012:function:myfunction"}, + }; + var unexpectedAttributes = new[] { "key8", @@ -2275,6 +2285,7 @@ public void SpanCustomAttributes() ( () => Assert.That(allSpans, Has.Count.EqualTo(2)), () => SpanAssertions.HasAttributes(expectedAttributes, AttributeClassification.UserAttributes, testSpan), + () => SpanAssertions.HasAttributes(expectedCloudSdkAttributes, AttributeClassification.AgentAttributes, testSpan), () => SpanAssertions.DoesNotHaveAttributes(unexpectedAttributes, AttributeClassification.UserAttributes, testSpan) ); } From 662c1cd7d7bbdb5cb7ec23ebfe0e3de9bb9b9dc4 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 10 Dec 2024 16:11:41 -0800 Subject: [PATCH 27/28] More cleanup and code coverage --- .../NewRelic/Agent/Core/Segments/Segment.cs | 6 +- .../AwsSdk/ArnBuilder.cs | 5 +- .../UnitTests/CompositeTests/AgentApiTests.cs | 60 +++++++++++++++---- .../Core.UnitTest/Segments/SegmentTests.cs | 53 ++++++++++++++++ 4 files changed, 104 insertions(+), 20 deletions(-) diff --git a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs index 0e6d995c26..968e7d654b 100644 --- a/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs +++ b/src/Agent/NewRelic/Agent/Core/Segments/Segment.cs @@ -451,13 +451,13 @@ public ISpan AddCustomAttribute(string key, object value) public ISpan AddCloudSdkAttribute(string key, object value) { - SpanAttributeValueCollection customAttribValues; + SpanAttributeValueCollection attribValues; lock (_attribValuesSyncRoot) { - customAttribValues = _attribValues ?? (_attribValues = new SpanAttributeValueCollection()); + attribValues = _attribValues ?? (_attribValues = new SpanAttributeValueCollection()); } - AttribDefs.GetCloudSdkAttribute(key).TrySetValue(customAttribValues, value); + AttribDefs.GetCloudSdkAttribute(key).TrySetValue(attribValues, value); return this; } diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs index 4030057429..d40a53aaaa 100644 --- a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs +++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/AwsSdk/ArnBuilder.cs @@ -119,11 +119,8 @@ public string BuildFromPartialLambdaArn(string invocationName) return null; } + // The member Region cannot be blank (it has a default) so we don't need to check it here region = !string.IsNullOrEmpty(region) ? region : Region; - if (string.IsNullOrEmpty(region)) - { - return null; - } if (!string.IsNullOrEmpty(alias)) { diff --git a/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs b/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs index 206275898b..5da764b13e 100644 --- a/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs +++ b/tests/Agent/UnitTests/CompositeTests/AgentApiTests.cs @@ -2211,10 +2211,10 @@ public void Test_GetLinkingMetadataReturnsValidValuesIfDTEnabled() #endregion GetLinkingMetadataCATS - #region Span Attributes + #region Span Custom Attributes [Test] - public void SpanAttributes() + public void SpanCustomAttributes() { var agentWrapperApi = _compositeTestAgent.GetAgent(); var dtm1 = DateTime.Now; @@ -2238,9 +2238,6 @@ public void SpanAttributes() segment.AddCustomAttribute("key7", dtm2); segment.AddCustomAttribute("key8", null); segment.AddCustomAttribute("", dtm2); - segment.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); - segment.AddCloudSdkAttribute("aws.region", "us-west-2"); - segment.AddCloudSdkAttribute("cloud.resource_id", "arn:aws:lambda:us-west-2:123456789012:function:myfunction"); var singleStringValue = new StringValues("avalue"); var multiStringValue = new StringValues(new[] { "onevalue", "twovalue", "threevalue" }); @@ -2260,13 +2257,6 @@ public void SpanAttributes() new ExpectedAttribute(){ Key = "key9b", Value = "onevalue,twovalue,threevalue"} }; - var expectedCloudSdkAttributes = new[] - { - new ExpectedAttribute(){ Key = "cloud.platform", Value = "aws_lambda"}, - new ExpectedAttribute(){ Key = "aws.region", Value = "us-west-2"}, - new ExpectedAttribute(){ Key = "cloud.resource_id", Value = "arn:aws:lambda:us-west-2:123456789012:function:myfunction"}, - }; - var unexpectedAttributes = new[] { "key8", @@ -2285,11 +2275,55 @@ public void SpanAttributes() ( () => Assert.That(allSpans, Has.Count.EqualTo(2)), () => SpanAssertions.HasAttributes(expectedAttributes, AttributeClassification.UserAttributes, testSpan), - () => SpanAssertions.HasAttributes(expectedCloudSdkAttributes, AttributeClassification.AgentAttributes, testSpan), () => SpanAssertions.DoesNotHaveAttributes(unexpectedAttributes, AttributeClassification.UserAttributes, testSpan) ); } #endregion + + #region Span Cloud SDK Attributes + [Test] + public void SpanCloudSdkAttributes() + { + var agentWrapperApi = _compositeTestAgent.GetAgent(); + var dtm1 = DateTime.Now; + var dtm2 = DateTimeOffset.Now; + + // ACT + var transaction = agentWrapperApi.CreateTransaction( + isWeb: true, + category: EnumNameCache.GetName(WebTransactionType.Action), + transactionDisplayName: "name", + doNotTrackAsUnitOfWork: true); + + var segment = agentWrapperApi.StartTransactionSegmentOrThrow("segment"); + + segment.AddCloudSdkAttribute("cloud.platform", "aws_lambda"); + segment.AddCloudSdkAttribute("aws.region", "us-west-2"); + segment.AddCloudSdkAttribute("cloud.resource_id", "arn:aws:lambda:us-west-2:123456789012:function:myfunction"); + + var expectedAttributes = new[] + { + new ExpectedAttribute(){ Key = "cloud.platform", Value = "aws_lambda"}, + new ExpectedAttribute(){ Key = "aws.region", Value = "us-west-2"}, + new ExpectedAttribute(){ Key = "cloud.resource_id", Value = "arn:aws:lambda:us-west-2:123456789012:function:myfunction"}, + }; + + segment.End(); + transaction.End(); + + _compositeTestAgent.Harvest(); + + var allSpans = _compositeTestAgent.SpanEvents; + var testSpan = allSpans.LastOrDefault(); + + NrAssert.Multiple + ( + () => Assert.That(allSpans, Has.Count.EqualTo(2)), + () => SpanAssertions.HasAttributes(expectedAttributes, AttributeClassification.AgentAttributes, testSpan) + ); + } + + #endregion } } diff --git a/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs b/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs index 7a9e87fbe0..e54082e069 100644 --- a/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs +++ b/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs @@ -58,5 +58,58 @@ public void DurationOrZero_ReturnsDuration_IfDurationIsSet() Assert.That(duration, Is.EqualTo(TimeSpan.FromSeconds(1))); } + [Test] + public void Misc_Segment_Setters() + { + var segment = new Segment(TransactionSegmentStateHelpers.GetItransactionSegmentState(), new MethodCallData("Type", "Method", 1)); + Assert.That(segment.IsLeaf, Is.False); + segment.MakeLeaf(); + Assert.That(segment.IsLeaf, Is.True); + + Assert.That(segment.GetTransactionTraceName(), Is.Null); + segment.SetName("foo"); + Assert.That(segment.GetTransactionTraceName(), Is.EqualTo("foo")); + + Assert.That(segment.Combinable, Is.False); + segment.MakeCombinable(); + Assert.That(segment.Combinable, Is.True); + + Assert.That(segment.IsExternal, Is.False); + } + + [Test] + public void NoOpSegment() + { + var segment = new NoOpSegment(); + Assert.That(segment.IsDone, Is.True); + Assert.That(segment.IsValid, Is.False); + Assert.That(segment.IsDone, Is.True); + Assert.That(segment.DurationShouldBeDeductedFromParent, Is.False); + Assert.That(segment.IsLeaf, Is.False); + Assert.That(segment.IsExternal, Is.False); + Assert.That(segment.SpanId, Is.Null); + Assert.That(segment.SegmentData, Is.Not.Null); + Assert.That(segment.AttribDefs, Is.Not.Null); + Assert.That(segment.AttribValues, Is.Not.Null); + Assert.That(segment.TypeName, Is.EqualTo(string.Empty)); + Assert.That(segment.UserCodeFunction, Is.EqualTo(string.Empty)); + Assert.That(segment.UserCodeNamespace, Is.EqualTo(string.Empty)); + Assert.That(segment.SegmentNameOverride, Is.Null); + + Assert.DoesNotThrow(() => segment.End()); + Assert.DoesNotThrow(() => segment.End(new Exception())); + Assert.DoesNotThrow(() => segment.EndStackExchangeRedis()); + Assert.DoesNotThrow(() => segment.MakeCombinable()); + Assert.DoesNotThrow(() => segment.MakeLeaf()); + Assert.DoesNotThrow(() => segment.RemoveSegmentFromCallStack()); + Assert.DoesNotThrow(() => segment.SetMessageBrokerDestination("destination")); + Assert.DoesNotThrow(() => segment.SetSegmentData(null)); + Assert.DoesNotThrow(() => segment.AddCustomAttribute("key", "value")); + Assert.DoesNotThrow(() => segment.AddCloudSdkAttribute("key", "value")); + Assert.DoesNotThrow(() => segment.SetName("name")); + Assert.That(segment.GetCategory(), Is.EqualTo(string.Empty)); + Assert.That(segment.DurationOrZero, Is.EqualTo(TimeSpan.Zero)); + + } } } From 4537d5de4b7de4ec3955cddede48f1911cf9b964 Mon Sep 17 00:00:00 2001 From: chynesNR Date: Tue, 10 Dec 2024 16:25:30 -0800 Subject: [PATCH 28/28] Fixed unit test --- tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs | 1 - 1 file changed, 1 deletion(-) diff --git a/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs b/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs index e54082e069..6c7c59aab8 100644 --- a/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs +++ b/tests/Agent/UnitTests/Core.UnitTest/Segments/SegmentTests.cs @@ -66,7 +66,6 @@ public void Misc_Segment_Setters() segment.MakeLeaf(); Assert.That(segment.IsLeaf, Is.True); - Assert.That(segment.GetTransactionTraceName(), Is.Null); segment.SetName("foo"); Assert.That(segment.GetTransactionTraceName(), Is.EqualTo("foo"));