using Logging; using Newtonsoft.Json; using Serialization = Newtonsoft.Json.Serialization; using System.Net.Http.Headers; using System.Net.Http.Json; using Utils; namespace Core { public interface IEndpoint { string HttpGetString(string route); T HttpGetJson(string route); TResponse HttpPostJson(string route, TRequest body); string HttpPostJson(string route, TRequest body); TResponse HttpPostString(string route, string body); string HttpPostStream(string route, Stream stream); Stream HttpGetStream(string route); T Deserialize(string json); } internal class Endpoint : IEndpoint { private readonly ILog log; private readonly IHttp http; private readonly Address address; private readonly string baseUrl; private readonly string? logAlias; public Endpoint(ILog log, IHttp http, Address address, string baseUrl, string? logAlias) { this.log = log; this.http = http; this.address = address; this.baseUrl = baseUrl; this.logAlias = logAlias; } public string HttpGetString(string route) { return http.OnClient(client => { return GetString(client, route); }, $"HTTP-GET:{route}"); } public T HttpGetJson(string route) { return http.OnClient(client => { var json = GetString(client, route); return Deserialize(json); }, $"HTTP-GET:{route}"); } public TResponse HttpPostJson(string route, TRequest body) { return http.OnClient(client => { var response = PostJson(client, route, body); var json = Time.Wait(response.Content.ReadAsStringAsync()); if (!response.IsSuccessStatusCode) { throw new HttpRequestException(json); } Log(GetUrl() + route, json); return Deserialize(json); }, $"HTTP-POST-JSON: {route}"); } public string HttpPostJson(string route, TRequest body) { return http.OnClient(client => { var response = PostJson(client, route, body); return Time.Wait(response.Content.ReadAsStringAsync()); }, $"HTTP-POST-JSON: {route}"); } public TResponse HttpPostString(string route, string body) { return http.OnClient(client => { var response = PostJsonString(client, route, body); if (response == null) throw new Exception("Received no response."); var result = Deserialize(response); if (result == null) throw new Exception("Failed to deserialize response"); return result; }, $"HTTO-POST-JSON: {route}"); } public string HttpPostStream(string route, Stream stream) { return http.OnClient(client => { var url = GetUrl() + route; Log(url, "~ STREAM ~"); var content = new StreamContent(stream); content.Headers.ContentType = new MediaTypeHeaderValue("application/octet-stream"); var response = Time.Wait(client.PostAsync(url, content)); var str = Time.Wait(response.Content.ReadAsStringAsync()); Log(url, str); return str; }, $"HTTP-POST-STREAM: {route}"); } public Stream HttpGetStream(string route) { return http.OnClient(client => { var url = GetUrl() + route; Log(url, "~ STREAM ~"); return Time.Wait(client.GetStreamAsync(url)); }, $"HTTP-GET-STREAM: {route}"); } public T Deserialize(string json) { var errors = new List(); var deserialized = JsonConvert.DeserializeObject(json, new JsonSerializerSettings() { Error = delegate (object? sender, Serialization.ErrorEventArgs args) { if (args.CurrentObject == args.ErrorContext.OriginalObject) { errors.Add($""" Member: '{args.ErrorContext.Member?.ToString() ?? ""}' Path: {args.ErrorContext.Path} Error: {args.ErrorContext.Error.Message} """); args.ErrorContext.Handled = true; } } }); if (errors.Count > 0) { throw new JsonSerializationException($"Failed to deserialize JSON '{json}' with exception(s): \n{string.Join("\n", errors)}"); } else if (deserialized == null) { throw new JsonSerializationException($"Failed to deserialize JSON '{json}': resulting deserialized object is null"); } return deserialized; } private string GetString(HttpClient client, string route) { var url = GetUrl() + route; Log(url, ""); var result = Time.Wait(client.GetAsync(url)); var str = Time.Wait(result.Content.ReadAsStringAsync()); Log(url, str); return str; } private HttpResponseMessage PostJson(HttpClient client, string route, TRequest body) { var url = GetUrl() + route; using var content = JsonContent.Create(body); Log(url, JsonConvert.SerializeObject(body)); return Time.Wait(client.PostAsync(url, content)); } private string PostJsonString(HttpClient client, string route, string body) { var url = GetUrl() + route; Log(url, body); var content = new StringContent(body); content.Headers.ContentType = MediaTypeHeaderValue.Parse("application/json"); var result = Time.Wait(client.PostAsync(url, content)); var str = Time.Wait(result.Content.ReadAsStringAsync()); Log(url, str); return str; } private string GetUrl() { return $"{address.Host}:{address.Port}{baseUrl}"; } private void Log(string url, string message) { if (logAlias != null) { log.Debug($"({logAlias})({url}) = '{message}'", 3); } else { log.Debug($"({url}) = '{message}'", 3); } } } }