HttpClient.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. using System;
  2. using System.Net;
  3. using System.Threading;
  4. using System.Threading.Tasks;
  5. namespace NetCoreServer
  6. {
  7. /// <summary>
  8. /// HTTP client is used to communicate with HTTP Web server. It allows to send GET, POST, PUT, DELETE requests and receive HTTP result.
  9. /// </summary>
  10. /// <remarks>Thread-safe.</remarks>
  11. public class HttpClient : TcpClient
  12. {
  13. /// <summary>
  14. /// Initialize HTTP client with a given IP address and port number
  15. /// </summary>
  16. /// <param name="address">IP address</param>
  17. /// <param name="port">Port number</param>
  18. public HttpClient(IPAddress address, int port) : base(address, port) { Request = new HttpRequest(); Response = new HttpResponse(); }
  19. /// <summary>
  20. /// Initialize HTTP client with a given IP address and port number
  21. /// </summary>
  22. /// <param name="address">IP address</param>
  23. /// <param name="port">Port number</param>
  24. public HttpClient(string address, int port) : base(address, port) { Request = new HttpRequest(); Response = new HttpResponse(); }
  25. /// <summary>
  26. /// Initialize HTTP client with a given DNS endpoint
  27. /// </summary>
  28. /// <param name="endpoint">DNS endpoint</param>
  29. public HttpClient(DnsEndPoint endpoint) : base(endpoint) { Request = new HttpRequest(); Response = new HttpResponse(); }
  30. /// <summary>
  31. /// Initialize HTTP client with a given IP endpoint
  32. /// </summary>
  33. /// <param name="endpoint">IP endpoint</param>
  34. public HttpClient(IPEndPoint endpoint) : base(endpoint) { Request = new HttpRequest(); Response = new HttpResponse(); }
  35. /// <summary>
  36. /// Get the HTTP request
  37. /// </summary>
  38. public HttpRequest Request { get; protected set; }
  39. /// <summary>
  40. /// Get the HTTP response
  41. /// </summary>
  42. protected HttpResponse Response { get; set; }
  43. #region Send request / Send request body
  44. /// <summary>
  45. /// Send the current HTTP request (synchronous)
  46. /// </summary>
  47. /// <returns>Size of sent data</returns>
  48. public long SendRequest() => SendRequest(Request);
  49. /// <summary>
  50. /// Send the HTTP request (synchronous)
  51. /// </summary>
  52. /// <param name="request">HTTP request</param>
  53. /// <returns>Size of sent data</returns>
  54. public long SendRequest(HttpRequest request) => Send(request.Cache.Data, request.Cache.Offset, request.Cache.Size);
  55. /// <summary>
  56. /// Send the HTTP request body (synchronous)
  57. /// </summary>
  58. /// <param name="body">HTTP request body</param>
  59. /// <returns>Size of sent data</returns>
  60. public long SendRequestBody(string body) => Send(body);
  61. /// <summary>
  62. /// Send the HTTP request body (synchronous)
  63. /// </summary>
  64. /// <param name="body">HTTP request body as a span of characters</param>
  65. /// <returns>Size of sent data</returns>
  66. public long SendRequestBody(ReadOnlySpan<char> body) => Send(body);
  67. /// <summary>
  68. /// Send the HTTP request body (synchronous)
  69. /// </summary>
  70. /// <param name="buffer">HTTP request body buffer</param>
  71. /// <returns>Size of sent data</returns>
  72. public long SendRequestBody(byte[] buffer) => Send(buffer);
  73. /// <summary>
  74. /// Send the HTTP request body (synchronous)
  75. /// </summary>
  76. /// <param name="buffer">HTTP request body buffer</param>
  77. /// <param name="offset">HTTP request body buffer offset</param>
  78. /// <param name="size">HTTP request body size</param>
  79. /// <returns>Size of sent data</returns>
  80. public long SendRequestBody(byte[] buffer, long offset, long size) => Send(buffer, offset, size);
  81. /// <summary>
  82. /// Send the HTTP request body (synchronous)
  83. /// </summary>
  84. /// <param name="buffer">HTTP request body buffer as a span of bytes</param>
  85. /// <returns>Size of sent data</returns>
  86. public long SendRequestBody(ReadOnlySpan<byte> buffer) => Send(buffer);
  87. /// <summary>
  88. /// Send the current HTTP request (asynchronous)
  89. /// </summary>
  90. /// <returns>'true' if the current HTTP request was successfully sent, 'false' if the session is not connected</returns>
  91. public bool SendRequestAsync() => SendRequestAsync(Request);
  92. /// <summary>
  93. /// Send the HTTP request (asynchronous)
  94. /// </summary>
  95. /// <param name="request">HTTP request</param>
  96. /// <returns>'true' if the current HTTP request was successfully sent, 'false' if the session is not connected</returns>
  97. public bool SendRequestAsync(HttpRequest request) => SendAsync(request.Cache.Data, request.Cache.Offset, request.Cache.Size);
  98. /// <summary>
  99. /// Send the HTTP request body (asynchronous)
  100. /// </summary>
  101. /// <param name="body">HTTP request body</param>
  102. /// <returns>'true' if the HTTP request body was successfully sent, 'false' if the session is not connected</returns>
  103. public bool SendRequestBodyAsync(string body) => SendAsync(body);
  104. /// <summary>
  105. /// Send the HTTP request body (asynchronous)
  106. /// </summary>
  107. /// <param name="body">HTTP request body as a span of characters</param>
  108. /// <returns>'true' if the HTTP request body was successfully sent, 'false' if the session is not connected</returns>
  109. public bool SendRequestBodyAsync(ReadOnlySpan<char> body) => SendAsync(body);
  110. /// <summary>
  111. /// Send the HTTP request body (asynchronous)
  112. /// </summary>
  113. /// <param name="buffer">HTTP request body buffer</param>
  114. /// <returns>'true' if the HTTP request body was successfully sent, 'false' if the session is not connected</returns>
  115. public bool SendRequestBodyAsync(byte[] buffer) => SendAsync(buffer);
  116. /// <summary>
  117. /// Send the HTTP request body (asynchronous)
  118. /// </summary>
  119. /// <param name="buffer">HTTP request body buffer</param>
  120. /// <param name="offset">HTTP request body buffer offset</param>
  121. /// <param name="size">HTTP request body size</param>
  122. /// <returns>'true' if the HTTP request body was successfully sent, 'false' if the session is not connected</returns>
  123. public bool SendRequestBodyAsync(byte[] buffer, long offset, long size) => SendAsync(buffer, offset, size);
  124. /// <summary>
  125. /// Send the HTTP request body (asynchronous)
  126. /// </summary>
  127. /// <param name="buffer">HTTP request body buffer as a span of bytes</param>
  128. /// <returns>'true' if the HTTP request body was successfully sent, 'false' if the session is not connected</returns>
  129. public bool SendRequestBodyAsync(ReadOnlySpan<byte> buffer) => SendAsync(buffer);
  130. #endregion
  131. #region Session handlers
  132. protected override void OnReceived(byte[] buffer, long offset, long size)
  133. {
  134. // Receive HTTP response header
  135. if (Response.IsPendingHeader())
  136. {
  137. if (Response.ReceiveHeader(buffer, (int)offset, (int)size))
  138. OnReceivedResponseHeader(Response);
  139. size = 0;
  140. }
  141. // Check for HTTP response error
  142. if (Response.IsErrorSet)
  143. {
  144. OnReceivedResponseError(Response, "Invalid HTTP response!");
  145. Response.Clear();
  146. Disconnect();
  147. return;
  148. }
  149. // Receive HTTP response body
  150. if (Response.ReceiveBody(buffer, (int)offset, (int)size))
  151. {
  152. OnReceivedResponse(Response);
  153. Response.Clear();
  154. return;
  155. }
  156. // Check for HTTP response error
  157. if (Response.IsErrorSet)
  158. {
  159. OnReceivedResponseError(Response, "Invalid HTTP response!");
  160. Response.Clear();
  161. Disconnect();
  162. return;
  163. }
  164. }
  165. protected override void OnDisconnected()
  166. {
  167. // Receive HTTP response body
  168. if (Response.IsPendingBody())
  169. {
  170. OnReceivedResponse(Response);
  171. Response.Clear();
  172. return;
  173. }
  174. }
  175. /// <summary>
  176. /// Handle HTTP response header received notification
  177. /// </summary>
  178. /// <remarks>Notification is called when HTTP response header was received from the server.</remarks>
  179. /// <param name="response">HTTP request</param>
  180. protected virtual void OnReceivedResponseHeader(HttpResponse response) {}
  181. /// <summary>
  182. /// Handle HTTP response received notification
  183. /// </summary>
  184. /// <remarks>Notification is called when HTTP response was received from the server.</remarks>
  185. /// <param name="response">HTTP response</param>
  186. protected virtual void OnReceivedResponse(HttpResponse response) {}
  187. /// <summary>
  188. /// Handle HTTP response error notification
  189. /// </summary>
  190. /// <remarks>Notification is called when HTTP response error was received from the server.</remarks>
  191. /// <param name="response">HTTP response</param>
  192. /// <param name="error">HTTP response error</param>
  193. protected virtual void OnReceivedResponseError(HttpResponse response, string error) {}
  194. #endregion
  195. }
  196. /// <summary>
  197. /// HTTP extended client make requests to HTTP Web server with returning Task as a synchronization primitive.
  198. /// </summary>
  199. /// <remarks>Thread-safe.</remarks>
  200. public class HttpClientEx : HttpClient
  201. {
  202. /// <summary>
  203. /// Initialize HTTP client with a given IP address and port number
  204. /// </summary>
  205. /// <param name="address">IP address</param>
  206. /// <param name="port">Port number</param>
  207. public HttpClientEx(IPAddress address, int port) : base(address, port) {}
  208. /// <summary>
  209. /// Initialize HTTP client with a given IP address and port number
  210. /// </summary>
  211. /// <param name="address">IP address</param>
  212. /// <param name="port">Port number</param>
  213. public HttpClientEx(string address, int port) : base(address, port) {}
  214. /// <summary>
  215. /// Initialize HTTP client with a given DNS endpoint
  216. /// </summary>
  217. /// <param name="endpoint">DNS endpoint</param>
  218. public HttpClientEx(DnsEndPoint endpoint) : base(endpoint) {}
  219. /// <summary>
  220. /// Initialize HTTP client with a given IP endpoint
  221. /// </summary>
  222. /// <param name="endpoint">IP endpoint</param>
  223. public HttpClientEx(IPEndPoint endpoint) : base(endpoint) {}
  224. #region Send request
  225. /// <summary>
  226. /// Send current HTTP request
  227. /// </summary>
  228. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  229. /// <returns>HTTP request Task</returns>
  230. public Task<HttpResponse> SendRequest(TimeSpan? timeout = null) => SendRequest(Request, timeout);
  231. /// <summary>
  232. /// Send HTTP request
  233. /// </summary>
  234. /// <param name="request">HTTP request</param>
  235. /// <param name="timeout">HTTP request timeout (default is 1 minute)</param>
  236. /// <returns>HTTP request Task</returns>
  237. public Task<HttpResponse> SendRequest(HttpRequest request, TimeSpan? timeout = null)
  238. {
  239. timeout ??= TimeSpan.FromMinutes(1);
  240. _tcs = new TaskCompletionSource<HttpResponse>();
  241. Request = request;
  242. // Check if the HTTP request is valid
  243. if (Request.IsEmpty || Request.IsErrorSet)
  244. {
  245. SetPromiseError("Invalid HTTP request!");
  246. return _tcs.Task;
  247. }
  248. if (!IsConnected)
  249. {
  250. // Connect to the Web server
  251. if (!ConnectAsync())
  252. {
  253. SetPromiseError("Connection failed!");
  254. return _tcs.Task;
  255. }
  256. }
  257. else
  258. {
  259. // Send prepared HTTP request
  260. if (!SendRequestAsync())
  261. {
  262. SetPromiseError("Failed to send HTTP request!");
  263. return _tcs.Task;
  264. }
  265. }
  266. void TimeoutHandler(object state)
  267. {
  268. // Disconnect on timeout
  269. OnReceivedResponseError(Response, "Timeout!");
  270. Response.Clear();
  271. DisconnectAsync();
  272. }
  273. // Create a new timeout timer
  274. if (_timer == null)
  275. _timer = new Timer(TimeoutHandler, null, Timeout.Infinite, Timeout.Infinite);
  276. // Start the timeout timer
  277. _timer.Change((int)timeout.Value.TotalMilliseconds, Timeout.Infinite);
  278. return _tcs.Task;
  279. }
  280. /// <summary>
  281. /// Send HEAD request
  282. /// </summary>
  283. /// <param name="url">URL to request</param>
  284. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  285. /// <returns>HTTP request Task</returns>
  286. public Task<HttpResponse> SendHeadRequest(string url, TimeSpan? timeout = null) => SendRequest(Request.MakeHeadRequest(url), timeout);
  287. /// <summary>
  288. /// Send GET request
  289. /// </summary>
  290. /// <param name="url">URL to request</param>
  291. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  292. /// <returns>HTTP request Task</returns>
  293. public Task<HttpResponse> SendGetRequest(string url, TimeSpan? timeout = null) => SendRequest(Request.MakeGetRequest(url), timeout);
  294. /// <summary>
  295. /// Send POST request
  296. /// </summary>
  297. /// <param name="url">URL to request</param>
  298. /// <param name="content">Content</param>
  299. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  300. /// <returns>HTTP request Task</returns>
  301. public Task<HttpResponse> SendPostRequest(string url, string content, TimeSpan? timeout = null) => SendRequest(Request.MakePostRequest(url, content), timeout);
  302. /// <summary>
  303. /// Send PUT request
  304. /// </summary>
  305. /// <param name="url">URL to request</param>
  306. /// <param name="content">Content</param>
  307. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  308. /// <returns>HTTP request Task</returns>
  309. public Task<HttpResponse> SendPutRequest(string url, string content, TimeSpan? timeout = null) => SendRequest(Request.MakePutRequest(url, content), timeout);
  310. /// <summary>
  311. /// Send DELETE request
  312. /// </summary>
  313. /// <param name="url">URL to request</param>
  314. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  315. /// <returns>HTTP request Task</returns>
  316. public Task<HttpResponse> SendDeleteRequest(string url, TimeSpan? timeout = null) => SendRequest(Request.MakeDeleteRequest(url), timeout);
  317. /// <summary>
  318. /// Send OPTIONS request
  319. /// </summary>
  320. /// <param name="url">URL to request</param>
  321. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  322. /// <returns>HTTP request Task</returns>
  323. public Task<HttpResponse> SendOptionsRequest(string url, TimeSpan? timeout = null) => SendRequest(Request.MakeOptionsRequest(url), timeout);
  324. /// <summary>
  325. /// Send TRACE request
  326. /// </summary>
  327. /// <param name="url">URL to request</param>
  328. /// <param name="timeout">Current HTTP request timeout (default is 1 minute)</param>
  329. /// <returns>HTTP request Task</returns>
  330. public Task<HttpResponse> SendTraceRequest(string url, TimeSpan? timeout = null) => SendRequest(Request.MakeTraceRequest(url), timeout);
  331. #endregion
  332. #region Session handlers
  333. protected override void OnConnected()
  334. {
  335. // Send prepared HTTP request on connect
  336. if (!Request.IsEmpty && !Request.IsErrorSet)
  337. if (!SendRequestAsync())
  338. SetPromiseError("Failed to send HTTP request!");
  339. }
  340. protected override void OnDisconnected()
  341. {
  342. // Cancel timeout check timer
  343. _timer?.Change(Timeout.Infinite, Timeout.Infinite);
  344. base.OnDisconnected();
  345. }
  346. protected override void OnReceivedResponse(HttpResponse response)
  347. {
  348. // Cancel timeout check timer
  349. _timer?.Change(Timeout.Infinite, Timeout.Infinite);
  350. SetPromiseValue(response);
  351. }
  352. protected override void OnReceivedResponseError(HttpResponse response, string error)
  353. {
  354. // Cancel timeout check timer
  355. _timer?.Change(Timeout.Infinite, Timeout.Infinite);
  356. SetPromiseError(error);
  357. }
  358. #endregion
  359. private TaskCompletionSource<HttpResponse> _tcs = new TaskCompletionSource<HttpResponse>();
  360. private Timer _timer;
  361. private void SetPromiseValue(HttpResponse response)
  362. {
  363. Response = new HttpResponse();
  364. _tcs.SetResult(response);
  365. Request.Clear();
  366. }
  367. private void SetPromiseError(string error)
  368. {
  369. _tcs.SetException(new Exception(error));
  370. Request.Clear();
  371. }
  372. #region IDisposable implementation
  373. // Disposed flag.
  374. private bool _disposed;
  375. protected override void Dispose(bool disposingManagedResources)
  376. {
  377. if (!_disposed)
  378. {
  379. if (disposingManagedResources)
  380. {
  381. // Dispose managed resources here...
  382. _timer?.Dispose();
  383. }
  384. // Dispose unmanaged resources here...
  385. // Set large fields to null here...
  386. // Mark as disposed.
  387. _disposed = true;
  388. }
  389. // Call Dispose in the base class.
  390. base.Dispose(disposingManagedResources);
  391. }
  392. // The derived class does not have a Finalize method
  393. // or a Dispose method without parameters because it inherits
  394. // them from the base class.
  395. #endregion
  396. }
  397. }