using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Net.Http; using System.Threading; using System.Threading.Tasks; using HtmlAgilityPack; using Wabbajack.Common; using Wabbajack.Common.Exceptions; using Wabbajack.Lib.LibCefHelpers; namespace Wabbajack.Lib.Http { public class Client { public List<(string, string?)> Headers = new List<(string, string?)>(); public List Cookies = new List(); public async Task GetAsync(string url, HttpCompletionOption responseHeadersRead = HttpCompletionOption.ResponseHeadersRead, bool errorsAsExceptions = true, bool retry = true, CancellationToken? token = null) { var request = new HttpRequestMessage(HttpMethod.Get, url); return await SendAsync(request, responseHeadersRead, errorsAsExceptions: errorsAsExceptions, retry: retry, token: token); } public async Task GetAsync(Uri url, HttpCompletionOption responseHeadersRead = HttpCompletionOption.ResponseHeadersRead, bool errorsAsExceptions = true, CancellationToken? token = null) { var request = new HttpRequestMessage(HttpMethod.Get, url); return await SendAsync(request, responseHeadersRead, errorsAsExceptions: errorsAsExceptions, token:token); } public async Task PostAsync(string url, HttpContent content, HttpCompletionOption responseHeadersRead = HttpCompletionOption.ResponseHeadersRead, bool errorsAsExceptions = true) { var request = new HttpRequestMessage(HttpMethod.Post, url) {Content = content}; return await SendAsync(request, responseHeadersRead, errorsAsExceptions); } public async Task PutAsync(string url, HttpContent content, HttpCompletionOption responseHeadersRead = HttpCompletionOption.ResponseHeadersRead) { var request = new HttpRequestMessage(HttpMethod.Put, url) {Content = content}; return await SendAsync(request, responseHeadersRead); } public async Task GetStringAsync(string url, CancellationToken? token = null) { var request = new HttpRequestMessage(HttpMethod.Get, url); return await SendStringAsync(request, token: token); } public async Task GetStringAsync(Uri url, CancellationToken? token = null) { var request = new HttpRequestMessage(HttpMethod.Get, url); return await SendStringAsync(request, token: token); } public async Task DeleteStringAsync(string url) { var request = new HttpRequestMessage(HttpMethod.Delete, url); return await SendStringAsync(request); } private async Task SendStringAsync(HttpRequestMessage request, CancellationToken? token = null) { using var result = await SendAsync(request, token: token); if (!result.IsSuccessStatusCode) { Utils.Log("Internal Error"); Utils.Log(await result.Content.ReadAsStringAsync()); throw new Exception( $"Bad HTTP request {result.StatusCode} {result.ReasonPhrase} - {request.RequestUri}"); } return await result.Content.ReadAsStringAsync(); } public async Task SendAsync(HttpRequestMessage msg, HttpCompletionOption responseHeadersRead = HttpCompletionOption.ResponseHeadersRead, bool errorsAsExceptions = true, bool retry = true, CancellationToken? token = null) { msg = FixupMessage(msg); foreach (var (k, v) in Headers) { if (k == "Host") msg.Headers.Host = v; else msg.Headers.Add(k, v); } if (Cookies.Count > 0) Cookies.ForEach(c => ClientFactory.Cookies.Add(c)); int retries = 0; HttpResponseMessage response; TOP: try { response = await ClientFactory.Client.SendAsync(msg, responseHeadersRead, token ?? CancellationToken.None); if (response.IsSuccessStatusCode) return response; if (errorsAsExceptions) { response.Dispose(); throw new HttpException(response); } return response; } catch (Exception ex) { if (!retry) throw; if (ex is HttpException http) { if (http.Code != 503 && http.Code != 521) throw; retries++; var ms = Utils.NextRandom(100, 1000); Utils.Log($"Got a {http.Code} from {msg.RequestUri} retrying in {ms}ms"); await Task.Delay(ms, token ?? CancellationToken.None); msg = CloneMessage(msg); goto TOP; } if (retries > Consts.MaxHTTPRetries) throw; retries++; Utils.LogStraightToFile(ex.ToString()); Utils.Log($"Http Connect error to {msg.RequestUri} retry {retries}"); await Task.Delay(100 * retries, token ?? CancellationToken.None); msg = CloneMessage(msg); goto TOP; } } private Dictionary> _workaroundMappings = new() { {"patches.wabbajack.org", () => (Consts.NetworkWorkaroundHost, "patches.wabbajack.org")}, {"authored-files.wabbajack.org", () => (Consts.NetworkWorkaroundHost, "authored-files.wabbajack.org")}, {"mirror.wabbajack.org", () => (Consts.NetworkWorkaroundHost, "mirror.wabbajack.org")}, {"build.wabbajack.org", () => (Consts.NetworkWorkaroundHost, "proxy-build.wabbajack.org")}, {"test-files.wabbajack.org", () => (Consts.NetworkWorkaroundHost, "test-files.wabbajack.org")}, }; private HttpRequestMessage FixupMessage(HttpRequestMessage msg) { if (!Consts.UseNetworkWorkaroundMode) return msg; var uri = new UriBuilder(msg.RequestUri!); if (!_workaroundMappings.TryGetValue(uri.Host, out var f)) return msg; var (ip, host) = f(); uri.Host = ip; msg.Headers.Host = host; msg.RequestUri = uri.Uri; return msg; } private HttpRequestMessage CloneMessage(HttpRequestMessage msg) { var new_message = new HttpRequestMessage(msg.Method, msg.RequestUri); foreach (var header in msg.Headers) new_message.Headers.Add(header.Key, header.Value); new_message.Content = msg.Content; return new_message; } public async Task GetJsonAsync(string s) { var result = await GetStringAsync(s); return result.FromJsonString(); } public async Task GetHtmlAsync(string s, CancellationToken? token = null) { var body = await GetStringAsync(s, token: token); var doc = new HtmlDocument(); doc.LoadHtml(body); return doc; } public Client WithHeader((string MetricsKeyHeader, string) header) { var newHeaders = Headers.Cons(header).ToList(); var client = new Client {Headers = newHeaders, Cookies = Cookies,}; return client; } public void AddCookies(Helpers.Cookie[] cookies) { Cookies.AddRange(cookies.Select(c => new Cookie {Domain = c.Domain, Name = c.Name, Value = c.Value, Path = c.Path})); } public void UseChromeUserAgent() { Headers.Add(("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.82 Safari/537.36")); } public async Task DownloadAsync(Uri url, AbsolutePath path) { using var response = await GetAsync(url); await using var content = await response.Content.ReadAsStreamAsync(); path.Parent.CreateDirectory(); await using var of = await path.Create(); await content.CopyToAsync(of); } } }