using System; using System.Collections.Generic; using System.Net.Http; using System.Net.Http.Headers; using System.Text; using System.Text.Json; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.Logging; namespace NucuCar.Core.Http { /// /// A simple HttpClient wrapper designed to make it easier to work with web requests with media type application/json. /// It implements a simple retry mechanism. /// public class MinimalHttpClient : IDisposable { #region Fields public ILogger Logger { get; set; } public int MaxRetries { get => maxRetries; set { if (value < 0 || value > 10) { throw new ArgumentOutOfRangeException($"Maximum retries allowed value is between 0 and 10!"); } maxRetries = value; } } public int Timeout { get => timeout; set { if (value < 0 || value > 10000) { throw new ArgumentOutOfRangeException($"Timeout allowed value is between 0 and 10000!"); } timeout = value; } } // ReSharper disable InconsistentNaming protected int maxRetries; protected int timeout; // ReSharper restore InconsistentNaming private readonly HttpClient _httpClient; #endregion #region Constructors public MinimalHttpClient() { _httpClient = new HttpClient(); maxRetries = 3; timeout = 10000; Logger = null; } public MinimalHttpClient(string baseAddress) : this() { _httpClient.BaseAddress = new Uri(baseAddress); } public MinimalHttpClient(string baseAddress, int maxRetries) : this(baseAddress) { MaxRetries = maxRetries; } #endregion #region Public Methods public void ClearAuthorizationHeader() { _httpClient.DefaultRequestHeaders.Authorization = null; } public void Authorization(string scheme, string token) { _httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue(scheme, token); } public void Authorization(string token) { Authorization("Bearer", token); } public async Task GetAsync(string path) { var request = _makeRequest(HttpMethod.Get, path); return await SendAsync(request); } public async Task PostAsync(string path, Dictionary data) { var request = _makeRequest(HttpMethod.Post, path); request.Content = _makeContent(data); return await SendAsync(request); } public async Task PutAsync(string path, Dictionary data) { var request = _makeRequest(HttpMethod.Put, path); request.Content = _makeContent(data); return await SendAsync(request); } public async Task DeleteAsync(string path, Dictionary data) { var request = _makeRequest(HttpMethod.Delete, path); request.Content = _makeContent(data); return await SendAsync(request); } /// /// Makes a request with timeout and retry support. /// /// The request to make. /// public virtual async Task SendAsync(HttpRequestMessage requestMessage) { var currentRetry = 0; HttpResponseMessage responseMessage = null; while (currentRetry < maxRetries) { try { // We need a request copy because we can't send the same request multiple times. var requestCopy = new HttpRequestMessage(requestMessage.Method, requestMessage.RequestUri); requestCopy.Headers.Authorization = requestMessage.Headers.Authorization; requestCopy.Content = requestMessage.Content; responseMessage = await _sendAsync(requestCopy); break; } catch (TaskCanceledException) { Logger?.LogError("Request timeout for {Uri}!", requestMessage.RequestUri); } catch (HttpRequestException e) { // The request failed due to an underlying issue such as network connectivity, DNS failure, // server certificate validation or timeout. Logger?.LogError("HttpRequestException timeout for {Uri}!", requestMessage.RequestUri); Logger?.LogError("{ErrorMessage}", e.Message); } finally { currentRetry += 1; } } return responseMessage; } public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } #endregion #region NonPublic Methods /// /// Creates a StringContent with media type of application.json and encodes it with UTF8. /// /// A dictionary representing JSON data. /// private StringContent _makeContent(Dictionary data) { return new StringContent(JsonSerializer.Serialize(data), Encoding.UTF8, "application/json"); } /// /// Creates a HttpRequestMessage, applies the auth header and constructs the uri. /// /// The HttpMethod to use /// The path, whether it is relative to the base or a new one. /// private HttpRequestMessage _makeRequest(HttpMethod method, string path) { var uri = _httpClient.BaseAddress == null ? new Uri(path) : new Uri(_httpClient.BaseAddress, path); var requestMessage = new HttpRequestMessage { Method = method, RequestUri = uri }; requestMessage.Headers.Authorization = _httpClient.DefaultRequestHeaders.Authorization; return requestMessage; } /// /// Makes a request which gets cancelled after Timeout. /// /// /// private async Task _sendAsync(HttpRequestMessage requestMessage) { var cts = new CancellationTokenSource(); HttpResponseMessage response; // Make sure we cancel after a certain timeout. cts.CancelAfter(timeout); try { response = await _httpClient.SendAsync(requestMessage, HttpCompletionOption.ResponseContentRead, cts.Token); } finally { cts.Dispose(); } return response; } protected virtual void Dispose(bool disposing) { if (disposing) { _httpClient.Dispose(); } } #endregion } /// /// HttpClientResponseMessageExtension provides extensions methods for the HttpResponseMessage class. /// public static class HttpResponseMessageExtension { /// /// Extension used to deserialize the body of a HttpResponseMessage into Json. /// /// The HttpResponseMessage message. /// A JsonElement. public static async Task GetJson(this HttpResponseMessage responseMessage) { return JsonSerializer.Deserialize(await responseMessage.Content.ReadAsStringAsync()); } } }