HttpClientService.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  1. using Microsoft.Extensions.DependencyInjection;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Net;
  6. using System.Net.Http;
  7. using System.Net.Http.Headers;
  8. using System.Threading.Tasks;
  9. namespace EVCB_OCPP.WSServer.Service
  10. {
  11. public class HttpClientService
  12. {
  13. /// <summary>
  14. /// 要求逾時前等候的時間長度
  15. /// #網域名稱系統(DNS)查詢最多可能需要15秒的時間才會傳回或超時
  16. /// 預設60秒
  17. /// </summary>
  18. public int Timeout { get => _timeout; set => _timeout = value; }
  19. /// <summary>
  20. /// 取得或設定使用 HttpClient 物件提出要求時,所允許的同時連線 數目上限 (每個伺服器端點)。
  21. /// 請注意,此限制是按照每個伺服器端點計算,例如值 256 允許 http://www.adatum.com/ 使用 256 個同時連線,
  22. /// 而 http://www.adventure-works.com/ 另有 256 個同時連線。
  23. /// 預設100個
  24. /// </summary>
  25. public int MaxConnectionsPerServer { get => _maxConnectionsPerServer; set => _maxConnectionsPerServer = value; }
  26. /// <summary>
  27. /// 設定HttpMessageHandler的生命週期,如果其存留期間尚未過期,HttpMessageHandler 執行個體可從集區重複使用(建立新的 HttpClient 執行個體時)
  28. /// 因為處理常式通常會管理自己專屬的底層 HTTP 連線。 建立比所需數目更多的處理常式,可能會導致連線延遲。 有些處理常式也會保持連線無限期地開啟,這可能導致處理常式無法回應 DNS (網域名稱系統)變更。
  29. /// 預設處理常式存留時間為120秒。
  30. /// </summary>
  31. public int HandlerLifetime { get => _handlerLifetime; set => _handlerLifetime = value; }
  32. private IHttpClientFactory _clientFactory = null;
  33. private IServiceCollection _services = new ServiceCollection();
  34. private int _handlerLifetime = 2;
  35. private int _maxConnectionsPerServer = 300;
  36. private int _timeout = 60;
  37. public HttpClientService(string baseAddress = "")
  38. {
  39. _services.AddHttpClient("Default", c =>
  40. {
  41. if (!string.IsNullOrEmpty(baseAddress))
  42. {
  43. c.BaseAddress = new Uri(baseAddress);
  44. }
  45. c.Timeout = TimeSpan.FromSeconds(_timeout);
  46. c.DefaultRequestHeaders.Add("Cache-Control", "no-cache");
  47. })
  48. .AddTypedClient<HttpClient>().SetHandlerLifetime(TimeSpan.FromMinutes(_handlerLifetime)).ConfigurePrimaryHttpMessageHandler((h =>
  49. {
  50. return new HttpClientHandler
  51. {
  52. MaxConnectionsPerServer = _maxConnectionsPerServer
  53. };
  54. }));
  55. Init();
  56. }
  57. private void Init()
  58. {
  59. _clientFactory = _services.BuildServiceProvider()
  60. .GetRequiredService<IHttpClientFactory>();
  61. }
  62. public virtual async Task<HttpResponse> PostJsonAsync(string Url, string bodyData, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  63. {
  64. HttpResponse result = new HttpResponse() { IsError = false };
  65. try
  66. {
  67. var client = _clientFactory.CreateClient(clientName);
  68. if (!string.IsNullOrEmpty(authorizationToken))
  69. {
  70. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  71. }
  72. if (headers != null)
  73. {
  74. for (int idx = 0; idx < headers.Count; idx++)
  75. {
  76. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  77. }
  78. }
  79. HttpContent content = new StringContent(bodyData);
  80. content.Headers.ContentType = new MediaTypeWithQualityHeaderValue("application/json");
  81. content.Headers.ContentType.CharSet = "UTF-8";
  82. var response = await client.PostAsync(Url, content).ConfigureAwait(false);
  83. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  84. result.Headers = response.Headers;
  85. result.RequestMessage = response.RequestMessage;
  86. result.StatusCode = response.StatusCode;
  87. result.Response = await response.Content.ReadAsStringAsync();
  88. }
  89. catch (Exception ex)
  90. {
  91. result.IsError = true;
  92. result.Exception = ex;
  93. }
  94. return result;
  95. }
  96. public virtual async Task<HttpResponse> GetJsonAsync(string Url, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  97. {
  98. HttpResponse result = new HttpResponse() { IsError = false };
  99. try
  100. {
  101. var client = _clientFactory.CreateClient(clientName);
  102. if (!string.IsNullOrEmpty(authorizationToken))
  103. {
  104. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  105. }
  106. if (headers != null)
  107. {
  108. for (int idx = 0; idx < headers.Count; idx++)
  109. {
  110. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  111. }
  112. }
  113. // client.DefaultRequestHeaders.Add("Content-Type", "application/json");
  114. var response = await client.GetAsync(Url).ConfigureAwait(false);
  115. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  116. result.Headers = response.Headers;
  117. result.RequestMessage = response.RequestMessage;
  118. result.StatusCode = response.StatusCode;
  119. result.Response = await response.Content.ReadAsStringAsync();
  120. }
  121. catch (Exception ex)
  122. {
  123. result.IsError = true;
  124. result.Exception = ex;
  125. }
  126. return result;
  127. }
  128. public virtual async Task<HttpResponse> PutJsonAsync(string Url, string bodyData, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  129. {
  130. HttpResponse result = new HttpResponse() { IsError = false };
  131. try
  132. {
  133. var client = _clientFactory.CreateClient(clientName);
  134. if (!string.IsNullOrEmpty(authorizationToken))
  135. {
  136. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  137. }
  138. if (headers != null)
  139. {
  140. for (int idx = 0; idx < headers.Count; idx++)
  141. {
  142. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  143. }
  144. }
  145. HttpContent content = new StringContent(bodyData);
  146. content.Headers.ContentType = new MediaTypeWithQualityHeaderValue("application/json");
  147. content.Headers.ContentType.CharSet = "UTF-8";
  148. var response = await client.PutAsync(Url, content).ConfigureAwait(false);
  149. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  150. result.Headers = response.Headers;
  151. result.RequestMessage = response.RequestMessage;
  152. result.StatusCode = response.StatusCode;
  153. result.Response = await response.Content.ReadAsStringAsync();
  154. }
  155. catch (Exception ex)
  156. {
  157. result.IsError = true;
  158. result.Exception = ex;
  159. }
  160. return result;
  161. }
  162. public virtual async Task<HttpResponse> DeleteJsonAsync(string Url, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  163. {
  164. HttpResponse result = new HttpResponse() { IsError = false };
  165. try
  166. {
  167. var client = _clientFactory.CreateClient(clientName);
  168. if (!string.IsNullOrEmpty(authorizationToken))
  169. {
  170. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  171. }
  172. if (headers != null)
  173. {
  174. for (int idx = 0; idx < headers.Count; idx++)
  175. {
  176. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  177. }
  178. }
  179. // client.DefaultRequestHeaders.Add("Content-Type", "application/json");
  180. var response = await client.DeleteAsync(Url).ConfigureAwait(false);
  181. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  182. result.Headers = response.Headers;
  183. result.RequestMessage = response.RequestMessage;
  184. result.StatusCode = response.StatusCode;
  185. result.Response = await response.Content.ReadAsStringAsync();
  186. }
  187. catch (Exception ex)
  188. {
  189. result.IsError = true;
  190. result.Exception = ex;
  191. }
  192. return result;
  193. }
  194. public virtual async Task<HttpResponse> PostFormDataAsync(string Url, Dictionary<string, string> bodyData, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  195. {
  196. HttpResponse result = new HttpResponse() { IsError = false };
  197. try
  198. {
  199. var client = _clientFactory.CreateClient(clientName);
  200. ServicePointManager.ServerCertificateValidationCallback += (sender, cert, chain, sslPolicyErrors) => true;
  201. if (!string.IsNullOrEmpty(authorizationToken))
  202. {
  203. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  204. }
  205. if (headers != null)
  206. {
  207. for (int idx = 0; idx < headers.Count; idx++)
  208. {
  209. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  210. }
  211. }
  212. var content = new MultipartFormDataContent();
  213. foreach (var keyValuePair in bodyData)
  214. {
  215. content.Add(new StringContent(keyValuePair.Value), "\"" + keyValuePair.Key + "\"");
  216. }
  217. var response = await client.PostAsync(Url, content).ConfigureAwait(false);
  218. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  219. result.Headers = response.Headers;
  220. result.RequestMessage = response.RequestMessage;
  221. result.StatusCode = response.StatusCode;
  222. result.Response = await response.Content.ReadAsStringAsync();
  223. }
  224. catch (Exception ex)
  225. {
  226. result.IsError = true;
  227. result.Exception = ex;
  228. }
  229. return result;
  230. }
  231. }
  232. public class HttpResponse
  233. {
  234. public bool IsError { internal set; get; }
  235. public Exception Exception { internal set; get; }
  236. public string Response { internal set; get; }
  237. /// <summary>
  238. /// 摘要:
  239. /// 取得或設定 HTTP 回應的狀態碼。
  240. /// 傳回:
  241. /// HTTP 回應的狀態碼。
  242. /// </summary>
  243. public HttpStatusCode StatusCode { get; internal set; }
  244. /// <summary>
  245. /// 摘要:
  246. /// 取得 HTTP 回應標頭的集合。
  247. /// 傳回:
  248. /// HTTP 回應標頭的集合。
  249. /// </summary>
  250. public HttpResponseHeaders Headers { get; internal set; }
  251. /// <summary>
  252. /// 摘要:
  253. /// 取得或設定導致此回應訊息的要求訊息。
  254. /// 傳回:
  255. /// 導致此回應訊息的要求訊息。
  256. /// </summary>
  257. public HttpRequestMessage RequestMessage { get; internal set; }
  258. /// <summary>
  259. /// 摘要:
  260. /// 取得指示 HTTP 回應是否成功的值。
  261. /// 傳回:
  262. /// 指示 HTTP 回應是否成功的值。 如果 System.Net.Http.HttpResponseMessage.StatusCode 在 200-299
  263. /// 的範圍內,則為 true;否則為 false。
  264. /// </summary>
  265. public bool IsSuccessStatusCode { get; internal set; }
  266. }
  267. }