Check string content of response before retrying with Polly

后端 未结 1 1651
既然无缘
既然无缘 2021-01-02 17:16

I\'m working with a very flaky API. Sometimes I get 500 Server Error with Timeout, some other time I also get 500 Server Error because

相关标签:
1条回答
  • 2021-01-02 17:26

    In general, you can configure Polly policies to respond to the results of an execution (not just an exception), for example check an HttpResponseMessage.StatusCode with a predicate. Examples here in the Polly readme.

    There is not however an in-built way to configure a single Polly policy to respond additionally to the content of the response message. This is because (as your example shows) obtaining that content requires a second async call, which may itself raise network errors.

    This tl;dr engenders complications about how to express (in a simple syntax) a single policy which manages two different async steps with potentially different error handling for each step. Prior related discussion on Polly github: comment welcome.

    As such, where a sequence requires two separate async calls, the Polly team currently recommends expressing this as two separate policies, similar to the example in the end of this answer.


    The particular example in your question may not work because the onRetryAsync delegate (throwing FlakyApiException) is not itself guarded by the policy. A policy only guards the execution of delegates executed through .Execute/ExecuteAsync(...).


    One approach could be to use two policies, a retry policy which retries all typical http exceptions and status codes including 500s; then inside that a Polly FallbackPolicy which traps the status code 500 representing SqlDateTime overflow, and excludes that from being retried by rethrowing as some distinguishing exception (CustomSqlDateOverflowException).

            IAsyncPolicy<HttpResponseMessage> rejectSqlError = Policy<HttpResponseMessage>
                .HandleResult(r => r.StatusCode == HttpStatusCode.InternalServerError)
                .FallbackAsync(async (delegateOutcome, context, token) =>
                {
                    String stringContent = await delegateOutcome.Result.Content.ReadAsStringAsync(); // Could wrap this line in an additional policy as desired.
                    if (delegateOutcome.Result.StatusCode == HttpStatusCode.InternalServerError && stringContent.Contains("SqlDateTime overflow"))
                    {
                        throw new CustomSqlDateOverflowException(); // Replace 500 SqlDateTime overflow with something else.
                    }
                    else
                    {
                        return delegateOutcome.Result; // render all other 500s as they were
                    }
                }, async (delegateOutcome, context) => { /* log (if desired) that InternalServerError was checked for what kind */ });
    
            IAsyncPolicy<HttpResponseMessage> retryPolicy = Policy<HttpResponseMessage>
                .Handle<HttpRequestException>()
                .OrResult(r => r.StatusCode == HttpStatusCode.InternalServerError)
                .OrResult(r => /* condition for any other errors you want to handle */)
                .WaitAndRetry(5, retryAttempt => TimeSpan.FromSeconds(Math.Pow(2, retryAttempt)),
                    async (exception, timeSpan, context) =>
                    {
                        /* log (if desired) retry being invoked */
                    });
    
            HttpResponseMessage response = await retryPolicy.WrapAsync(rejectSqlError)
                .ExecuteAsync(() => client.PostAsync(requestUri, new StringContent(serialisedParameters, Encoding.UTF8, "application/json"), cancellationToken));
    
    0 讨论(0)
提交回复
热议问题