C# abp框架Http辅助类
一、定义接口
为什么要定义接口而不直接使用静态类,因为接口可以注入缓存对象,这样就能从缓存中读取指定的请求头
using System; using System.Collections.Generic; using System.Net.Http; using System.Text; using System.Threading.Tasks; using Volo.Abp.Application.Services; namespace Test.Common { /// <summary> /// HTTP请求接口 /// </summary> public interface IHttpClientUtils: IApplicationService { /// <summary> /// 发送HTTP请求,注意Get请求第4个参数才是缓存key,因此第3个参数请设置为null /// </summary> /// <param name="method">HttpMethod.Get、HttpMethod.Post、HttpMethod.Put、HttpMethod.Delete</param> /// <param name="url">完整地址</param> /// <param name="model">请求体对象</param> /// <param name="cacheKey">鉴权请求头缓存key</param> /// <param name="headers">请求头</param> /// <returns></returns> Task<T> SendAsync<T>(HttpMethod method, string url, object model = null, string cacheKey = "", Dictionary<string, string> headers = null); /// <summary> /// 发送HTTP表单请求 /// </summary> /// <param name="method">HttpMethod.Post、HttpMethod.Put</param> /// <param name="url">完整地址</param> /// <param name="model">请求体对象</param> /// <param name="cacheKey">鉴权请求头缓存key</param> /// <param name="headers">请求头</param> /// <returns></returns> Task<T> SendFormAsync<T>(HttpMethod method, string url, object model, string cacheKey = "", Dictionary<string, string> headers = null); } }
二、实现接口
1、支持https
2、支持从缓存读取鉴权请求头
3、支持对象携带文件上传
using Microsoft.Extensions.Logging; using Newtonsoft.Json; using RestSharp; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Net.Http; using System.Net.Http.Headers; using System.Net.Security; using System.Security.Cryptography.X509Certificates; using System.Text; using System.Threading.Tasks; using Volo.Abp.Application.Services; using Volo.Abp.Caching; using Volo.Abp.Json; namespace Test.Common { /// <summary> /// HTTP请求实现 /// </summary> public class HttpClientUtils: ApplicationService, IHttpClientUtils { private readonly ILogger<HttpClientUtils> _logger; private readonly IDistributedCache<Dictionary<string, string>> _cache; public HttpClientUtils( ILogger<HttpClientUtils> logger, IDistributedCache<Dictionary<string, string>> cache) { _cache = cache; _logger = logger; } /// <summary> /// 发送HTTP请求 /// </summary> /// <param name="method">HttpMethod.Get、HttpMethod.Post、HttpMethod.Put、HttpMethod.Delete</param> /// <param name="url">完整地址</param> /// <param name="model">请求体对象</param> /// <param name="cacheKey">鉴权请求头缓存key</param> /// <param name="headers">请求头</param> /// <returns></returns> public async Task<T> SendAsync<T>(HttpMethod method, string url, object model = null, string cacheKey = "", Dictionary<string, string> headers = null) { _logger.LogInformation($"SendAsync {method.Method} url = {url}, cacheKey = {cacheKey}, data = {JsonConvert.SerializeObject(model)}"); try { using (var client = new HttpClient()) { if (!url.StartsWith("http", StringComparison.OrdinalIgnoreCase)) { url = $"http://{url}"; } using (var request = new HttpRequestMessage(method, url)) { if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase)) { SetHttps(); } var sc = new StreamContent(new MemoryStream()); if (model != null) { var jsonStr = JsonConvert.SerializeObject(model); var bytes = Encoding.UTF8.GetBytes(jsonStr); sc = new StreamContent(new MemoryStream(bytes)); } HeaderHandler(request, sc.Headers, "application/json", headers, cacheKey); request.Content = sc; var rsp = await client.SendAsync(request); return ResponseHandler<T>(rsp); } } } catch (Exception e) { _logger.LogError($"SendAsync {method.Method} 请求:{url}, 错误消息:{e.Message}, 请求参数:{JsonConvert.SerializeObject(model)}"); throw new Exception($"请求HTTP服务{new Uri(url).AbsolutePath}异常:{e.Message}"); } } /// <summary> /// 发送HTTP表单请求 /// </summary> /// <param name="method">HttpMethod.Post、HttpMethod.Put</param> /// <param name="url">完整地址</param> /// <param name="model">请求体对象</param> /// <param name="cacheKey">鉴权请求头缓存key</param> /// <param name="headers">请求头</param> /// <returns></returns> public async Task<T> SendFormAsync<T>(HttpMethod method, string url, object model, string cacheKey = "", Dictionary<string, string> headers = null) { _logger.LogInformation($"SendFormAsync {method.Method} url = {url}, cacheKey = {cacheKey}, data = {JsonConvert.SerializeObject(model)}"); try { var client = new RestClient(); var request = new RestRequest(url, Method.POST); if (method == HttpMethod.Put) { request.Method = Method.PUT; } SetHeader(request, cacheKey, headers); request.AlwaysMultipartFormData = true; var props = model.GetType().GetProperties(); foreach (var item in props) { var k = item.Name; var v = item.GetValue(model); if (v == null) { _logger.LogInformation($"SendFormAsync {method.Method} url = {url} {k} value is null"); continue; } if (v.GetType().Name == "Dictionary`2") // 文件只支持字典类型 { if (v is Dictionary<string, byte[]>) { var files = v as Dictionary<string, byte[]>; foreach (var obj in files) // 多个文件只能一个个添加,不能集合到List中 { string ext = Path.GetExtension(obj.Key); ext = ext.RemovePreFix("."); _logger.LogInformation($"SendFormAsync file key = {k}, name = {obj.Key}, ext = {ext}, size = {obj.Value.Length}"); request.AddFileBytes(k, obj.Value, obj.Key, $"image/{ext}"); } } } else { request.AddParameter(k, v); } } var response = await client.ExecuteAsync(request); if (response.StatusCode != HttpStatusCode.OK && response.StatusCode != HttpStatusCode.Created) { _logger.LogError($"SendFormAsync {method.Method} {url} 响应失败: {response.Content}"); throw new Exception($"服务器响应失败:status = {response.StatusCode}"); } _logger.LogInformation($"SendFormAsync {method.Method} {url} Response: {response.Content}"); return JsonConvert.DeserializeObject<T>(response.Content, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Ignore }); } catch (Exception e) { _logger.LogError($"SendFormAsync {method.Method} 请求:{url}, 错误消息:{e.Message}, 请求参数:{JsonConvert.SerializeObject(model)}"); throw new Exception($"请求HTTP服务{new Uri(url).AbsolutePath}异常:{e.Message}"); } } /// <summary> /// 设置请求头 /// </summary> /// <param name="request"></param> /// <param name="cacheKey"></param> private void SetHeader(RestRequest request, string cacheKey, Dictionary<string, string> headers) { // 鉴权请求头 var cacheHeaders = _cache.Get(cacheKey); if (cacheHeaders != null) { foreach (var dic in cacheHeaders) { _logger.LogInformation($"SetHeader cache header: {dic.Key} = {dic.Value}"); if (dic.Key.Equals("authorization", StringComparison.OrdinalIgnoreCase)) { request.AddHeader("Authorization", dic.Value); } else { request.AddHeader(dic.Key, dic.Value); } } } // 默认请求头 if (headers != null && headers.Count > 0) { foreach (var dic in headers) { request.AddHeader(dic.Key, dic.Value); } } } /// <summary> /// 处理请求头 /// </summary> /// <param name="request">new HttpRequestMessage</param> /// <param name="contentHeader">new MultipartFormDataContent</param> /// <param name="contentType">multipart/form-data</param> /// <param name="headers">Dictionary<string, string></param> /// <param name="cacheKey">authorization</param> private void HeaderHandler(HttpRequestMessage request, HttpContentHeaders contentHeader, string contentType, Dictionary<string, string> headers, string cacheKey) { // 添加默认请求头 if (headers != null && headers.ContainsKey("Content-Type")) { contentType = headers["Content-Type"]; } if (string.IsNullOrEmpty(contentType)) { contentType = "application/json"; // 默认application/json } _logger.LogInformation($"HeaderHandler {request.Method.Method} {request.RequestUri.AbsolutePath} default contentType header: {contentType}"); if (contentType.IndexOf("multipart/form-data") != -1) { contentHeader.Remove("Content-Type"); // MediaTypeHeaderValue不支持解析boundary,所以先删除再Add contentHeader.TryAddWithoutValidation("Content-Type", contentType); } else { contentHeader.ContentType = new MediaTypeHeaderValue(contentType); } // 添加鉴权请求头 var cacheHeaders = _cache.Get(cacheKey); if (cacheHeaders != null) { foreach (var dic in cacheHeaders) { _logger.LogInformation($"HeaderHandler {request.Method.Method} {request.RequestUri.AbsolutePath} cache header: {dic.Key} = {dic.Value}"); if (dic.Key.Equals("authorization", StringComparison.OrdinalIgnoreCase)) { request.Headers.Authorization = new AuthenticationHeaderValue(dic.Value); // request.SetBearerToken(dic.Value); // 这个方法里加了前缀Bearer,针对接口不同 } else { contentHeader.Add(dic.Key, dic.Value); // cookie、token ... } } } // 添加参数请求头 if (headers != null && headers.Count > 0) { foreach (var dic in headers) { if (contentHeader.ContentType.MediaType == dic.Value) { continue; } contentHeader.Add(dic.Key, dic.Value); } } } /// <summary> /// 处理HTTP响应 /// </summary> /// <typeparam name="T"></typeparam> /// <param name="rsp"></param> /// <returns></returns> /// <exception cref="Exception"></exception> private T ResponseHandler<T>(HttpResponseMessage rsp) { if (rsp.StatusCode != HttpStatusCode.OK && rsp.StatusCode != HttpStatusCode.Created) { _logger.LogError($"ResponseHandler {rsp.RequestMessage.Method.Method} {rsp.RequestMessage.RequestUri.AbsoluteUri} 响应失败: {rsp.Content.ReadAsStringAsync().Result}"); throw new Exception($"服务器响应失败:status = {rsp.StatusCode}"); } if (rsp is T) { return (T)(object)rsp; // 返回HttpResponseMessage,用于获取响应头 } else { var json = rsp.Content.ReadAsStringAsync().Result; _logger.LogInformation($"ResponseHandler {rsp.RequestMessage.Method.Method} {rsp.RequestMessage.RequestUri.AbsoluteUri} 响应:{json}"); return JsonConvert.DeserializeObject<T>(json, new JsonSerializerSettings { NullValueHandling = NullValueHandling.Ignore }); // return rsp.Content.ReadFromJsonAsync<T>().Result; // 会检查int null类型报错 } } /// <summary> /// 设置HTTPS /// </summary> private void SetHttps() { // set remote certificate Validation auto pass ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(RemoteCertificateValidate); ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls13 | SecurityProtocolType.Tls12 | SecurityProtocolType.Tls11 | SecurityProtocolType.Tls; // FIX:修复不同.Net版对一些SecurityProtocolType枚举支持情况不一致导致编译失败等问题,这里统一使用数值 //ServicePointManager.SecurityProtocol = (SecurityProtocolType)48 | (SecurityProtocolType)3072 | (SecurityProtocolType)768 | (SecurityProtocolType)192; } /// <summary> /// 远程证书验证 /// </summary> /// <param name="sender"></param> /// <param name="cert"></param> /// <param name="chain"></param> /// <param name="error"></param> /// <returns>验证是否通过,始终通过</returns> private bool RemoteCertificateValidate(object sender, X509Certificate cert, X509Chain chain, SslPolicyErrors error) { return true; } } }
三、遇到问题
1、无法添加请求头content-type
var s = new HttpRequestMessage()
s.Headers.Add("Content-Type", "")
报错:Misused header name, 'Content-Type'. Make sure request headers are used with HttpRequestMessage,
var b = new HttpContentHeader()
b.Add("Content-Type", "")
报错:Cannot add value because header 'Content-Type' does not support multiple values.
var b = new HttpContentHeader()
b.ContentType = new MediaTypeHeaderValue("multipart/form-data; boundary=----8db27a5c21ea4f8");
报错:The format of value 'multipart/form-data; boundary=----8db27a5c21ea4f8' is invalid
报错:'multipart/form-data; ----8db27a507065123' is invalid
报错:'multipart/form-data; boundary=--8db27ca4cfafb57' is invalid
1、解决方案:
请看方法:HeaderHandler