在 .NET Core 中结合 HttpClientFactory 使用 Polly(下篇)
下面主要讲几个Polly和HttpClientFactory在ASP.NET Core中结合使用的用例。
用例:应用超时策略
HttpClient已经有了一个Timeout属性,但是在使用重试策略时该如何应用呢?Polly的超时策略又适用于什么地方?
HttpClient.Timeout属性设置的超时将被应用于HttpClient实例的所有调用,包括重试之间的所有尝试和等待。要在每次重试中使用超时,就要在Polly的超时策略之前配置重试策略。
在这种情况下,你可能希望重试策略在每次单个超时时重试。为此,需要让重试策略处理超时策略抛出的TimeoutRejectedException异常。
下面这个示例使用了上篇提到的Polly.Extensions.Http这个包,它可以很方便地为Http错误(比如HttpRequestException、Http 5XX和Http 408等)添加额外的处理。
using Polly.Extensions.Http;
var retryPolicy=HttpPolicyExtensions
.HandleTransientHttpError()
.Or
.WaitAndRetryAsync(new[]
{
TimeSpan.FromSeconds(1),
TimeSpan.FromSeconds(5),
TimeSpan.FromSeconds(10)
});
// 为每个重试定义超时策略
var timeoutPolicy=Policy.TimeoutAsync
serviceCollection.AddHttpClient("GitHub", client=>
{
client.BaseAddress=new Uri("api.github/");
client.DefaultRequestHeaders.Add("Accept", "application/vnd.github.v3+json");
client.Timeout=TimeSpan.FromSeconds(60); // 默认超时时间
})
.AddPolicyHandler(retryPolicy)
// 将超时策略放在重试策略之内,每次重试会应用此超时策略
.AddPolicyHandler(timeoutPolicy);用例:缓存策略
Polly 的缓存策略可以在通过IHttpClientFactory配置的委托处理程序中使用。Polly是通用的(不与Http请求绑定),因此在编写代码时,Polly缓存策略从Polly.Context中确定要使用的缓存键。可以通过HttpRequestMessage请求上的一个扩展方法来设置这个参数:
request.SetPolicyExecutionContext(new Polly.Context("CacheKeyToUseWithThisRequest"));
由于Polly缓存策略是在HttpResponseMessage级别的委托代理服务上进行缓存,因此还需要考虑下面的问题。
HttpResponseMessage级别的缓存是否合适?
如果你想重用HttpResponseMessage,那么在HttpResponseMessage级别上进行缓存可能非常合适。
但在某些情况下,比如调用WebService来获取一些序列化数据,然后反序列化到应用程序中的本地类型,HttpResponseMessage可能不是缓存的最佳粒度。
在这些情况下,HttpResponseMessage级别上的缓存意味着每次命中缓存都会重复读取数据流和反序列化,这在性能方面是不必要的。
在更高级别缓存可能更合适——例如,缓存流或反序列化到应用程序的本地类型的结果。
缓存HttpResponseMessage还要考虑以下三点:
HttpResponseMessage可以包含HttpContent,它只能向前读取流(只能读取一次)。这可能意味着,当CachePolicy第二次从缓存中检索它时,除非重新初始化流指针,否则无法重新读取流。考虑去个性化和时间戳。缓存的个人特有信息和时间戳可能不适合重新提供给后续的请求。注意只缓存状态码为200(OK)的响应。考虑使用Response.EnsureSuccessStatusCode()等方法确保只有成功的响应才能传递给缓存策略。或者你可以使用这里(t/Ehnr78P)描述的自定义ITtlStrategy。用例:在手游购买策略执行和调用之间交换信息
Polly策略的每次执行都会携带Polly.Context类的一个执行域实例(execution-scoped instance),该类的作用是提供上下文,并允许在执行前、执行中和执行后阶段之间交换信息(译注:类似于HttpContext)。
对于通过HttpClientFactory和Polly配置的HttpClient,可以在执行之前使用扩展方法
HttpRequestMessage.SetPolicyExecutionContext(context)来设置被用于Http调用的上下文Polly.Context。该上下文具有字典语义,允许您传递任意数据。
var context=new Polly.Context();
context["MyCustomData"]=foo;
HttpRequestMessage request=new HttpRequestMessage(HttpMethod.Get, requestUri);
request.SetPolicyExecutionContext(context);
var response=await client.SendAsync(request, cancellationToken);
Polly将该上下文实例作为输入参数传递给策略上配置的任何委托钩子(例如onRetry)。例如下面这个已经预先配置了策略的HttpClient:
var retryPolicy=HttpPolicyExtensions
.HandleTransientHttpError()
.WaitAndRetryAsync(new[]
{
TimeSpan.FromSeconds(1),
TimeSpan.FromSeconds(5),
TimeSpan.FromSeconds(10)
},
onRetryAsync: async (outcome, timespan, retryCount, ctx)=> {
/* Do something with ctx["MyCustomData"] */
// ...
});
委托钩子可以在执行期间设置其上下文信息:
var retryPolicy=HttpPolicyExtensions
.HandleTransientHttpError()
.WaitAndRetryAsync(new[]
{
TimeSpan.FromSeconds(1),
TimeSpan.FromSeconds(5),
TimeSpan.FromSeconds(10)
},
onRetryAsync: async (outcome, timespan, retryCount, ctx)=> {
ctx["RetriesInvoked"]=retryCount;
// ...
});
这些信息可以在执行后从上下文中读取:
var response=await client.SendAsync(request, cancellationToken);
var context=response.RequestMessage?.GetPolicyExecutionContext(); // 如果还没有保存在局部变量中
if (context?.TryGetValue("RetriesInvoked", out int? retriesNeeded) false)
{
// Do something with int? retriesNeeded
}
注意,只有在执行之前使用
HttpRequestMessage.SetPolicyExecutionContext(context)设置了上下文时,
HttpRequestMessage.GetPolicyExecutionContext()的获得的上下文才可用。
推荐阅读
-
在.NET Core 3.0中的WPF中使用IOC图文教程
-
在.net Core 中像以前那样的使用HttpContext.Current
-
在ASP.NET Core中创建内部使用Scoped服务的Quartz.NET宿主服务
-
[.Net Core] 在 Mvc 中简单使用日志组件
-
ASP.NET Core 2.1 中的 HttpClientFactory (Part 3) 使用Handler实现传出请求中间件
-
jwt-在asp.net core中的使用jwt
-
asp.net core 使用HttpClientFactory Polly实现熔断降级
-
使用Rotativa在ASP.NET Core MVC中创建PDF
-
如何使用Rotativa在ASP.NET Core MVC中创建PDF详解
-
在ASP.NET Core 3.0中使用Swagger