进阶功能
进阶功能
Uri 拼接规则
所有的 Uri 拼接都是通过 new Uri(Uri baseUri, Uri relativeUri) 这个构造器生成。
带/
结尾的 baseUri
http://a.com/
+b/c/d
=http://a.com/b/c/d
http://a.com/path1/
+b/c/d
=http://a.com/path1/b/c/d
http://a.com/path1/path2/
+b/c/d
=http://a.com/path1/path2/b/c/d
不带/
结尾的 baseUri
http://a.com
+b/c/d
=http://a.com/b/c/d
http://a.com/path1
+b/c/d
=http://a.com/b/c/d
http://a.com/path1/path2
+b/c/d
=http://a.com/path1/b/c/d
事实上http://a.com
与http://a.com/
是完全一样的,他们的 path 都是/
,所以才会表现一样。为了避免低级错误的出现,请使用的标准 baseUri 书写方式,即使用/
作为 baseUri 的结尾的第一种方式。
请求异常处理
请求一个接口,不管出现何种异常,最终都抛出 HttpRequestException,HttpRequestException 的内部异常为实际具体异常,之所以设计为内部异常,是为了完好的保存内部异常的堆栈信息。
WebApiClientCore 内部的很多异常都基于 ApiException 这个异常抽象类,也就是很多情况下抛出的异常都是内部异常为某个 ApiException 的 HttpRequestException。
try
{
var datas = await api.GetAsync();
}
catch (HttpRequestException ex) when (ex.InnerException is ApiInvalidConfigException configException)
{
// 请求配置异常
}
catch (HttpRequestException ex) when (ex.InnerException is ApiResponseStatusException statusException)
{
// 响应状态码异常
}
catch (HttpRequestException ex) when (ex.InnerException is ApiException apiException)
{
// 抽象的api异常
}
catch (HttpRequestException ex) when (ex.InnerException is SocketException socketException)
{
// socket连接层异常
}
catch (HttpRequestException ex)
{
// 请求异常
}
catch (Exception ex)
{
// 异常
}
请求条件性重试
使用ITask<>
异步声明,就有 Retry 的扩展,Retry 的条件可以为捕获到某种 Exception 或响应模型符合某种条件。
public interface IUserApi
{
[HttpGet("api/users/{id}")]
ITask<User> GetAsync(string id);
}
var result = await userApi.GetAsync(id: "id001")
.Retry(maxCount: 3)
.WhenCatch<HttpRequestException>()
.WhenResult(r => r.Age <= 0);
ITask<>
可以精确控制某个方法的重试逻辑,如果想全局性实现重试;请结合使用 Polly 来实现;对于.NET8
以上,可以选择使用 Resilience 来实现。
表单集合处理
按照 OpenApi,一个集合在 Uri 的 Query 或表单中支持 5 种表述方式,分别是:
- Csv // 逗号分隔
- Ssv // 空格分隔
- Tsv // 反斜杠分隔
- Pipes // 竖线分隔
- Multi // 多个同名键的键值对
对于 id = ["001","002"]
这样的数组值,在 PathQueryAttribute 与 FormContentAttribute 处理后分别是:
CollectionFormat | Data |
---|---|
[PathQuery(CollectionFormat = CollectionFormat.Csv)] | id=001,002 |
[PathQuery(CollectionFormat = CollectionFormat.Ssv)] | id=001 002 |
[PathQuery(CollectionFormat = CollectionFormat.Tsv)] | id=001\002 |
[PathQuery(CollectionFormat = CollectionFormat.Pipes)] | id=001|002 |
[PathQuery(CollectionFormat = CollectionFormat.Multi)] | id=001&id=002 |
调整缺省参数特性
WebApiClientCore 是基于元数据来执行请求和处理响应,可以自定义 Api 方法的描述,填充上想要的特性即可。现代 Web 接口中,json 请求几乎占据了大部分的场景,所以你的客户端接口提交的内容往往也是 json 内容,以下 UseJsonFirstApiActionDescriptor 行为在非GET或HEAD请求的缺省参数特性声明时,为复杂参数类型的参数应用 JsonContentAttribute。
services
.AddWebApiClient()
.UseJsonFirstApiActionDescriptor();
你可以为 IUserApi 标注上你的一个自定义 ApiFilterAttribute 子类,用于观察调用 PostAsync 时的 ApiRequestContext.ActionDescriptor.Parameters[0].Attributes,会发现集合里面默认加上了 JsonContentAttribute。
public interface IUserApi
{
[HttpPost("api/users")]
Task<User> PostAsync(/*[JsonContent]*/ User user);
}
适配畸形接口
不友好的参数名别名
例如服务器要求一个 Query 参数的名字为field-Name
,这个是C#
关键字或变量命名不允许的,我们可以使用[AliasAsAttribute]
来达到这个要求:
public interface IUserApi
{
[HttpGet("api/users")]
ITask<string> GetAsync([AliasAs("field-Name")] string fieldName);
}
然后最终请求 uri 变为 api/users/?field-name=fileNameValue
Form 的某个字段为 json 文本
字段 | 值 |
---|---|
field1 | someValue |
field2 | {"name":"sb","age":18} |
field2 对应的 .NET 模型为
public class Field2
{
public string Name {get; set;}
public int Age {get; set;}
}
常规下我们得把 field2 的实例 json 序列化得到 json 文本,然后赋值给 field2 这个 string 属性,使用[JsonFormField]特性可以轻松帮我们自动完成 Field2 类型的 json 序列化并将结果字符串作为表单的一个字段。
public interface IUserApi
{
Task PostAsync([FormField] string field1, [JsonFormField] Field2 field2)
}
Form 的字段多层嵌套
字段 | 值 |
---|---|
field1 | someValue |
field2.name | sb |
field2.age | 18 |
Form 对应的 .NET 模型为
public class FormModel
{
public string Field1 {get; set;}
public Field2 Field2 {get; set;}
}
public class Field2
{
public string Name {get; set;}
public int Age {get; set;}
}
合理情况下,对于复杂嵌套结构的数据模型,应当设计为使用 applicaiton/json 提交 FormModel,但服务提供方设计为使用 x-www-form-urlencoded 来提交 FormModel,我可以配置 KeyValueSerializeOptions 来达到这个格式要求:
services.AddHttpApi<IUserApi>().ConfigureHttpApi(o =>
{
o.KeyValueSerializeOptions.KeyNamingStyle = KeyNamingStyle.FullName;
});
响应的 Content-Type 不是预期值
响应的内容通过肉眼看上是 json 内容,但响应头里的 Content-Type 为非预期值 application/json或 application/xml,而是诸如 text/html 等。这好比客户端提交 json 内容时指示请求头的 Content-Type 值为 text/plain 一样,让服务端无法处理。
解决办法是在 Interface 或 Method 声明[JsonReturn]
特性,并设置其 EnsureMatchAcceptContentType 属性为 false,表示 Content-Type 不是期望值匹配也要处理。
[JsonReturn(EnsureMatchAcceptContentType = false)]
public interface IUserApi
{
}
动态 HttpHost
使用 UriAttribute 传绝对 Uri 参
[LoggingFilter]
public interface IUserApi
{
[HttpGet]
ITask<User> GetAsync([Uri] string urlString, [PathQuery] string id);
}
自定义 HttpHostBaseAttribute 实现
[ServiceNameHost("baidu")] // 使用自定义的ServiceNameHostAttribute
public interface IUserApi
{
[HttpGet("api/users/{id}")]
Task<User> GetAsync(string id);
[HttpPost("api/users")]
Task<User> PostAsync([JsonContent] User user);
}
/// <summary>
/// 以服务名来确定主机的特性
/// </summary>
public class ServiceNameHostAttribute : HttpHostBaseAttribute
{
public string ServiceName { get; }
public ServiceNameHostAttribute(string serviceName)
{
this.ServiceName = serviceName;
}
public override Task OnRequestAsync(ApiRequestContext context)
{
// HostProvider是你自己的服务,数据来源可以是db或其它等等,要求此服务已经注入了DI
HostProvider hostProvider = context.HttpContext.ServiceProvider.GetRequiredService<HostProvider>();
string host = hostProvider.ResolveHost(this.ServiceName);
// 最终目的是设置请求消息的RequestUri的属性
context.HttpContext.RequestMessage.RequestUri = new Uri(host);
}
}
请求签名
动态追加请求签名
例如每个请求的 Uri 额外的动态添加一个叫 sign 的 query 参数,这个 sign 可能和请求参数值有关联,每次都需要计算。 我们可以自定义 ApiFilterAttribute 的子来实现自己的 sign 功能,然后把自定义 Filter 声明到 Interface 或 Method 即可
public class SignFilterAttribute : ApiFilterAttribute
{
public override Task OnRequestAsync(ApiRequestContext context)
{
var signService = context.HttpContext.ServiceProvider.GetRequiredService<SignService>();
var sign = signService.SignValue(DateTime.Now);
context.HttpContext.RequestMessage.AddUrlQuery("sign", sign);
return Task.CompletedTask;
}
}
[SignFilter]
public interface IUserApi
{
...
}
请求表单的字段排序
不知道是哪门公司起的所谓的“签名算法”,往往要表单的字段排序等。
public interface IUserApi
{
[HttpGet("/path")]
Task<HttpResponseMessage> PostAsync([SortedFormContent] Model model);
}
public class SortedFormContentAttribute : FormContentAttribute
{
protected override IEnumerable<KeyValue> SerializeToKeyValues(ApiParameterContext context)
{
这里可以排序、加上其它衍生字段等
return base.SerializeToKeyValues(context).OrderBy(item => item.Key);
}
}
.NET8 AOT 发布
System.Text.Json 中使用源生成功能之后,使项目AOT发布成为可能。
json 序列化源生成示例
[JsonSerializable(typeof(User[]))] // 这里要挂上所有接口中使用到的 json 模型类型
[JsonSerializable(typeof(YourModel[]))]
public partial class AppJsonSerializerContext : JsonSerializerContext
{
}
在 WebApiClientCore 的全局配置中添加 json 源生成的上下文
services
.AddWebApiClient()
.ConfigureHttpApi(options => // json SG生成器配置
{
options.PrependJsonSerializerContext(AppJsonSerializerContext.Default);
});
HttpClient 的配置
这部分是 Httpclient Factory 的内容,这里不做过多介绍。
services.AddHttpApi<IUserApi>().ConfigureHttpClient(httpClient =>
{
httpClient.Timeout = TimeSpan.FromMinutes(1d);
httpClient.DefaultRequestVersion = HttpVersion.Version20;
httpClient.DefaultVersionPolicy = HttpVersionPolicy.RequestVersionOrLower;
});
主 HttpMessageHandler 的配置
Http 代理配置
services.AddHttpApi<IUserApi>().ConfigureHttpApi(o =>
{
o.HttpHost = new Uri("http://localhost:5000/");
})
.ConfigurePrimaryHttpMessageHandler(() => new HttpClientHandler
{
UseProxy = true,
Proxy = new WebProxy
{
Address = new Uri("http://proxy.com"),
Credentials = new NetworkCredential
{
UserName = "useranme",
Password = "pasword"
}
}
});
客户端证书配置
有些服务器为了限制客户端的连接,开启了 https 双向验证,只允许它执有它颁发的证书的客户端进行连接
services.AddHttpApi<IUserApi>().ConfigureHttpApi(o =>
{
o.HttpHost = new Uri("http://localhost:5000/");
})
.ConfigurePrimaryHttpMessageHandler(() =>
{
var handler = new HttpClientHandler();
handler.ClientCertificates.Add(yourCert);
return handler;
});
维持 CookieContainer 不变
如果请求的接口不幸使用了 Cookie 保存身份信息机制,那么就要考虑维持 CookieContainer 实例不要跟随 HttpMessageHandler 的生命周期,默认的 HttpMessageHandler 最短只有 2 分钟的生命周期。
var cookieContainer = new CookieContainer();
services.AddHttpApi<IUserApi>().ConfigureHttpApi(o =>
{
o.HttpHost = new Uri("http://localhost:5000/");
})
.ConfigurePrimaryHttpMessageHandler(() =>
{
var handler = new HttpClientHandler();
handler.CookieContainer = cookieContainer;
return handler;
});
在接口配置中使用过滤器
除了能在接口声明中使用 IApiFilterAttribute 子类的特性标注之外,还可以在接口注册时的配置添加 IApiFilter 类型的过滤器,这些过滤器将对整个接口生效,且优先于通过特性标注的 IApiFilterAttribute 类型执行。
services.AddHttpApi<IUserApi>().ConfigureHttpApi(o =>
{
o.GlobalFilters.Add(new UserFiler());
});
public class UserFiler : IApiFilter
{
public Task OnRequestAsync(ApiRequestContext context)
{
throw new System.NotImplementedException();
}
public Task OnResponseAsync(ApiResponseContext context)
{
throw new System.NotImplementedException();
}
}
自定义请求内容与响应内容解析
除了常见的 xml 或 json 响应内容要反序列化为强类型结果模型,你可能会遇到其它的二进制协议响应内容,比如 google 的 ProtoBuf 二进制内容。
自定义请求内容处理特性
public class ProtobufContentAttribute : HttpContentAttribute
{
public string ContentType { get; set; } = "application/x-protobuf";
protected override Task SetHttpContentAsync(ApiParameterContext context)
{
var stream = new MemoryStream();
if (context.ParameterValue != null)
{
Serializer.NonGeneric.Serialize(stream, context.ParameterValue);
stream.Position = 0L;
}
var content = new StreamContent(stream);
content.Headers.ContentType = new MediaTypeHeaderValue(this.ContentType);
context.HttpContext.RequestMessage.Content = content;
return Task.CompletedTask;
}
}
自定义响应内容解析特性
public class ProtobufReturnAttribute : ApiReturnAttribute
{
public ProtobufReturnAttribute(string acceptContentType = "application/x-protobuf")
: base(new MediaTypeWithQualityHeaderValue(acceptContentType))
{
}
public override async Task SetResultAsync(ApiResponseContext context)
{
var stream = await context.HttpContext.ResponseMessage.Content.ReadAsStreamAsync();
context.Result = Serializer.NonGeneric.Deserialize(context.ApiAction.Return.DataType.Type, stream);
}
}
应用相关自定义特性
[ProtobufReturn]
public interface IProtobufApi
{
[HttpPut("/users/{id}")]
Task<User> UpdateAsync([Required, PathQuery] string id, [ProtobufContent] User user);
}
自定义 CookieAuthorizationHandler
对于使用 Cookie 机制的接口,只有在接口请求之后,才知道 Cookie 是否已失效。通过自定义 CookieAuthorizationHandler,可以做在请求某个接口过程中,遇到 Cookie 失效时自动刷新 Cookie 再重试请求接口。
首先,我们需要把登录接口与某它业务接口拆分在不同的接口定义,例如 IUserApi 和 IUserLoginApi
[HttpHost("http://localhost:5000/")]
public interface IUserLoginApi
{
[HttpPost("/users")]
Task<HttpResponseMessage> LoginAsync([JsonContent] Account account);
}
然后实现自动登录的 CookieAuthorizationHandler
public class AutoRefreshCookieHandler : CookieAuthorizationHandler
{
private readonly IUserLoginApi api;
public AutoRefreshCookieHandler(IUserLoginApi api)
{
this.api = api;
}
/// <summary>
/// 登录并刷新Cookie
/// </summary>
/// <returns>返回登录响应消息</returns>
protected override Task<HttpResponseMessage> RefreshCookieAsync()
{
return this.api.LoginAsync(new Account
{
account = "admin",
password = "123456"
});
}
}
最后,注册 IUserApi、IUserLoginApi,并为 IUserApi 配置 AutoRefreshCookieHandler
services
.AddHttpApi<IUserLoginApi>();
services
.AddHttpApi<IUserApi>()
.AddHttpMessageHandler(s => new AutoRefreshCookieHandler(s.GetRequiredService<IUserLoginApi>()));
现在,调用 IUserApi 的任意接口,只要响应的状态码为 401,就触发 IUserLoginApi 登录,然后将登录得到的 cookie 来重试请求接口,最终响应为正确的结果。你也可以重写 CookieAuthorizationHandler 的 IsUnauthorizedAsync(HttpResponseMessage) 方法来指示响应是未授权状态。
自定义日志输出目标
[CustomLogging]
public interface IUserApi
{
}
public class CustomLoggingAttribute : LoggingFilterAttribute
{
protected override Task WriteLogAsync(ApiResponseContext context, LogMessage logMessage)
{
// 这里把logMessage输出到你的目标
return Task.CompletedTask;
}
}
自定义缓存提供者
默认的缓存提供者为内存缓存,如果希望将缓存保存到其它存储位置,则需要自定义 缓存提者,并注册替换默认的缓存提供者。
public static IWebApiClientBuilder UseRedisResponseCacheProvider(this IWebApiClientBuilder builder)
{
builder.Services.AddSingleton<IResponseCacheProvider, RedisResponseCacheProvider>();
return builder;
}
public class RedisResponseCacheProvider : IResponseCacheProvider
{
public string Name => nameof(RedisResponseCacheProvider);
public Task<ResponseCacheResult> GetAsync(string key)
{
// 从redis获取缓存
throw new NotImplementedException();
}
public Task SetAsync(string key, ResponseCacheEntry entry, TimeSpan expiration)
{
// 把缓存内容写入redis
throw new NotImplementedException();
}
}
自定义自解释的参数类型
在某些极限情况下,比如人脸比对的接口,我们输入模型与传输模型未必是对等的,例如:
服务端要求的 json 模型
{
"image1": "图片1的base64",
"image2": "图片2的base64"
}
客户端期望的业务模型
public class FaceModel
{
public Bitmap Image1 {get; set;}
public Bitmap Image2 {get; set;}
}
我们希望构造模型实例时传入 Bitmap 对象,但传输的时候变成 Bitmap 的 base64 值,所以我们要改造 FaceModel,让它实现 IApiParameter 接口:
public class FaceModel : IApiParameter
{
public Bitmap Image1 { get; set; }
public Bitmap Image2 { get; set; }
public Task OnRequestAsync(ApiParameterContext context)
{
var image1 = GetImageBase64(this.Image1);
var image2 = GetImageBase64(this.Image2);
var model = new { image1, image2 };
var options = context.HttpContext.HttpApiOptions.JsonSerializeOptions;
context.HttpContext.RequestMessage.Content = new JsonContent(model,options);
}
private static string GetImageBase64(Bitmap image)
{
using var stream = new MemoryStream();
image.Save(stream, System.Drawing.Imaging.ImageFormat.Jpeg);
return Convert.ToBase64String(stream.ToArray());
}
}
最后,我们在使用改进后的 FaceModel 来请求
public interface IFaceApi
{
[HttpPost("/somePath")]
Task<HttpResponseMessage> PostAsync(FaceModel faces);
}