diff --git a/.github/workflows/build_profiler.yml b/.github/workflows/build_profiler.yml
index b6a3748b32..622bc84e75 100644
--- a/.github/workflows/build_profiler.yml
+++ b/.github/workflows/build_profiler.yml
@@ -7,6 +7,13 @@ on:
- "feature/**"
paths-ignore:
- ".github/**" # skip changes to the .github folder (workflows, etc.)
+
+ # needs to run on every push to main to keep CodeCov in sync
+ push:
+ branches:
+ - main
+ paths-ignore:
+ - ".github/**" # skip changes to the .github folder (workflows, etc.)
# this workflow can be called from another workflow
workflow_call:
diff --git a/.github/workflows/run_unit_tests.yml b/.github/workflows/run_unit_tests.yml
index 46d2b9a9a4..019d79a8e3 100644
--- a/.github/workflows/run_unit_tests.yml
+++ b/.github/workflows/run_unit_tests.yml
@@ -5,6 +5,11 @@ on:
branches:
- main
- "feature/**"
+
+ # needs to run on every push to main to keep CodeCov in sync
+ push:
+ branches:
+ - main
workflow_dispatch: # allows for manual trigger
diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUCache.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUCache.cs
new file mode 100644
index 0000000000..9dc81d1446
--- /dev/null
+++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUCache.cs
@@ -0,0 +1,121 @@
+// Copyright 2020 New Relic, Inc. All rights reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+using System;
+using System.Collections.Generic;
+using System.Threading;
+
+namespace NewRelic.Agent.Extensions.Caching
+{
+ ///
+ /// A thread-safe LRU cache implementation.
+ ///
+ ///
+ ///
+ public class LRUCache
+ {
+ private readonly int _capacity;
+ private readonly Dictionary> _cacheMap;
+ private readonly LinkedList _lruList;
+ private readonly ReaderWriterLockSlim _lock = new();
+
+ public LRUCache(int capacity)
+ {
+ if (capacity <= 0)
+ {
+ throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
+ }
+
+ _capacity = capacity;
+ _cacheMap = new Dictionary>(capacity);
+ _lruList = new LinkedList();
+ }
+
+ public TValue Get(TKey key)
+ {
+ _lock.EnterUpgradeableReadLock();
+ try
+ {
+ if (_cacheMap.TryGetValue(key, out var node))
+ {
+ // Move the accessed node to the front of the list
+ _lock.EnterWriteLock();
+ try
+ {
+ _lruList.Remove(node);
+ _lruList.AddFirst(node);
+ }
+ finally
+ {
+ _lock.ExitWriteLock();
+ }
+ return node.Value.Value;
+ }
+ throw new KeyNotFoundException("The given key was not present in the cache.");
+ }
+ finally
+ {
+ _lock.ExitUpgradeableReadLock();
+ }
+ }
+
+ public void Put(TKey key, TValue value)
+ {
+ _lock.EnterWriteLock();
+ try
+ {
+ if (_cacheMap.TryGetValue(key, out var node))
+ {
+ // Update the value and move the node to the front of the list
+ node.Value.Value = value;
+ _lruList.Remove(node);
+ _lruList.AddFirst(node);
+ }
+ else
+ {
+ if (_cacheMap.Count >= _capacity)
+ {
+ // Remove the least recently used item
+ var lruNode = _lruList.Last;
+ _cacheMap.Remove(lruNode.Value.Key);
+ _lruList.RemoveLast();
+ }
+
+ // Add the new item to the cache
+ var cacheItem = new CacheItem(key, value);
+ var newNode = new LinkedListNode(cacheItem);
+ _lruList.AddFirst(newNode);
+ _cacheMap[key] = newNode;
+ }
+ }
+ finally
+ {
+ _lock.ExitWriteLock();
+ }
+ }
+ public bool ContainsKey(TKey key)
+ {
+ _lock.EnterReadLock();
+ try
+ {
+ return _cacheMap.ContainsKey(key);
+ }
+ finally
+ {
+ _lock.ExitReadLock();
+ }
+ }
+
+ private class CacheItem
+ {
+ public TKey Key { get; }
+ public TValue Value { get; set; }
+
+ public CacheItem(TKey key, TValue value)
+ {
+ Key = key;
+ Value = value;
+ }
+ }
+ }
+}
diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUHashSet.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUHashSet.cs
new file mode 100644
index 0000000000..33ee77a056
--- /dev/null
+++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/LRUHashSet.cs
@@ -0,0 +1,114 @@
+// Copyright 2020 New Relic, Inc. All rights reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+using System;
+using System.Collections.Generic;
+using System.Threading;
+
+namespace NewRelic.Agent.Extensions.Caching
+{
+ ///
+ /// A thread-safe LRU HashSet implementation.
+ ///
+ ///
+ public class LRUHashSet
+ {
+ private readonly int _capacity;
+ private readonly HashSet _hashSet;
+ private readonly LinkedList _lruList;
+ private readonly ReaderWriterLockSlim _lock = new();
+
+ public LRUHashSet(int capacity)
+ {
+ if (capacity <= 0)
+ {
+ throw new ArgumentException("Capacity must be greater than zero.", nameof(capacity));
+ }
+
+ _capacity = capacity;
+ _hashSet = new HashSet();
+ _lruList = new LinkedList();
+ }
+
+ public bool Add(T item)
+ {
+ _lock.EnterWriteLock();
+ try
+ {
+ if (_hashSet.Contains(item))
+ {
+ // Move the accessed item to the front of the list
+ _lruList.Remove(item);
+ _lruList.AddFirst(item);
+ return false;
+ }
+ else
+ {
+ if (_hashSet.Count >= _capacity)
+ {
+ // Remove the least recently used item
+ var lruItem = _lruList.Last.Value;
+ _hashSet.Remove(lruItem);
+ _lruList.RemoveLast();
+ }
+
+ // Add the new item to the set and list
+ _hashSet.Add(item);
+ _lruList.AddFirst(item);
+ return true;
+ }
+ }
+ finally
+ {
+ _lock.ExitWriteLock();
+ }
+ }
+
+ public bool Contains(T item)
+ {
+ _lock.EnterReadLock();
+ try
+ {
+ return _hashSet.Contains(item);
+ }
+ finally
+ {
+ _lock.ExitReadLock();
+ }
+ }
+
+ public bool Remove(T item)
+ {
+ _lock.EnterWriteLock();
+ try
+ {
+ if (_hashSet.Remove(item))
+ {
+ _lruList.Remove(item);
+ return true;
+ }
+ return false;
+ }
+ finally
+ {
+ _lock.ExitWriteLock();
+ }
+ }
+
+ public int Count
+ {
+ get
+ {
+ _lock.EnterReadLock();
+ try
+ {
+ return _hashSet.Count;
+ }
+ finally
+ {
+ _lock.ExitReadLock();
+ }
+ }
+ }
+ }
+}
diff --git a/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/WeakReferenceKey.cs b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/WeakReferenceKey.cs
new file mode 100644
index 0000000000..478498fe5e
--- /dev/null
+++ b/src/Agent/NewRelic/Agent/Extensions/NewRelic.Agent.Extensions/Caching/WeakReferenceKey.cs
@@ -0,0 +1,50 @@
+// Copyright 2020 New Relic, Inc. All rights reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+using System;
+
+namespace NewRelic.Agent.Extensions.Caching
+{
+ ///
+ /// Creates an object that can be used as a dictionary key, which holds a WeakReference<T>
+ ///
+ ///
+ public class WeakReferenceKey where T : class
+ {
+ private WeakReference WeakReference { get; }
+
+ public WeakReferenceKey(T cacheKey)
+ {
+ WeakReference = new WeakReference(cacheKey);
+ }
+
+ public override bool Equals(object obj)
+ {
+ if (obj is WeakReferenceKey otherKey)
+ {
+ if (WeakReference.TryGetTarget(out var thisTarget) &&
+ otherKey.WeakReference.TryGetTarget(out var otherTarget))
+ {
+ return ReferenceEquals(thisTarget, otherTarget);
+ }
+ }
+
+ return false;
+ }
+
+ public override int GetHashCode()
+ {
+ if (WeakReference.TryGetTarget(out var target))
+ {
+ return target.GetHashCode();
+ }
+
+ return 0;
+ }
+
+ ///
+ /// Gets the value from the WeakReference or null if the target has been garbage collected.
+ ///
+ public T Value => WeakReference.TryGetTarget(out var target) ? target : null;
+ }
+}
diff --git a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs
index 615a9ea8e5..d61c7d642f 100644
--- a/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs
+++ b/src/Agent/NewRelic/Agent/Extensions/Providers/Wrapper/AwsSdk/AmazonServiceClientWrapper.cs
@@ -4,49 +4,62 @@
using System;
using NewRelic.Agent.Api;
using NewRelic.Agent.Extensions.AwsSdk;
+using NewRelic.Agent.Extensions.Caching;
using NewRelic.Agent.Extensions.Providers.Wrapper;
-namespace NewRelic.Providers.Wrapper.AwsSdk
+namespace NewRelic.Providers.Wrapper.AwsSdk;
+
+public class AmazonServiceClientWrapper : IWrapper
{
- public class AmazonServiceClientWrapper : IWrapper
+ private const int LRUCapacity = 100;
+ // cache the account id per instance of AmazonServiceClient.Config
+ public static LRUCache, string> AwsAccountIdByClientConfigCache = new(LRUCapacity);
+
+ // cache instances of AmazonServiceClient
+ private static readonly LRUHashSet> AmazonServiceClientInstanceCache = new(LRUCapacity);
+
+ public bool IsTransactionRequired => false;
+
+ public CanWrapResponse CanWrap(InstrumentedMethodInfo instrumentedMethodInfo)
{
- ///
- /// 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; }
+ return new CanWrapResponse(instrumentedMethodInfo.RequestedWrapperName == nameof(AmazonServiceClientWrapper));
+ }
+
+ public AfterWrappedMethodDelegate BeforeWrappedMethod(InstrumentedMethodCall instrumentedMethodCall, IAgent agent, ITransaction transaction)
+ {
+ object client = instrumentedMethodCall.MethodCall.InvocationTarget;
- public bool IsTransactionRequired => false;
+ var weakReferenceKey = new WeakReferenceKey