Using the Polly Timeout when making a Http Request

Full source code available here.

Want to learn more about Polly? Check out my Pluralsight course on it.

When making remote service requests the remote side will sometimes take longer than acceptable to respond. You have a few choices in handling this.

  1. Put up with it, but this means you’re stuck waiting for some unknown period and hope you get a response.
  2. Wait for the local HttpClient to timeout.
  3. Fire off more requests expecting one of them to respond quickly. But now you have multiple requests open holding resources on your side.

Polly offers another approach. The Polly Timeout Policy allows you to specify how long a request should take to respond and if it doesn’t respond in the time period you specify, a cancellation token is used to release held resources. The Timeout policy can be combined with a retry policy to fire off another request as soon as the timeout occurs.

This is great for GET requests, but be careful with anything that has side effects, search for idempotent if you don’t know what I mean.

Code

I define two policies, the first is a simple Retry Policy that is set to retry three times. The other is a Timeout Policy that is set to react if no response is received within one second.

 1readonly RetryPolicy<HttpResponseMessage> _httpRetryPolicy;
 2readonly TimeoutPolicy _timeoutPolicy;
 3
 4public CatalogController()
 5{
 6	_httpRetryPolicy =
 7		Policy.HandleResult<HttpResponseMessage>(r => !r.IsSuccessStatusCode)
 8			.Or<TimeoutRejectedException>()
 9			.RetryAsync(3);
10
11	_timeoutPolicy = Policy.TimeoutAsync(1);
12}

Next I chain (or wrap) these policies around a http request executed by HttpClient.

1HttpResponseMessage response = await 
2	_httpRetryPolicy.ExecuteAsync(() =>
3		_timeoutPolicy.ExecuteAsync(async token => 
4			await httpClient.GetAsync(requestEndpoint, token), CancellationToken.None));

Line 4 makes the request to the remote service using the HttpClient. Line 3, executes the timeout policy, if the http client does NOT respond with 1 second the timeout policy will throw a TimeoutRejectedExcetion. Line 2, the retry policy condition will trigger when a TimeoutRejectedException occurs, and a retry will be performed. Line 1, sets the return value from the _httpRetryPolicy to the HttpResponse.

Execution

  1. The HttpClient calls the remote inventory service which has been written to take 10 seconds to respond for the first three requests, the fourth replies promptly.
  2. For the first three requests, after 1 second no response is received from the remote controller, so the retry policy throws a TimeoutRejectedException and cancels the request by the HttpClient.
  3. The retry policy condition is triggered and the a retry is executed.

This sequence repeats itself two more times.

  1. Finally, on the fourth attempt the request succeeds in less than a second and the a response is returned.

Conclusion

Timeout lets you preempt a slow response and when combined with a retry policy it allows you to send another request quickly.  
Full source code available here.

comments powered by Disqus

Related