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