When using Azure SDK .NET client libraries in high throughput applications, it’s important to know how to maximize performance and avoid extra allocations while preventing bugs that could be introduced by accessing data from multiple threads. This article covers the best practices for using clients and models efficiently.
Client lifetime
The main rule of Azure SDK client lifetime management is: treat clients as singletons.
There is no need to keep more than one instance of a client for a given set of constructor parameters or client options. This can be implemented in many ways: creating an instance once and passing it around as a parameter, storing an instance in a field, or registering it as a singleton in a dependency injection container of your choice.
❌ Bad (extra allocations and initialization):
foreach (var secretName in secretNames)
{
var client = new SecretClient(
new Uri("<secrets_endpoint>"),
new DefaultAzureCredential());
KeyVaultSecret secret = client.GetSecret(secretName);
// code omitted for brevity
}
✔️ Good:
var client = new SecretClient(
new Uri("<secrets_endpoint>"),
new DefaultAzureCredential());
foreach (var secretName in secretNames)
{
KeyVaultSecret secret = client.GetSecret(secretName);
// code omitted for brevity
}
✔️ Also good:
public class Program
{
internal static SecretClient Client;
public static void Main()
{
Client = new SecretClient(
new Uri("<secrets_endpoint>"),
new DefaultAzureCredential());
}
}
public class OtherClass
{
public string DoWork()
{
KeyVaultSecret secret = Program.Client.GetSecret(settingName);
// code omitted for brevity
}
}
Thread-safety: Clients are thread-safe
We guarantee that all client instance methods are thread-safe and independent of each other (guideline). This ensures that the recommendation of reusing client instances is always safe, even across threads.
✔️ Good:
var client = new SecretClient(
new Uri("<secrets_endpoint>"),
new DefaultAzureCredential());
foreach (var secretName in secretNames)
{
// Using clients from parallel threads
Task.Run(() => {
// code that uses client here
});
}
Thread-safety: Models are not thread-safe
Because most model use-cases involve a single thread and to avoid incurring an extra synchronization cost the input and output models of the client methods are non-thread-safe and can only be accessed by one thread at a time. The following sample illustrates a bug where accessing a model from multiple threads might cause an undefined behavior.
❌ Bad:
KeyVaultSecret newSecret = client.SetSecret("secret", "value");
foreach (var tag in tags)
{
// Don't use model type from parallel threads
Task.Run(() => newSecret.Properties.Tags[tag] = CalculateTagValue(tag));
}
client.UpdateSecretProperties(newSecret.Properties);
If you need to access the model from different threads use a synchronization primitive.
✔️ Good:
KeyVaultSecret newSecret = client.SetSecret("secret", "value");
foreach (var tag in tags)
{
Task.Run(() =>
{
lock (newSecret)
{
newSecret.Properties.Tags[tag] = CalculateTagValue(tag);
}
);
}
client.UpdateSecretProperties(newSecret.Properties);
Clients are immutable
Clients are immutable after being created, which also makes them safe to share and reuse safely (guideline). This means that after the client is constructed, you cannot change the endpoint it connects to, the credential, and other values passed via the client options.
❌ Bad (configuration changes are ignored):
var secretClientOptions = new SecretClientOptions()
{
Retry =
{
Delay = TimeSpan.FromSeconds(5)
}
};
var mySecretClient = new SecretClient(
new Uri("<...>"),
new DefaultAzureCredential(),
secretClientOptions);
// This has no effect on the mySecretClient instance
secretClientOptions.Retry.Delay = TimeSpan.FromSeconds(100);
NOTE: An important exception from this rule are credential type implementations that are required to support rolling the key after the client was created (guideline). Examples of such types include AzureKeyCredential and StorageSharedKeyCredential. This feature is to enable long-running applications while using limited-time keys that need to be rolled periodically without requiring application restart or client re-creation.
Clients are not disposable: Shared HttpClient as default
One question that comes up often is why aren’t HTTP-based Azure clients implementing IDisposable
while internally using an HttpClient
that is disposable? All Azure SDK clients, by default, use a single shared HttpClient
instance and don’t create any other resources that need to be actively freed. The shared client instance persists throughout the entire application lifetime.
// Both clients reuse the shared HttpClient and don't need to be disposed
var blobClient = new BlobClient(new Uri(sasUri));
var blobClient2 = new BlobClient(new Uri(sasUri2));
Explicitly disposing customer provided HttpClient instances
If you provide a custom instance of HttpClient
to an Azure client, you become responsible for managing the HttpClient
lifetime and disposing it at the right time. We recommend following HttpClient
best practices when customizing the transport.
var httpClient = new HttpClient();
var clientOptions = new BlobClientOptions()
{
Transport = new HttpClientTransport(httpClient)
}
// Both client would use the HttpClient instance provided in clientOptions
var blobClient = new BlobClient(new Uri(sasUri), clientOptions);
var blobClient2 = new BlobClient(new Uri(sasUri2), clientOptions);
//...
// some time later
httpClient.Dispose();
Using ASP.NET Core
If you are using Azure SDK clients in an ASP.NET Core application, client lifetime management can be simplified with the Microsoft.Extensions.Azure package that provides seamless integration of Azure clients with the ASP.NET Core dependency injection and configuration systems. See the Best practices for using Azure SDK with ASP.NET Core blog post or Microsoft.Extensions.Azure package readme for details.
0 comments