Skip to content

Commit 612624c

Browse files
Fix race condition in GrpcMonitor's GrpcChannel management (pulumi#304)
While creating integration tests I faced a race condition in GrpcMonitor when the channel retrieved from the static _monitorChannels was null although the key had already been set. Replacing the complex implementation with custom locking by the standard pattern ConcurrentDictionary<K, Lazy<V>> has resolved this.
1 parent d3ab058 commit 612624c

File tree

2 files changed

+26
-31
lines changed

2 files changed

+26
-31
lines changed
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
component: sdk
2+
kind: Bug Fixes
3+
body: Fix race condition in GrpcMonitor's GrpcChannel management
4+
time: 2024-07-21T08:17:42.5134291+02:00
5+
custom:
6+
PR: "304"

sdk/Pulumi/Deployment/GrpcMonitor.cs

+20-31
Original file line numberDiff line numberDiff line change
@@ -13,43 +13,32 @@ namespace Pulumi
1313
internal class GrpcMonitor : IMonitor
1414
{
1515
private readonly ResourceMonitor.ResourceMonitorClient _client;
16+
1617
// Using a static dictionary to keep track of and re-use gRPC channels
1718
// According to the docs (https://docs.microsoft.com/en-us/aspnet/core/grpc/performance?view=aspnetcore-6.0#reuse-grpc-channels), creating GrpcChannels is expensive so we keep track of a bunch of them here
18-
private static readonly ConcurrentDictionary<string, GrpcChannel> _monitorChannels = new ConcurrentDictionary<string, GrpcChannel>();
19-
private static readonly object _channelsLock = new object();
19+
private static readonly ConcurrentDictionary<string, Lazy<GrpcChannel>> _monitorChannels = new();
20+
2021
public GrpcMonitor(string monitorAddress)
2122
{
22-
// maxRpcMessageSize raises the gRPC Max Message size from `4194304` (4mb) to `419430400` (400mb)
23+
var monitorChannel = _monitorChannels.GetOrAdd(monitorAddress, LazyCreateChannel);
24+
this._client = new ResourceMonitor.ResourceMonitorClient(monitorChannel.Value);
25+
}
26+
27+
private static Lazy<GrpcChannel> LazyCreateChannel(string monitorAddress)
28+
{
29+
return new Lazy<GrpcChannel>(() => CreateChannel(monitorAddress));
30+
}
31+
32+
private static GrpcChannel CreateChannel(string monitorAddress)
33+
{
2334
const int maxRpcMessageSize = 400 * 1024 * 1024;
24-
if (_monitorChannels.TryGetValue(monitorAddress, out var monitorChannel))
35+
var channel = GrpcChannel.ForAddress(new Uri($"http://{monitorAddress}"), new GrpcChannelOptions
2536
{
26-
// A channel already exists for this address
27-
this._client = new ResourceMonitor.ResourceMonitorClient(monitorChannel);
28-
}
29-
else
30-
{
31-
lock (_channelsLock)
32-
{
33-
if (_monitorChannels.TryGetValue(monitorAddress, out var existingChannel))
34-
{
35-
// A channel already exists for this address
36-
this._client = new ResourceMonitor.ResourceMonitorClient(monitorChannel);
37-
}
38-
else
39-
{
40-
// Inititialize the monitor channel once for this monitor address
41-
var channel = GrpcChannel.ForAddress(new Uri($"http://{monitorAddress}"), new GrpcChannelOptions
42-
{
43-
MaxReceiveMessageSize = maxRpcMessageSize,
44-
MaxSendMessageSize = maxRpcMessageSize,
45-
Credentials = ChannelCredentials.Insecure
46-
});
47-
48-
_monitorChannels[monitorAddress] = channel;
49-
this._client = new ResourceMonitor.ResourceMonitorClient(channel);
50-
}
51-
}
52-
}
37+
MaxReceiveMessageSize = maxRpcMessageSize,
38+
MaxSendMessageSize = maxRpcMessageSize,
39+
Credentials = ChannelCredentials.Insecure
40+
});
41+
return channel;
5342
}
5443

5544
public async Task<SupportsFeatureResponse> SupportsFeatureAsync(SupportsFeatureRequest request)

0 commit comments

Comments
 (0)