NOTE: Samples in this file apply only to packages that follow Azure SDK Design Guidelines. Names of such packages usually start with Azure
.
To modify the retry options use the Retry
property of client options class.
Be default clients are setup to retry 3 times with exponential retry kind and initial delay of 0.8 sec.
SecretClientOptions options = new SecretClientOptions()
{
Retry =
{
Delay = TimeSpan.FromSeconds(2),
MaxRetries = 10,
Mode = RetryMode.Fixed
}
};
Using RetryOptions
to configure retry behavior is sufficient for the vast majority of scenarios. For more advanced scenarios, it is possible to create a custom retry policy and set it to the RetryPolicy
property of client options class. This can be accomplished by implementing a retry policy that derives from the abstract RetryPolicy
class. The RetryPolicy
class contains hooks to determine if a request should be retried and how long to wait before retrying. In the following example, we implement a policy that will prevent retries from taking place if the overall processing time has exceeded some threshold. Notice that the policy takes in RetryOptions
as one of the constructor parameters and passes it to the base constructor. By doing this, we are able to delegate to the base RetryPolicy
as needed (either by explicitly invoking the base methods, or by not overriding methods that we do not need to customize) which will respect the RetryOptions
.
internal class GlobalTimeoutRetryPolicy : RetryPolicy
{
private readonly TimeSpan _timeout;
public GlobalTimeoutRetryPolicy(RetryOptions options, TimeSpan timeout) : base(options)
{
_timeout = timeout;
}
protected internal override bool ShouldRetry(HttpMessage message, Exception exception)
{
return ShouldRetryInternalAsync(message, exception, false).EnsureCompleted();
}
protected internal override ValueTask<bool> ShouldRetryAsync(HttpMessage message, Exception exception)
{
return ShouldRetryInternalAsync(message, exception, true);
}
private ValueTask<bool> ShouldRetryInternalAsync(HttpMessage message, Exception exception, bool async)
{
TimeSpan elapsedTime = message.ProcessingContext.StartTime - DateTimeOffset.UtcNow;
if (elapsedTime > _timeout)
{
return new ValueTask<bool>(false);
}
return async ? base.ShouldRetryAsync(message, exception) : new ValueTask<bool>(base.ShouldRetry(message, exception));
}
}
Here is how we would configure the client to use the policy we just created.
var retryOptions = new RetryOptions
{
Delay = TimeSpan.FromSeconds(2),
MaxRetries = 10,
Mode = RetryMode.Fixed
};
SecretClientOptions options = new SecretClientOptions()
{
RetryPolicy = new GlobalTimeoutRetryPolicy(retryOptions, timeout: TimeSpan.FromSeconds(30))
};
It's also possible to have full control over the retry logic by setting the RetryPolicy
property to an implementation of HttpPipelinePolicy
where you would need to implement the retry loop yourself. One use case for this is if you want to implement your own retry policy with Polly. Note that if you replace the RetryPolicy
with a HttpPipelinePolicy
, you will need to make sure to update the HttpMessage.ProcessingContext
that other pipeline policies may be relying on.
internal class PollyPolicy : HttpPipelinePolicy
{
public override void Process(HttpMessage message, ReadOnlyMemory<HttpPipelinePolicy> pipeline)
{
Policy.Handle<IOException>()
.Or<RequestFailedException>(ex => ex.Status == 0)
.OrResult<Response>(r => r.Status >= 400)
.WaitAndRetry(
new[]
{
// some custom retry delay pattern
TimeSpan.FromSeconds(1),
TimeSpan.FromSeconds(2),
TimeSpan.FromSeconds(3)
},
onRetry: (result, _) =>
{
// Since we are overriding the RetryPolicy, it is our responsibility to increment the RetryNumber
// that other policies in the pipeline may be depending on.
var context = message.ProcessingContext;
context.RetryNumber++;
}
)
.Execute(() =>
{
ProcessNext(message, pipeline);
return message.Response;
});
}
public override ValueTask ProcessAsync(HttpMessage message, ReadOnlyMemory<HttpPipelinePolicy> pipeline)
{
// async version omitted for brevity
throw new NotImplementedException();
}
}
To set the policy, use the RetryPolicy
property of client options class.
SecretClientOptions options = new SecretClientOptions()
{
RetryPolicy = new PollyPolicy()
};
A note to library authors: Library-specific response classifiers will be respected if a user sets a custom policy deriving from
RetryPolicy
as long as they call into the baseShouldRetry
method. If a user doesn't call the base method, or sets aHttpPipelinePolicy
in theRetryPolicy
property, then the library-specific response classifiers will not be respected.
using HttpClient client = new HttpClient();
SecretClientOptions options = new SecretClientOptions
{
Transport = new HttpClientTransport(client)
};
using HttpClientHandler handler = new HttpClientHandler()
{
Proxy = new WebProxy(new Uri("http://example.com"))
};
SecretClientOptions options = new SecretClientOptions
{
Transport = new HttpClientTransport(handler)
};
You can also configure a proxy using the following environment variables:
HTTP_PROXY
: the proxy server used on HTTP requests.HTTPS_PROXY
: the proxy server used on HTTPS requests.ALL_PROXY
: the proxy server used on HTTP and HTTPS requests in caseHTTP_PROXY
orHTTPS_PROXY
are not defined.NO_PROXY
: a comma-separated list of hostnames that should be excluded from proxying.
Warning: setting these environment variables will affect every new client created within the current process.