HttpClientService.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  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 static 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. Init(baseAddress);
  40. }
  41. private void Init(string baseAddress)
  42. {
  43. if (_clientFactory is not null)
  44. {
  45. return;
  46. }
  47. IServiceCollection _services = new ServiceCollection();
  48. _services.AddHttpClient("Default", c =>
  49. {
  50. if (!string.IsNullOrEmpty(baseAddress))
  51. {
  52. c.BaseAddress = new Uri(baseAddress);
  53. }
  54. c.Timeout = TimeSpan.FromSeconds(_timeout);
  55. c.DefaultRequestHeaders.Add("Cache-Control", "no-cache");
  56. })
  57. .AddTypedClient<HttpClient>().SetHandlerLifetime(TimeSpan.FromMinutes(_handlerLifetime)).ConfigurePrimaryHttpMessageHandler((h =>
  58. {
  59. return new HttpClientHandler
  60. {
  61. MaxConnectionsPerServer = _maxConnectionsPerServer
  62. };
  63. }));
  64. _clientFactory = _services.BuildServiceProvider()
  65. .GetRequiredService<IHttpClientFactory>();
  66. }
  67. public virtual async Task<HttpResponse> PostJsonAsync(string Url, string bodyData, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  68. {
  69. HttpResponse result = new HttpResponse() { IsError = false };
  70. try
  71. {
  72. var client = _clientFactory.CreateClient(clientName);
  73. if (!string.IsNullOrEmpty(authorizationToken))
  74. {
  75. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  76. }
  77. if (headers != null)
  78. {
  79. for (int idx = 0; idx < headers.Count; idx++)
  80. {
  81. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  82. }
  83. }
  84. HttpContent content = new StringContent(bodyData);
  85. content.Headers.ContentType = new MediaTypeWithQualityHeaderValue("application/json");
  86. content.Headers.ContentType.CharSet = "UTF-8";
  87. var response = await client.PostAsync(Url, content).ConfigureAwait(false);
  88. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  89. result.Headers = response.Headers;
  90. result.RequestMessage = response.RequestMessage;
  91. result.StatusCode = response.StatusCode;
  92. result.Response = await response.Content.ReadAsStringAsync();
  93. }
  94. catch (Exception ex)
  95. {
  96. result.IsError = true;
  97. result.Exception = ex;
  98. }
  99. return result;
  100. }
  101. public virtual async Task<HttpResponse> GetJsonAsync(string Url, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  102. {
  103. HttpResponse result = new HttpResponse() { IsError = false };
  104. try
  105. {
  106. var client = _clientFactory.CreateClient(clientName);
  107. if (!string.IsNullOrEmpty(authorizationToken))
  108. {
  109. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  110. }
  111. if (headers != null)
  112. {
  113. for (int idx = 0; idx < headers.Count; idx++)
  114. {
  115. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  116. }
  117. }
  118. // client.DefaultRequestHeaders.Add("Content-Type", "application/json");
  119. var response = await client.GetAsync(Url).ConfigureAwait(false);
  120. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  121. result.Headers = response.Headers;
  122. result.RequestMessage = response.RequestMessage;
  123. result.StatusCode = response.StatusCode;
  124. result.Response = await response.Content.ReadAsStringAsync();
  125. }
  126. catch (Exception ex)
  127. {
  128. result.IsError = true;
  129. result.Exception = ex;
  130. }
  131. return result;
  132. }
  133. public virtual async Task<HttpResponse> PutJsonAsync(string Url, string bodyData, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  134. {
  135. HttpResponse result = new HttpResponse() { IsError = false };
  136. try
  137. {
  138. var client = _clientFactory.CreateClient(clientName);
  139. if (!string.IsNullOrEmpty(authorizationToken))
  140. {
  141. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  142. }
  143. if (headers != null)
  144. {
  145. for (int idx = 0; idx < headers.Count; idx++)
  146. {
  147. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  148. }
  149. }
  150. HttpContent content = new StringContent(bodyData);
  151. content.Headers.ContentType = new MediaTypeWithQualityHeaderValue("application/json");
  152. content.Headers.ContentType.CharSet = "UTF-8";
  153. var response = await client.PutAsync(Url, content).ConfigureAwait(false);
  154. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  155. result.Headers = response.Headers;
  156. result.RequestMessage = response.RequestMessage;
  157. result.StatusCode = response.StatusCode;
  158. result.Response = await response.Content.ReadAsStringAsync();
  159. }
  160. catch (Exception ex)
  161. {
  162. result.IsError = true;
  163. result.Exception = ex;
  164. }
  165. return result;
  166. }
  167. public virtual async Task<HttpResponse> DeleteJsonAsync(string Url, Dictionary<string, string> headers, string clientName = "Default", bool bearerToken = false, string authorizationToken = null)
  168. {
  169. HttpResponse result = new HttpResponse() { IsError = false };
  170. try
  171. {
  172. var client = _clientFactory.CreateClient(clientName);
  173. if (!string.IsNullOrEmpty(authorizationToken))
  174. {
  175. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  176. }
  177. if (headers != null)
  178. {
  179. for (int idx = 0; idx < headers.Count; idx++)
  180. {
  181. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  182. }
  183. }
  184. // client.DefaultRequestHeaders.Add("Content-Type", "application/json");
  185. var response = await client.DeleteAsync(Url).ConfigureAwait(false);
  186. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  187. result.Headers = response.Headers;
  188. result.RequestMessage = response.RequestMessage;
  189. result.StatusCode = response.StatusCode;
  190. result.Response = await response.Content.ReadAsStringAsync();
  191. }
  192. catch (Exception ex)
  193. {
  194. result.IsError = true;
  195. result.Exception = ex;
  196. }
  197. return result;
  198. }
  199. 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)
  200. {
  201. HttpResponse result = new HttpResponse() { IsError = false };
  202. try
  203. {
  204. var client = _clientFactory.CreateClient(clientName);
  205. ServicePointManager.ServerCertificateValidationCallback += (sender, cert, chain, sslPolicyErrors) => true;
  206. if (!string.IsNullOrEmpty(authorizationToken))
  207. {
  208. client.DefaultRequestHeaders.Authorization = bearerToken ? new AuthenticationHeaderValue("Bearer", authorizationToken) : new AuthenticationHeaderValue(authorizationToken);
  209. }
  210. if (headers != null)
  211. {
  212. for (int idx = 0; idx < headers.Count; idx++)
  213. {
  214. client.DefaultRequestHeaders.Add(headers.ElementAt(idx).Key, headers.ElementAt(idx).Value);
  215. }
  216. }
  217. var content = new MultipartFormDataContent();
  218. foreach (var keyValuePair in bodyData)
  219. {
  220. content.Add(new StringContent(keyValuePair.Value), "\"" + keyValuePair.Key + "\"");
  221. }
  222. var response = await client.PostAsync(Url, content).ConfigureAwait(false);
  223. result.IsSuccessStatusCode = response.IsSuccessStatusCode;
  224. result.Headers = response.Headers;
  225. result.RequestMessage = response.RequestMessage;
  226. result.StatusCode = response.StatusCode;
  227. result.Response = await response.Content.ReadAsStringAsync();
  228. }
  229. catch (Exception ex)
  230. {
  231. result.IsError = true;
  232. result.Exception = ex;
  233. }
  234. return result;
  235. }
  236. }
  237. public class HttpResponse
  238. {
  239. public bool IsError { internal set; get; }
  240. public Exception Exception { internal set; get; }
  241. public string Response { internal set; get; }
  242. /// <summary>
  243. /// 摘要:
  244. /// 取得或設定 HTTP 回應的狀態碼。
  245. /// 傳回:
  246. /// HTTP 回應的狀態碼。
  247. /// </summary>
  248. public HttpStatusCode StatusCode { get; internal set; }
  249. /// <summary>
  250. /// 摘要:
  251. /// 取得 HTTP 回應標頭的集合。
  252. /// 傳回:
  253. /// HTTP 回應標頭的集合。
  254. /// </summary>
  255. public HttpResponseHeaders Headers { get; internal set; }
  256. /// <summary>
  257. /// 摘要:
  258. /// 取得或設定導致此回應訊息的要求訊息。
  259. /// 傳回:
  260. /// 導致此回應訊息的要求訊息。
  261. /// </summary>
  262. public HttpRequestMessage RequestMessage { get; internal set; }
  263. /// <summary>
  264. /// 摘要:
  265. /// 取得指示 HTTP 回應是否成功的值。
  266. /// 傳回:
  267. /// 指示 HTTP 回應是否成功的值。 如果 System.Net.Http.HttpResponseMessage.StatusCode 在 200-299
  268. /// 的範圍內,則為 true;否則為 false。
  269. /// </summary>
  270. public bool IsSuccessStatusCode { get; internal set; }
  271. }
  272. }