|
|
|
@ -4,6 +4,7 @@ using System.Diagnostics;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Net;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using NLog;
|
|
|
|
|
using NzbDrone.Common.Cache;
|
|
|
|
|
using NzbDrone.Common.EnvironmentInfo;
|
|
|
|
@ -24,6 +25,16 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
HttpResponse Post(HttpRequest request);
|
|
|
|
|
HttpResponse<T> Post<T>(HttpRequest request)
|
|
|
|
|
where T : new();
|
|
|
|
|
|
|
|
|
|
Task<HttpResponse> ExecuteAsync(HttpRequest request);
|
|
|
|
|
Task DownloadFileAsync(string url, string fileName);
|
|
|
|
|
Task<HttpResponse> GetAsync(HttpRequest request);
|
|
|
|
|
Task<HttpResponse<T>> GetAsync<T>(HttpRequest request)
|
|
|
|
|
where T : new();
|
|
|
|
|
Task<HttpResponse> HeadAsync(HttpRequest request);
|
|
|
|
|
Task<HttpResponse> PostAsync(HttpRequest request);
|
|
|
|
|
Task<HttpResponse<T>> PostAsync<T>(HttpRequest request)
|
|
|
|
|
where T : new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class HttpClient : IHttpClient
|
|
|
|
@ -54,11 +65,11 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
_cookieContainerCache = cacheManager.GetCache<CookieContainer>(typeof(HttpClient));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse Execute(HttpRequest request)
|
|
|
|
|
public async Task<HttpResponse> ExecuteAsync(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
var cookieContainer = InitializeRequestCookies(request);
|
|
|
|
|
|
|
|
|
|
var response = ExecuteRequest(request, cookieContainer);
|
|
|
|
|
var response = await ExecuteRequestAsync(request, cookieContainer);
|
|
|
|
|
|
|
|
|
|
if (request.AllowAutoRedirect && response.HasHttpRedirect)
|
|
|
|
|
{
|
|
|
|
@ -77,7 +88,7 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
throw new WebException($"Too many automatic redirections were attempted for {autoRedirectChain.Join(" -> ")}", WebExceptionStatus.ProtocolError);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
response = ExecuteRequest(request, cookieContainer);
|
|
|
|
|
response = await ExecuteRequestAsync(request, cookieContainer);
|
|
|
|
|
}
|
|
|
|
|
while (response.HasHttpRedirect);
|
|
|
|
|
}
|
|
|
|
@ -104,7 +115,12 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
return response;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private HttpResponse ExecuteRequest(HttpRequest request, CookieContainer cookieContainer)
|
|
|
|
|
public HttpResponse Execute(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
return ExecuteAsync(request).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task<HttpResponse> ExecuteRequestAsync(HttpRequest request, CookieContainer cookieContainer)
|
|
|
|
|
{
|
|
|
|
|
foreach (var interceptor in _requestInterceptors)
|
|
|
|
|
{
|
|
|
|
@ -113,7 +129,7 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
|
|
|
|
|
if (request.RateLimit != TimeSpan.Zero)
|
|
|
|
|
{
|
|
|
|
|
_rateLimitService.WaitAndPulse(request.Url.Host, request.RateLimit);
|
|
|
|
|
await _rateLimitService.WaitAndPulseAsync(request.Url.Host, request.RateLimit);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_logger.Trace(request);
|
|
|
|
@ -122,7 +138,7 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
|
|
|
|
|
PrepareRequestCookies(request, cookieContainer);
|
|
|
|
|
|
|
|
|
|
var response = _httpDispatcher.GetResponse(request, cookieContainer);
|
|
|
|
|
var response = await _httpDispatcher.GetResponseAsync(request, cookieContainer);
|
|
|
|
|
|
|
|
|
|
HandleResponseCookies(response, cookieContainer);
|
|
|
|
|
|
|
|
|
@ -231,7 +247,7 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void DownloadFile(string url, string fileName)
|
|
|
|
|
public async Task DownloadFileAsync(string url, string fileName)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
@ -247,7 +263,7 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
using (var webClient = new GZipWebClient())
|
|
|
|
|
{
|
|
|
|
|
webClient.Headers.Add(HttpRequestHeader.UserAgent, _userAgentBuilder.GetUserAgent());
|
|
|
|
|
webClient.DownloadFile(url, fileName);
|
|
|
|
|
await webClient.DownloadFileTaskAsync(url, fileName);
|
|
|
|
|
stopWatch.Stop();
|
|
|
|
|
_logger.Debug("Downloading Completed. took {0:0}s", stopWatch.Elapsed.Seconds);
|
|
|
|
|
}
|
|
|
|
@ -255,49 +271,94 @@ namespace NzbDrone.Common.Http
|
|
|
|
|
catch (WebException e)
|
|
|
|
|
{
|
|
|
|
|
_logger.Warn("Failed to get response from: {0} {1}", url, e.Message);
|
|
|
|
|
|
|
|
|
|
if (File.Exists(fileName))
|
|
|
|
|
{
|
|
|
|
|
File.Delete(fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
_logger.Warn(e, "Failed to get response from: " + url);
|
|
|
|
|
|
|
|
|
|
if (File.Exists(fileName))
|
|
|
|
|
{
|
|
|
|
|
File.Delete(fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse Get(HttpRequest request)
|
|
|
|
|
public void DownloadFile(string url, string fileName)
|
|
|
|
|
{
|
|
|
|
|
// https://docs.microsoft.com/en-us/archive/msdn-magazine/2015/july/async-programming-brownfield-async-development#the-thread-pool-hack
|
|
|
|
|
Task.Run(() => DownloadFileAsync(url, fileName)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<HttpResponse> GetAsync(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
request.Method = HttpMethod.GET;
|
|
|
|
|
return Execute(request);
|
|
|
|
|
return ExecuteAsync(request);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse<T> Get<T>(HttpRequest request)
|
|
|
|
|
public HttpResponse Get(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
return Task.Run(() => GetAsync(request)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<HttpResponse<T>> GetAsync<T>(HttpRequest request)
|
|
|
|
|
where T : new()
|
|
|
|
|
{
|
|
|
|
|
var response = Get(request);
|
|
|
|
|
var response = await GetAsync(request);
|
|
|
|
|
CheckResponseContentType(response);
|
|
|
|
|
return new HttpResponse<T>(response);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse Head(HttpRequest request)
|
|
|
|
|
public HttpResponse<T> Get<T>(HttpRequest request)
|
|
|
|
|
where T : new()
|
|
|
|
|
{
|
|
|
|
|
return Task.Run(() => GetAsync<T>(request)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<HttpResponse> HeadAsync(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
request.Method = HttpMethod.HEAD;
|
|
|
|
|
return Execute(request);
|
|
|
|
|
return ExecuteAsync(request);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse Post(HttpRequest request)
|
|
|
|
|
public HttpResponse Head(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
return Task.Run(() => HeadAsync(request)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<HttpResponse> PostAsync(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
request.Method = HttpMethod.POST;
|
|
|
|
|
return Execute(request);
|
|
|
|
|
return ExecuteAsync(request);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse<T> Post<T>(HttpRequest request)
|
|
|
|
|
public HttpResponse Post(HttpRequest request)
|
|
|
|
|
{
|
|
|
|
|
return Task.Run(() => PostAsync(request)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<HttpResponse<T>> PostAsync<T>(HttpRequest request)
|
|
|
|
|
where T : new()
|
|
|
|
|
{
|
|
|
|
|
var response = Post(request);
|
|
|
|
|
var response = await PostAsync(request);
|
|
|
|
|
CheckResponseContentType(response);
|
|
|
|
|
return new HttpResponse<T>(response);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HttpResponse<T> Post<T>(HttpRequest request)
|
|
|
|
|
where T : new()
|
|
|
|
|
{
|
|
|
|
|
return Task.Run(() => PostAsync<T>(request)).GetAwaiter().GetResult();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CheckResponseContentType(HttpResponse response)
|
|
|
|
|
{
|
|
|
|
|
if (response.Headers.ContentType != null && response.Headers.ContentType.Contains("text/html"))
|
|
|
|
|