如果发生异常,Polly框架CircuitBreakerAsync不会重试

Dem*_*ter 5 c# circuit-breaker async-await polly

我正在使用Polly框架进行瞬态故障处理.对于同步操作,Polly断路器策略工作正常,但是当我创建其异步版本时,它不会重试执行.请建议:

异步方法:

private async static Task HelloWorld()
    {
        if (DateTime.Now < programStartTime.AddSeconds(10))
        {
            Console.WriteLine("Task Failed.");
            throw new TimeoutException();
        }
        await Task.Delay(TimeSpan.FromSeconds(1));
        Console.WriteLine("Task Completed.");
    }
Run Code Online (Sandbox Code Playgroud)

Polly断路器异步策略:

private static void AsyncDemo3(Func<Task> action)
    {
        programStartTime = DateTime.Now;

        Policy policy = Policy
            .Handle<TimeoutException>()
            .CircuitBreakerAsync(3, TimeSpan.FromSeconds(2));
        try
        {
            var a = policy.ExecuteAndCaptureAsync(action, true).GetAwaiter().GetResult();
        }
        catch (AggregateException ex)
        {
            Console.WriteLine("Exception: " + ex.Message);
        }
        catch (Exception ex)
        {
            Console.WriteLine("Exception: " + ex.Message);
        }
    }
Run Code Online (Sandbox Code Playgroud)

执行Polly断路器政策:

AsyncDemo3(的HelloWorld);

请帮助找到并解决问题.

svi*_*ick 13

我相信你误解了断路器政策的作用.

它的作用是,如果给它调用给定次数并且每次都失败,那么它将停止调用给定方法一段时间.但它本身并没有重试.

因此,要做我认为你想做的事情,你需要将重试策略与断路器策略结合起来.一种方法是:

Policy retryPolicy = Policy.Handle<TimeoutException>().RetryAsync(3);

Policy circuitBreakerPolicy = Policy
    .Handle<TimeoutException>()
    .CircuitBreakerAsync(3, TimeSpan.FromSeconds(2));

try
{
    retryPolicy.ExecuteAsync(() => circuitBreakerPolicy.ExecuteAsync(action, true))
        .GetAwaiter().GetResult();
}
…
Run Code Online (Sandbox Code Playgroud)

此代码的输出是:

Task Failed.
Task Failed.
Task Failed.
Exception: The circuit is now open and is not allowing calls.
Run Code Online (Sandbox Code Playgroud)

  • Polly现在还添加了[PolicyWrap](https://github.com/App-vNext/Polly/wiki/PolicyWrap),它可以使组合策略的语法更加简洁:`retryPolicy.WrapAsync(circuitBreakerPolicy).ExecuteAsync(. ..)` (3认同)