mirror of
https://github.com/Readarr/Readarr.git
synced 2026-04-18 21:34:28 -04:00
Use modern HttpClient
(cherry picked from commit 402f8b296f17bf161824ec5ff40d67d036d00d94)
This commit is contained in:
@@ -17,6 +17,7 @@ using NzbDrone.Common.Http.Proxy;
|
|||||||
using NzbDrone.Common.TPL;
|
using NzbDrone.Common.TPL;
|
||||||
using NzbDrone.Test.Common;
|
using NzbDrone.Test.Common;
|
||||||
using NzbDrone.Test.Common.Categories;
|
using NzbDrone.Test.Common.Categories;
|
||||||
|
using HttpClient = NzbDrone.Common.Http.HttpClient;
|
||||||
|
|
||||||
namespace NzbDrone.Common.Test.Http
|
namespace NzbDrone.Common.Test.Http
|
||||||
{
|
{
|
||||||
@@ -31,6 +32,8 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
private string _httpBinHost;
|
private string _httpBinHost;
|
||||||
private string _httpBinHost2;
|
private string _httpBinHost2;
|
||||||
|
|
||||||
|
private System.Net.Http.HttpClient _httpClient = new ();
|
||||||
|
|
||||||
[OneTimeSetUp]
|
[OneTimeSetUp]
|
||||||
public void FixtureSetUp()
|
public void FixtureSetUp()
|
||||||
{
|
{
|
||||||
@@ -53,22 +56,13 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var req = WebRequest.Create($"https://{site}/get") as HttpWebRequest;
|
var res = _httpClient.GetAsync($"https://{site}/get").GetAwaiter().GetResult();
|
||||||
var res = req.GetResponse() as HttpWebResponse;
|
|
||||||
if (res.StatusCode != HttpStatusCode.OK)
|
if (res.StatusCode != HttpStatusCode.OK)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
res = _httpClient.GetAsync($"https://{site}/status/429").GetAwaiter().GetResult();
|
||||||
{
|
|
||||||
req = WebRequest.Create($"https://{site}/status/429") as HttpWebRequest;
|
|
||||||
res = req.GetResponse() as HttpWebResponse;
|
|
||||||
}
|
|
||||||
catch (WebException ex)
|
|
||||||
{
|
|
||||||
res = ex.Response as HttpWebResponse;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (res == null || res.StatusCode != (HttpStatusCode)429)
|
if (res == null || res.StatusCode != (HttpStatusCode)429)
|
||||||
{
|
{
|
||||||
@@ -165,7 +159,9 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
var response = Subject.Get<HttpBinResource>(request);
|
var response = Subject.Get<HttpBinResource>(request);
|
||||||
|
|
||||||
response.Resource.Headers["Accept-Encoding"].ToString().Should().Contain("gzip");
|
response.Resource.Headers["Accept-Encoding"].ToString().Should().Contain("gzip");
|
||||||
|
|
||||||
response.Resource.Gzipped.Should().BeTrue();
|
response.Resource.Gzipped.Should().BeTrue();
|
||||||
|
response.Resource.Brotli.Should().BeFalse();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
@@ -176,6 +172,8 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
var response = Subject.Get<HttpBinResource>(request);
|
var response = Subject.Get<HttpBinResource>(request);
|
||||||
|
|
||||||
response.Resource.Headers["Accept-Encoding"].ToString().Should().Contain("br");
|
response.Resource.Headers["Accept-Encoding"].ToString().Should().Contain("br");
|
||||||
|
|
||||||
|
response.Resource.Gzipped.Should().BeFalse();
|
||||||
response.Resource.Brotli.Should().BeTrue();
|
response.Resource.Brotli.Should().BeTrue();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -358,13 +356,38 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
{
|
{
|
||||||
var file = GetTempFilePath();
|
var file = GetTempFilePath();
|
||||||
|
|
||||||
Assert.Throws<WebException>(() => Subject.DownloadFile("https://download.readarr.com/wrongpath", file));
|
Assert.Throws<HttpException>(() => Subject.DownloadFile("https://download.sonarr.tv/wrongpath", file));
|
||||||
|
|
||||||
File.Exists(file).Should().BeFalse();
|
File.Exists(file).Should().BeFalse();
|
||||||
|
|
||||||
ExceptionVerification.ExpectedWarns(1);
|
ExceptionVerification.ExpectedWarns(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_not_write_redirect_content_to_stream()
|
||||||
|
{
|
||||||
|
var file = GetTempFilePath();
|
||||||
|
|
||||||
|
using (var fileStream = new FileStream(file, FileMode.Create))
|
||||||
|
{
|
||||||
|
var request = new HttpRequest($"http://{_httpBinHost}/redirect/1");
|
||||||
|
request.AllowAutoRedirect = false;
|
||||||
|
request.ResponseStream = fileStream;
|
||||||
|
|
||||||
|
var response = Subject.Get(request);
|
||||||
|
|
||||||
|
response.StatusCode.Should().Be(HttpStatusCode.Moved);
|
||||||
|
}
|
||||||
|
|
||||||
|
ExceptionVerification.ExpectedErrors(1);
|
||||||
|
|
||||||
|
File.Exists(file).Should().BeTrue();
|
||||||
|
|
||||||
|
var fileInfo = new FileInfo(file);
|
||||||
|
|
||||||
|
fileInfo.Length.Should().Be(0);
|
||||||
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void should_send_cookie()
|
public void should_send_cookie()
|
||||||
{
|
{
|
||||||
@@ -784,6 +807,28 @@ namespace NzbDrone.Common.Test.Http
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_correctly_use_basic_auth()
|
||||||
|
{
|
||||||
|
var request = new HttpRequest($"https://{_httpBinHost}/basic-auth/username/password");
|
||||||
|
request.Credentials = new BasicNetworkCredential("username", "password");
|
||||||
|
|
||||||
|
var response = Subject.Execute(request);
|
||||||
|
|
||||||
|
response.StatusCode.Should().Be(HttpStatusCode.OK);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_correctly_use_digest_auth()
|
||||||
|
{
|
||||||
|
var request = new HttpRequest($"https://{_httpBinHost}/digest-auth/auth/username/password");
|
||||||
|
request.Credentials = new NetworkCredential("username", "password");
|
||||||
|
|
||||||
|
var response = Subject.Execute(request);
|
||||||
|
|
||||||
|
response.StatusCode.Should().Be(HttpStatusCode.OK);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class HttpBinResource
|
public class HttpBinResource
|
||||||
|
|||||||
@@ -5,6 +5,5 @@ namespace NzbDrone.Common.Http.Dispatchers
|
|||||||
public interface IHttpDispatcher
|
public interface IHttpDispatcher
|
||||||
{
|
{
|
||||||
HttpResponse GetResponse(HttpRequest request, CookieContainer cookies);
|
HttpResponse GetResponse(HttpRequest request, CookieContainer cookies);
|
||||||
void DownloadFile(string url, string fileName);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Diagnostics;
|
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
using NzbDrone.Common.Http.Proxy;
|
using NzbDrone.Common.Http.Proxy;
|
||||||
|
|
||||||
@@ -12,120 +14,113 @@ namespace NzbDrone.Common.Http.Dispatchers
|
|||||||
{
|
{
|
||||||
public class ManagedHttpDispatcher : IHttpDispatcher
|
public class ManagedHttpDispatcher : IHttpDispatcher
|
||||||
{
|
{
|
||||||
|
private const string NO_PROXY_KEY = "no-proxy";
|
||||||
|
|
||||||
private readonly IHttpProxySettingsProvider _proxySettingsProvider;
|
private readonly IHttpProxySettingsProvider _proxySettingsProvider;
|
||||||
private readonly ICreateManagedWebProxy _createManagedWebProxy;
|
private readonly ICreateManagedWebProxy _createManagedWebProxy;
|
||||||
private readonly IUserAgentBuilder _userAgentBuilder;
|
private readonly IUserAgentBuilder _userAgentBuilder;
|
||||||
private readonly IPlatformInfo _platformInfo;
|
private readonly ICached<System.Net.Http.HttpClient> _httpClientCache;
|
||||||
|
private readonly ICached<CredentialCache> _credentialCache;
|
||||||
private readonly Logger _logger;
|
private readonly Logger _logger;
|
||||||
|
|
||||||
public ManagedHttpDispatcher(IHttpProxySettingsProvider proxySettingsProvider, ICreateManagedWebProxy createManagedWebProxy, IUserAgentBuilder userAgentBuilder, IPlatformInfo platformInfo, Logger logger)
|
public ManagedHttpDispatcher(IHttpProxySettingsProvider proxySettingsProvider,
|
||||||
|
ICreateManagedWebProxy createManagedWebProxy,
|
||||||
|
IUserAgentBuilder userAgentBuilder,
|
||||||
|
ICacheManager cacheManager,
|
||||||
|
Logger logger)
|
||||||
{
|
{
|
||||||
_proxySettingsProvider = proxySettingsProvider;
|
_proxySettingsProvider = proxySettingsProvider;
|
||||||
_createManagedWebProxy = createManagedWebProxy;
|
_createManagedWebProxy = createManagedWebProxy;
|
||||||
_userAgentBuilder = userAgentBuilder;
|
_userAgentBuilder = userAgentBuilder;
|
||||||
_platformInfo = platformInfo;
|
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
|
|
||||||
|
_httpClientCache = cacheManager.GetCache<System.Net.Http.HttpClient>(typeof(ManagedHttpDispatcher), "httpclient");
|
||||||
|
_credentialCache = cacheManager.GetCache<CredentialCache>(typeof(ManagedHttpDispatcher), "credentialcache");
|
||||||
}
|
}
|
||||||
|
|
||||||
public HttpResponse GetResponse(HttpRequest request, CookieContainer cookies)
|
public HttpResponse GetResponse(HttpRequest request, CookieContainer cookies)
|
||||||
{
|
{
|
||||||
var webRequest = (HttpWebRequest)WebRequest.Create((Uri)request.Url);
|
var requestMessage = new HttpRequestMessage(request.Method, (Uri)request.Url);
|
||||||
|
requestMessage.Headers.UserAgent.ParseAdd(_userAgentBuilder.GetUserAgent(request.UseSimplifiedUserAgent));
|
||||||
|
requestMessage.Headers.ConnectionClose = !request.ConnectionKeepAlive;
|
||||||
|
|
||||||
// Deflate is not a standard and could break depending on implementation.
|
var cookieHeader = cookies.GetCookieHeader((Uri)request.Url);
|
||||||
// we should just stick with the more compatible Gzip
|
if (cookieHeader.IsNotNullOrWhiteSpace())
|
||||||
//http://stackoverflow.com/questions/8490718/how-to-decompress-stream-deflated-with-java-util-zip-deflater-in-net
|
{
|
||||||
webRequest.AutomaticDecompression = DecompressionMethods.Brotli | DecompressionMethods.GZip;
|
requestMessage.Headers.Add("Cookie", cookieHeader);
|
||||||
|
}
|
||||||
webRequest.Method = request.Method.ToString();
|
|
||||||
webRequest.UserAgent = _userAgentBuilder.GetUserAgent(request.UseSimplifiedUserAgent);
|
|
||||||
webRequest.KeepAlive = request.ConnectionKeepAlive;
|
|
||||||
webRequest.AllowAutoRedirect = false;
|
|
||||||
webRequest.CookieContainer = cookies;
|
|
||||||
|
|
||||||
if (request.Credentials != null)
|
if (request.Credentials != null)
|
||||||
{
|
{
|
||||||
if (request.Credentials is BasicNetworkCredential nc)
|
if (request.Credentials is BasicNetworkCredential bc)
|
||||||
{
|
{
|
||||||
// Manually set header to avoid initial challenge response
|
// Manually set header to avoid initial challenge response
|
||||||
var authInfo = nc.UserName + ":" + nc.Password;
|
var authInfo = bc.UserName + ":" + bc.Password;
|
||||||
authInfo = Convert.ToBase64String(Encoding.GetEncoding("ISO-8859-1").GetBytes(authInfo));
|
authInfo = Convert.ToBase64String(Encoding.GetEncoding("ISO-8859-1").GetBytes(authInfo));
|
||||||
webRequest.Headers.Add("Authorization", "Basic " + authInfo);
|
requestMessage.Headers.Authorization = new AuthenticationHeaderValue("Basic", authInfo);
|
||||||
}
|
}
|
||||||
else
|
else if (request.Credentials is NetworkCredential nc)
|
||||||
{
|
{
|
||||||
webRequest.PreAuthenticate = true;
|
var creds = GetCredentialCache();
|
||||||
webRequest.Credentials = request.Credentials;
|
creds.Remove((Uri)request.Url, "Digest");
|
||||||
|
creds.Add((Uri)request.Url, "Digest", nc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
using var cts = new CancellationTokenSource();
|
||||||
if (request.RequestTimeout != TimeSpan.Zero)
|
if (request.RequestTimeout != TimeSpan.Zero)
|
||||||
{
|
{
|
||||||
webRequest.Timeout = (int)Math.Ceiling(request.RequestTimeout.TotalMilliseconds);
|
cts.CancelAfter(request.RequestTimeout);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// The default for System.Net.Http.HttpClient
|
||||||
|
cts.CancelAfter(TimeSpan.FromSeconds(100));
|
||||||
}
|
}
|
||||||
|
|
||||||
webRequest.Proxy = GetProxy(request.Url);
|
|
||||||
|
|
||||||
if (request.Headers != null)
|
if (request.Headers != null)
|
||||||
{
|
{
|
||||||
AddRequestHeaders(webRequest, request.Headers);
|
AddRequestHeaders(requestMessage, request.Headers);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpWebResponse httpWebResponse;
|
var httpClient = GetClient(request.Url);
|
||||||
|
|
||||||
|
HttpResponseMessage responseMessage;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (request.ContentData != null)
|
if (request.ContentData != null)
|
||||||
{
|
{
|
||||||
webRequest.ContentLength = request.ContentData.Length;
|
var content = new ByteArrayContent(request.ContentData);
|
||||||
using (var writeStream = webRequest.GetRequestStream())
|
content.Headers.Remove("Content-Type");
|
||||||
|
if (request.Headers.ContentType.IsNotNullOrWhiteSpace())
|
||||||
{
|
{
|
||||||
writeStream.Write(request.ContentData, 0, request.ContentData.Length);
|
content.Headers.Add("Content-Type", request.Headers.ContentType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
requestMessage.Content = content;
|
||||||
}
|
}
|
||||||
|
|
||||||
httpWebResponse = (HttpWebResponse)webRequest.GetResponse();
|
responseMessage = httpClient.Send(requestMessage, cts.Token);
|
||||||
}
|
}
|
||||||
catch (WebException e)
|
catch (HttpRequestException e)
|
||||||
{
|
{
|
||||||
httpWebResponse = (HttpWebResponse)e.Response;
|
_logger.Error(e, "HttpClient error");
|
||||||
|
throw;
|
||||||
if (httpWebResponse == null)
|
|
||||||
{
|
|
||||||
// The default messages for WebException on mono are pretty horrible.
|
|
||||||
if (e.Status == WebExceptionStatus.NameResolutionFailure)
|
|
||||||
{
|
|
||||||
throw new WebException($"DNS Name Resolution Failure: '{webRequest.RequestUri.Host}'", e.Status);
|
|
||||||
}
|
|
||||||
else if (e.ToString().Contains("TLS Support not"))
|
|
||||||
{
|
|
||||||
throw new TlsFailureException(webRequest, e);
|
|
||||||
}
|
|
||||||
else if (e.ToString().Contains("The authentication or decryption has failed."))
|
|
||||||
{
|
|
||||||
throw new TlsFailureException(webRequest, e);
|
|
||||||
}
|
|
||||||
else if (OsInfo.IsNotWindows)
|
|
||||||
{
|
|
||||||
throw new WebException($"{e.Message}: '{webRequest.RequestUri}'", e, e.Status, e.Response);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
byte[] data = null;
|
byte[] data = null;
|
||||||
|
|
||||||
using (var responseStream = httpWebResponse.GetResponseStream())
|
using (var responseStream = responseMessage.Content.ReadAsStream())
|
||||||
{
|
{
|
||||||
if (responseStream != null && responseStream != Stream.Null)
|
if (responseStream != null && responseStream != Stream.Null)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (request.ResponseStream != null)
|
if (request.ResponseStream != null && responseMessage.StatusCode == HttpStatusCode.OK)
|
||||||
{
|
{
|
||||||
// A target ResponseStream was specified, write to that instead.
|
// A target ResponseStream was specified, write to that instead.
|
||||||
|
// But only on the OK status code, since we don't want to write failures and redirects.
|
||||||
responseStream.CopyTo(request.ResponseStream);
|
responseStream.CopyTo(request.ResponseStream);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -135,102 +130,88 @@ namespace NzbDrone.Common.Http.Dispatchers
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
throw new WebException("Failed to read complete http response", ex, WebExceptionStatus.ReceiveFailure, httpWebResponse);
|
throw new WebException("Failed to read complete http response", ex, WebExceptionStatus.ReceiveFailure, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return new HttpResponse(request, new HttpHeader(httpWebResponse.Headers), data, httpWebResponse.StatusCode);
|
return new HttpResponse(request, new HttpHeader(responseMessage.Headers), data, responseMessage.StatusCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void DownloadFile(string url, string fileName)
|
protected virtual System.Net.Http.HttpClient GetClient(HttpUri uri)
|
||||||
{
|
{
|
||||||
try
|
|
||||||
{
|
|
||||||
var fileInfo = new FileInfo(fileName);
|
|
||||||
if (fileInfo.Directory != null && !fileInfo.Directory.Exists)
|
|
||||||
{
|
|
||||||
fileInfo.Directory.Create();
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.Debug("Downloading [{0}] to [{1}]", url, fileName);
|
|
||||||
|
|
||||||
var stopWatch = Stopwatch.StartNew();
|
|
||||||
var uri = new HttpUri(url);
|
|
||||||
|
|
||||||
using (var webClient = new GZipWebClient())
|
|
||||||
{
|
|
||||||
webClient.Headers.Add(HttpRequestHeader.UserAgent, _userAgentBuilder.GetUserAgent());
|
|
||||||
webClient.Proxy = GetProxy(uri);
|
|
||||||
webClient.DownloadFile(uri.FullUri, fileName);
|
|
||||||
stopWatch.Stop();
|
|
||||||
_logger.Debug("Downloading Completed. took {0:0}s", stopWatch.Elapsed.Seconds);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (WebException e)
|
|
||||||
{
|
|
||||||
_logger.Warn("Failed to get response from: {0} {1}", url, e.Message);
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
_logger.Warn(e, "Failed to get response from: " + url);
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected virtual IWebProxy GetProxy(HttpUri uri)
|
|
||||||
{
|
|
||||||
IWebProxy proxy = null;
|
|
||||||
|
|
||||||
var proxySettings = _proxySettingsProvider.GetProxySettings(uri);
|
var proxySettings = _proxySettingsProvider.GetProxySettings(uri);
|
||||||
|
|
||||||
|
var key = proxySettings?.Key ?? NO_PROXY_KEY;
|
||||||
|
|
||||||
|
return _httpClientCache.Get(key, () => CreateHttpClient(proxySettings));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual System.Net.Http.HttpClient CreateHttpClient(HttpProxySettings proxySettings)
|
||||||
|
{
|
||||||
|
var handler = new HttpClientHandler()
|
||||||
|
{
|
||||||
|
AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Brotli,
|
||||||
|
UseCookies = false, // sic - we don't want to use a shared cookie container
|
||||||
|
AllowAutoRedirect = false,
|
||||||
|
Credentials = GetCredentialCache(),
|
||||||
|
PreAuthenticate = true
|
||||||
|
};
|
||||||
|
|
||||||
if (proxySettings != null)
|
if (proxySettings != null)
|
||||||
{
|
{
|
||||||
proxy = _createManagedWebProxy.GetWebProxy(proxySettings);
|
handler.Proxy = _createManagedWebProxy.GetWebProxy(proxySettings);
|
||||||
}
|
}
|
||||||
|
|
||||||
return proxy;
|
var client = new System.Net.Http.HttpClient(handler)
|
||||||
|
{
|
||||||
|
Timeout = Timeout.InfiniteTimeSpan
|
||||||
|
};
|
||||||
|
|
||||||
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual void AddRequestHeaders(HttpWebRequest webRequest, HttpHeader headers)
|
protected virtual void AddRequestHeaders(HttpRequestMessage webRequest, HttpHeader headers)
|
||||||
{
|
{
|
||||||
foreach (var header in headers)
|
foreach (var header in headers)
|
||||||
{
|
{
|
||||||
switch (header.Key)
|
switch (header.Key)
|
||||||
{
|
{
|
||||||
case "Accept":
|
case "Accept":
|
||||||
webRequest.Accept = header.Value;
|
webRequest.Headers.Accept.ParseAdd(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Connection":
|
case "Connection":
|
||||||
webRequest.Connection = header.Value;
|
webRequest.Headers.Connection.Clear();
|
||||||
|
webRequest.Headers.Connection.Add(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Content-Length":
|
case "Content-Length":
|
||||||
webRequest.ContentLength = Convert.ToInt64(header.Value);
|
webRequest.Headers.Add("Content-Length", header.Value);
|
||||||
break;
|
break;
|
||||||
case "Content-Type":
|
case "Content-Type":
|
||||||
webRequest.ContentType = header.Value;
|
webRequest.Headers.Remove("Content-Type");
|
||||||
|
webRequest.Headers.Add("Content-Type", header.Value);
|
||||||
break;
|
break;
|
||||||
case "Date":
|
case "Date":
|
||||||
webRequest.Date = HttpHeader.ParseDateTime(header.Value);
|
webRequest.Headers.Remove("Date");
|
||||||
|
webRequest.Headers.Date = HttpHeader.ParseDateTime(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Expect":
|
case "Expect":
|
||||||
webRequest.Expect = header.Value;
|
webRequest.Headers.Expect.ParseAdd(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Host":
|
case "Host":
|
||||||
webRequest.Host = header.Value;
|
webRequest.Headers.Host = header.Value;
|
||||||
break;
|
break;
|
||||||
case "If-Modified-Since":
|
case "If-Modified-Since":
|
||||||
webRequest.IfModifiedSince = HttpHeader.ParseDateTime(header.Value);
|
webRequest.Headers.IfModifiedSince = HttpHeader.ParseDateTime(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Referer":
|
case "Referer":
|
||||||
webRequest.Referer = header.Value;
|
webRequest.Headers.Add("Referer", header.Value);
|
||||||
break;
|
break;
|
||||||
case "Transfer-Encoding":
|
case "Transfer-Encoding":
|
||||||
webRequest.TransferEncoding = header.Value;
|
webRequest.Headers.TransferEncoding.ParseAdd(header.Value);
|
||||||
break;
|
break;
|
||||||
case "User-Agent":
|
case "User-Agent":
|
||||||
webRequest.UserAgent = header.Value;
|
webRequest.Headers.UserAgent.ParseAdd(header.Value);
|
||||||
break;
|
break;
|
||||||
case "Proxy-Connection":
|
case "Proxy-Connection":
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
@@ -240,5 +221,10 @@ namespace NzbDrone.Common.Http.Dispatchers
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private CredentialCache GetCredentialCache()
|
||||||
|
{
|
||||||
|
return _credentialCache.Get("credentialCache", () => new CredentialCache());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Net;
|
|
||||||
|
|
||||||
namespace NzbDrone.Common.Http
|
|
||||||
{
|
|
||||||
public class GZipWebClient : WebClient
|
|
||||||
{
|
|
||||||
protected override WebRequest GetWebRequest(Uri address)
|
|
||||||
{
|
|
||||||
var request = (HttpWebRequest)base.GetWebRequest(address);
|
|
||||||
request.AutomaticDecompression = DecompressionMethods.GZip;
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,8 +1,10 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
@@ -119,8 +121,6 @@ namespace NzbDrone.Common.Http
|
|||||||
|
|
||||||
var stopWatch = Stopwatch.StartNew();
|
var stopWatch = Stopwatch.StartNew();
|
||||||
|
|
||||||
PrepareRequestCookies(request, cookieContainer);
|
|
||||||
|
|
||||||
var response = _httpDispatcher.GetResponse(request, cookieContainer);
|
var response = _httpDispatcher.GetResponse(request, cookieContainer);
|
||||||
|
|
||||||
HandleResponseCookies(response, cookieContainer);
|
HandleResponseCookies(response, cookieContainer);
|
||||||
@@ -187,57 +187,98 @@ namespace NzbDrone.Common.Http
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void PrepareRequestCookies(HttpRequest request, CookieContainer cookieContainer)
|
private void HandleResponseCookies(HttpResponse response, CookieContainer container)
|
||||||
{
|
{
|
||||||
// Don't collect persistnet cookies for intermediate/redirected urls.
|
foreach (Cookie cookie in container.GetCookies((Uri)response.Request.Url))
|
||||||
/*lock (_cookieContainerCache)
|
|
||||||
{
|
{
|
||||||
var presistentContainer = _cookieContainerCache.Get("container", () => new CookieContainer());
|
cookie.Expired = true;
|
||||||
var persistentCookies = presistentContainer.GetCookies((Uri)request.Url);
|
}
|
||||||
var existingCookies = cookieContainer.GetCookies((Uri)request.Url);
|
|
||||||
|
|
||||||
cookieContainer.Add(persistentCookies);
|
|
||||||
cookieContainer.Add(existingCookies);
|
|
||||||
}*/
|
|
||||||
}
|
|
||||||
|
|
||||||
private void HandleResponseCookies(HttpResponse response, CookieContainer cookieContainer)
|
|
||||||
{
|
|
||||||
var cookieHeaders = response.GetCookieHeaders();
|
var cookieHeaders = response.GetCookieHeaders();
|
||||||
|
|
||||||
if (cookieHeaders.Empty())
|
if (cookieHeaders.Empty())
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AddCookiesToContainer(response.Request.Url, cookieHeaders, container);
|
||||||
|
|
||||||
if (response.Request.StoreResponseCookie)
|
if (response.Request.StoreResponseCookie)
|
||||||
{
|
{
|
||||||
lock (_cookieContainerCache)
|
lock (_cookieContainerCache)
|
||||||
{
|
{
|
||||||
var persistentCookieContainer = _cookieContainerCache.Get("container", () => new CookieContainer());
|
var persistentCookieContainer = _cookieContainerCache.Get("container", () => new CookieContainer());
|
||||||
|
|
||||||
foreach (var cookieHeader in cookieHeaders)
|
AddCookiesToContainer(response.Request.Url, cookieHeaders, persistentCookieContainer);
|
||||||
{
|
}
|
||||||
try
|
}
|
||||||
{
|
}
|
||||||
persistentCookieContainer.SetCookies((Uri)response.Request.Url, cookieHeader);
|
|
||||||
}
|
private void AddCookiesToContainer(HttpUri url, string[] cookieHeaders, CookieContainer container)
|
||||||
catch (Exception ex)
|
{
|
||||||
{
|
foreach (var cookieHeader in cookieHeaders)
|
||||||
_logger.Debug(ex, "Invalid cookie in {0}", response.Request.Url);
|
{
|
||||||
}
|
try
|
||||||
}
|
{
|
||||||
|
container.SetCookies((Uri)url, cookieHeader);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.Debug(ex, "Invalid cookie in {0}", url);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void DownloadFile(string url, string fileName, string userAgent = null)
|
public void DownloadFile(string url, string fileName, string userAgent = null)
|
||||||
{
|
{
|
||||||
_httpDispatcher.DownloadFile(url, fileName);
|
var fileNamePart = fileName + ".part";
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var fileInfo = new FileInfo(fileName);
|
||||||
|
if (fileInfo.Directory != null && !fileInfo.Directory.Exists)
|
||||||
|
{
|
||||||
|
fileInfo.Directory.Create();
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.Debug("Downloading [{0}] to [{1}]", url, fileName);
|
||||||
|
|
||||||
|
var stopWatch = Stopwatch.StartNew();
|
||||||
|
using (var fileStream = new FileStream(fileNamePart, FileMode.Create, FileAccess.ReadWrite))
|
||||||
|
{
|
||||||
|
var request = new HttpRequest(url);
|
||||||
|
request.AllowAutoRedirect = true;
|
||||||
|
request.ResponseStream = fileStream;
|
||||||
|
var response = Get(request);
|
||||||
|
|
||||||
|
if (response.Headers.ContentType != null && response.Headers.ContentType.Contains("text/html"))
|
||||||
|
{
|
||||||
|
throw new HttpException(request, response, "Site responded with html content.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stopWatch.Stop();
|
||||||
|
|
||||||
|
if (File.Exists(fileName))
|
||||||
|
{
|
||||||
|
File.Delete(fileName);
|
||||||
|
}
|
||||||
|
|
||||||
|
File.Move(fileNamePart, fileName);
|
||||||
|
_logger.Debug("Downloading Completed. took {0:0}s", stopWatch.Elapsed.Seconds);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (File.Exists(fileNamePart))
|
||||||
|
{
|
||||||
|
File.Delete(fileNamePart);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public HttpResponse Get(HttpRequest request)
|
public HttpResponse Get(HttpRequest request)
|
||||||
{
|
{
|
||||||
request.Method = HttpMethod.GET;
|
request.Method = HttpMethod.Get;
|
||||||
return Execute(request);
|
return Execute(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -251,13 +292,13 @@ namespace NzbDrone.Common.Http
|
|||||||
|
|
||||||
public HttpResponse Head(HttpRequest request)
|
public HttpResponse Head(HttpRequest request)
|
||||||
{
|
{
|
||||||
request.Method = HttpMethod.HEAD;
|
request.Method = HttpMethod.Head;
|
||||||
return Execute(request);
|
return Execute(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
public HttpResponse Post(HttpRequest request)
|
public HttpResponse Post(HttpRequest request)
|
||||||
{
|
{
|
||||||
request.Method = HttpMethod.POST;
|
request.Method = HttpMethod.Post;
|
||||||
return Execute(request);
|
return Execute(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,11 +4,27 @@ using System.Collections.Generic;
|
|||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
|
|
||||||
namespace NzbDrone.Common.Http
|
namespace NzbDrone.Common.Http
|
||||||
{
|
{
|
||||||
|
public static class WebHeaderCollectionExtensions
|
||||||
|
{
|
||||||
|
public static NameValueCollection ToNameValueCollection(this HttpHeaders headers)
|
||||||
|
{
|
||||||
|
var result = new NameValueCollection();
|
||||||
|
foreach (var header in headers)
|
||||||
|
{
|
||||||
|
result.Add(header.Key, header.Value.ConcatToString(";"));
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public class HttpHeader : NameValueCollection, IEnumerable<KeyValuePair<string, string>>, IEnumerable
|
public class HttpHeader : NameValueCollection, IEnumerable<KeyValuePair<string, string>>, IEnumerable
|
||||||
{
|
{
|
||||||
public HttpHeader(NameValueCollection headers)
|
public HttpHeader(NameValueCollection headers)
|
||||||
@@ -16,6 +32,11 @@ namespace NzbDrone.Common.Http
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public HttpHeader(HttpHeaders headers)
|
||||||
|
: base(headers.ToNameValueCollection())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
public HttpHeader()
|
public HttpHeader()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +0,0 @@
|
|||||||
namespace NzbDrone.Common.Http
|
|
||||||
{
|
|
||||||
public enum HttpMethod
|
|
||||||
{
|
|
||||||
GET,
|
|
||||||
POST,
|
|
||||||
PUT,
|
|
||||||
DELETE,
|
|
||||||
HEAD,
|
|
||||||
OPTIONS,
|
|
||||||
PATCH,
|
|
||||||
MERGE
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,6 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -12,6 +13,7 @@ namespace NzbDrone.Common.Http
|
|||||||
{
|
{
|
||||||
public HttpRequest(string url, HttpAccept httpAccept = null)
|
public HttpRequest(string url, HttpAccept httpAccept = null)
|
||||||
{
|
{
|
||||||
|
Method = HttpMethod.Get;
|
||||||
Url = new HttpUri(url);
|
Url = new HttpUri(url);
|
||||||
Headers = new HttpHeader();
|
Headers = new HttpHeader();
|
||||||
AllowAutoRedirect = true;
|
AllowAutoRedirect = true;
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ using System.Collections.Generic;
|
|||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
|
|
||||||
@@ -34,7 +35,7 @@ namespace NzbDrone.Common.Http
|
|||||||
{
|
{
|
||||||
BaseUrl = new HttpUri(baseUrl);
|
BaseUrl = new HttpUri(baseUrl);
|
||||||
ResourceUrl = string.Empty;
|
ResourceUrl = string.Empty;
|
||||||
Method = HttpMethod.GET;
|
Method = HttpMethod.Get;
|
||||||
QueryParams = new List<KeyValuePair<string, string>>();
|
QueryParams = new List<KeyValuePair<string, string>>();
|
||||||
SuffixQueryParams = new List<KeyValuePair<string, string>>();
|
SuffixQueryParams = new List<KeyValuePair<string, string>>();
|
||||||
Segments = new Dictionary<string, string>();
|
Segments = new Dictionary<string, string>();
|
||||||
@@ -264,7 +265,7 @@ namespace NzbDrone.Common.Http
|
|||||||
|
|
||||||
public virtual HttpRequestBuilder Post()
|
public virtual HttpRequestBuilder Post()
|
||||||
{
|
{
|
||||||
Method = HttpMethod.POST;
|
Method = HttpMethod.Post;
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@@ -355,7 +356,7 @@ namespace NzbDrone.Common.Http
|
|||||||
|
|
||||||
public virtual HttpRequestBuilder AddFormParameter(string key, object value)
|
public virtual HttpRequestBuilder AddFormParameter(string key, object value)
|
||||||
{
|
{
|
||||||
if (Method != HttpMethod.POST)
|
if (Method != HttpMethod.Post)
|
||||||
{
|
{
|
||||||
throw new NotSupportedException("HttpRequest Method must be POST to add FormParameter.");
|
throw new NotSupportedException("HttpRequest Method must be POST to add FormParameter.");
|
||||||
}
|
}
|
||||||
@@ -371,7 +372,7 @@ namespace NzbDrone.Common.Http
|
|||||||
|
|
||||||
public virtual HttpRequestBuilder AddFormUpload(string name, string fileName, byte[] data, string contentType = "application/octet-stream")
|
public virtual HttpRequestBuilder AddFormUpload(string name, string fileName, byte[] data, string contentType = "application/octet-stream")
|
||||||
{
|
{
|
||||||
if (Method != HttpMethod.POST)
|
if (Method != HttpMethod.Post)
|
||||||
{
|
{
|
||||||
throw new NotSupportedException("HttpRequest Method must be POST to add FormUpload.");
|
throw new NotSupportedException("HttpRequest Method must be POST to add FormUpload.");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using NzbDrone.Common.Serializer;
|
using NzbDrone.Common.Serializer;
|
||||||
|
|
||||||
@@ -17,14 +18,14 @@ namespace NzbDrone.Common.Http
|
|||||||
public JsonRpcRequestBuilder(string baseUrl)
|
public JsonRpcRequestBuilder(string baseUrl)
|
||||||
: base(baseUrl)
|
: base(baseUrl)
|
||||||
{
|
{
|
||||||
Method = HttpMethod.POST;
|
Method = HttpMethod.Post;
|
||||||
JsonParameters = new List<object>();
|
JsonParameters = new List<object>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public JsonRpcRequestBuilder(string baseUrl, string method, IEnumerable<object> parameters)
|
public JsonRpcRequestBuilder(string baseUrl, string method, IEnumerable<object> parameters)
|
||||||
: base(baseUrl)
|
: base(baseUrl)
|
||||||
{
|
{
|
||||||
Method = HttpMethod.POST;
|
Method = HttpMethod.Post;
|
||||||
JsonMethod = method;
|
JsonMethod = method;
|
||||||
JsonParameters = parameters.ToList();
|
JsonParameters = parameters.ToList();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ namespace NzbDrone.Core.Test.Framework
|
|||||||
|
|
||||||
Mocker.SetConstant<IHttpProxySettingsProvider>(new HttpProxySettingsProvider(Mocker.Resolve<ConfigService>()));
|
Mocker.SetConstant<IHttpProxySettingsProvider>(new HttpProxySettingsProvider(Mocker.Resolve<ConfigService>()));
|
||||||
Mocker.SetConstant<ICreateManagedWebProxy>(new ManagedWebProxyFactory(Mocker.Resolve<CacheManager>()));
|
Mocker.SetConstant<ICreateManagedWebProxy>(new ManagedWebProxyFactory(Mocker.Resolve<CacheManager>()));
|
||||||
Mocker.SetConstant<IHttpDispatcher>(new ManagedHttpDispatcher(Mocker.Resolve<IHttpProxySettingsProvider>(), Mocker.Resolve<ICreateManagedWebProxy>(), Mocker.Resolve<UserAgentBuilder>(), Mocker.Resolve<IPlatformInfo>(), TestLogger));
|
Mocker.SetConstant<IHttpDispatcher>(new ManagedHttpDispatcher(Mocker.Resolve<IHttpProxySettingsProvider>(), Mocker.Resolve<ICreateManagedWebProxy>(), Mocker.Resolve<UserAgentBuilder>(), Mocker.Resolve<CacheManager>(), TestLogger));
|
||||||
Mocker.SetConstant<IHttpClient>(new HttpClient(new IHttpRequestInterceptor[0], Mocker.Resolve<CacheManager>(), Mocker.Resolve<RateLimitService>(), Mocker.Resolve<IHttpDispatcher>(), TestLogger));
|
Mocker.SetConstant<IHttpClient>(new HttpClient(new IHttpRequestInterceptor[0], Mocker.Resolve<CacheManager>(), Mocker.Resolve<RateLimitService>(), Mocker.Resolve<IHttpDispatcher>(), TestLogger));
|
||||||
Mocker.SetConstant<IReadarrCloudRequestBuilder>(new ReadarrCloudRequestBuilder());
|
Mocker.SetConstant<IReadarrCloudRequestBuilder>(new ReadarrCloudRequestBuilder());
|
||||||
Mocker.SetConstant<IMetadataRequestBuilder>(Mocker.Resolve<MetadataRequestBuilder>());
|
Mocker.SetConstant<IMetadataRequestBuilder>(Mocker.Resolve<MetadataRequestBuilder>());
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -30,7 +31,7 @@ namespace NzbDrone.Core.Test.IndexerTests.FileListTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/FileList/RecentFeed.json");
|
var recentFeed = ReadAllText(@"Files/Indexers/FileList/RecentFeed.json");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -35,15 +36,15 @@ namespace NzbDrone.Core.Test.IndexerTests.GazelleTests
|
|||||||
var indexFeed = ReadAllText(@"Files/Indexers/Gazelle/GazelleIndex.json");
|
var indexFeed = ReadAllText(@"Files/Indexers/Gazelle/GazelleIndex.json");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET && v.Url.FullUri.Contains("ajax.php?action=browse"))))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get && v.Url.FullUri.Contains("ajax.php?action=browse"))))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader { ContentType = "application/json" }, recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader { ContentType = "application/json" }, recentFeed));
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.POST && v.Url.FullUri.Contains("ajax.php?action=index"))))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Post && v.Url.FullUri.Contains("ajax.php?action=index"))))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), indexFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), indexFeed));
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.POST && v.Url.FullUri.Contains("login.php"))))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Post && v.Url.FullUri.Contains("login.php"))))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), indexFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), indexFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -88,7 +89,7 @@ namespace NzbDrone.Core.Test.IndexerTests.IPTorrentsTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/IPTorrents/IPTorrents.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/IPTorrents/IPTorrents.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -43,7 +44,7 @@ namespace NzbDrone.Core.Test.IndexerTests.NewznabTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Newznab/newznab_nzb_su.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Newznab/newznab_nzb_su.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -30,7 +31,7 @@ namespace NzbDrone.Core.Test.IndexerTests.NyaaTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Nyaa/Nyaa.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Nyaa/Nyaa.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -33,7 +34,7 @@ namespace NzbDrone.Core.Test.IndexerTests.OmgwtfnzbsTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Omgwtfnzbs/Omgwtfnzbs.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Omgwtfnzbs/Omgwtfnzbs.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -35,7 +36,7 @@ namespace NzbDrone.Core.Test.IndexerTests.RarbgTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Rarbg/RecentFeed_v2.json");
|
var recentFeed = ReadAllText(@"Files/Indexers/Rarbg/RecentFeed_v2.json");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
@@ -62,7 +63,7 @@ namespace NzbDrone.Core.Test.IndexerTests.RarbgTests
|
|||||||
public void should_parse_error_20_as_empty_results()
|
public void should_parse_error_20_as_empty_results()
|
||||||
{
|
{
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), "{ error_code: 20, error: \"some message\" }"));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), "{ error_code: 20, error: \"some message\" }"));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
@@ -74,7 +75,7 @@ namespace NzbDrone.Core.Test.IndexerTests.RarbgTests
|
|||||||
public void should_warn_on_unknown_error()
|
public void should_warn_on_unknown_error()
|
||||||
{
|
{
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), "{ error_code: 25, error: \"some message\" }"));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), "{ error_code: 25, error: \"some message\" }"));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -30,7 +31,7 @@ namespace NzbDrone.Core.Test.IndexerTests.TorrentleechTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Torrentleech/Torrentleech.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Torrentleech/Torrentleech.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
@@ -42,7 +43,7 @@ namespace NzbDrone.Core.Test.IndexerTests.TorznabTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Torznab/torznab_hdaccess_net.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Torznab/torznab_hdaccess_net.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
@@ -71,7 +72,7 @@ namespace NzbDrone.Core.Test.IndexerTests.TorznabTests
|
|||||||
var recentFeed = ReadAllText(@"Files/Indexers/Torznab/torznab_tpb.xml");
|
var recentFeed = ReadAllText(@"Files/Indexers/Torznab/torznab_tpb.xml");
|
||||||
|
|
||||||
Mocker.GetMock<IHttpClient>()
|
Mocker.GetMock<IHttpClient>()
|
||||||
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.GET)))
|
.Setup(o => o.Execute(It.Is<HttpRequest>(v => v.Method == HttpMethod.Get)))
|
||||||
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
.Returns<HttpRequest>(r => new HttpResponse(r, new HttpHeader(), recentFeed));
|
||||||
|
|
||||||
var releases = Subject.FetchRecent();
|
var releases = Subject.FetchRecent();
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
@@ -142,15 +143,19 @@ namespace NzbDrone.Core.Download.Clients.DownloadStation.Proxies
|
|||||||
return authResponse.Data.SId;
|
return authResponse.Data.SId;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected HttpRequestBuilder BuildRequest(DownloadStationSettings settings, string methodName, int apiVersion, HttpMethod httpVerb = HttpMethod.GET)
|
protected HttpRequestBuilder BuildRequest(DownloadStationSettings settings, string methodName, int apiVersion, HttpMethod httpVerb = null)
|
||||||
{
|
{
|
||||||
|
httpVerb ??= HttpMethod.Get;
|
||||||
|
|
||||||
var info = GetApiInfo(_apiType, settings);
|
var info = GetApiInfo(_apiType, settings);
|
||||||
|
|
||||||
return BuildRequest(settings, info, methodName, apiVersion, httpVerb);
|
return BuildRequest(settings, info, methodName, apiVersion, httpVerb);
|
||||||
}
|
}
|
||||||
|
|
||||||
private HttpRequestBuilder BuildRequest(DownloadStationSettings settings, DiskStationApiInfo apiInfo, string methodName, int apiVersion, HttpMethod httpVerb = HttpMethod.GET)
|
private HttpRequestBuilder BuildRequest(DownloadStationSettings settings, DiskStationApiInfo apiInfo, string methodName, int apiVersion, HttpMethod httpVerb = null)
|
||||||
{
|
{
|
||||||
|
httpVerb ??= HttpMethod.Get;
|
||||||
|
|
||||||
var requestBuilder = new HttpRequestBuilder(settings.UseSsl, settings.Host, settings.Port).Resource($"webapi/{apiInfo.Path}");
|
var requestBuilder = new HttpRequestBuilder(settings.UseSsl, settings.Host, settings.Port).Resource($"webapi/{apiInfo.Path}");
|
||||||
requestBuilder.Method = httpVerb;
|
requestBuilder.Method = httpVerb;
|
||||||
requestBuilder.LogResponseContent = true;
|
requestBuilder.LogResponseContent = true;
|
||||||
@@ -163,7 +168,7 @@ namespace NzbDrone.Core.Download.Clients.DownloadStation.Proxies
|
|||||||
throw new ArgumentOutOfRangeException(nameof(apiVersion));
|
throw new ArgumentOutOfRangeException(nameof(apiVersion));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (httpVerb == HttpMethod.POST)
|
if (httpVerb == HttpMethod.Post)
|
||||||
{
|
{
|
||||||
if (apiInfo.NeedsAuthentication)
|
if (apiInfo.NeedsAuthentication)
|
||||||
{
|
{
|
||||||
|
|||||||
+3
-2
@@ -1,4 +1,5 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -21,7 +22,7 @@ namespace NzbDrone.Core.Download.Clients.DownloadStation.Proxies
|
|||||||
|
|
||||||
public void AddTaskFromData(byte[] data, string filename, string downloadDirectory, DownloadStationSettings settings)
|
public void AddTaskFromData(byte[] data, string filename, string downloadDirectory, DownloadStationSettings settings)
|
||||||
{
|
{
|
||||||
var requestBuilder = BuildRequest(settings, "create", 2, HttpMethod.POST);
|
var requestBuilder = BuildRequest(settings, "create", 2, HttpMethod.Post);
|
||||||
|
|
||||||
if (downloadDirectory.IsNotNullOrWhiteSpace())
|
if (downloadDirectory.IsNotNullOrWhiteSpace())
|
||||||
{
|
{
|
||||||
|
|||||||
+2
-1
@@ -1,5 +1,6 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -22,7 +23,7 @@ namespace NzbDrone.Core.Download.Clients.DownloadStation.Proxies
|
|||||||
|
|
||||||
public void AddTaskFromData(byte[] data, string filename, string downloadDirectory, DownloadStationSettings settings)
|
public void AddTaskFromData(byte[] data, string filename, string downloadDirectory, DownloadStationSettings settings)
|
||||||
{
|
{
|
||||||
var requestBuilder = BuildRequest(settings, "create", 2, HttpMethod.POST);
|
var requestBuilder = BuildRequest(settings, "create", 2, HttpMethod.Post);
|
||||||
|
|
||||||
requestBuilder.AddFormParameter("type", "\"file\"");
|
requestBuilder.AddFormParameter("type", "\"file\"");
|
||||||
requestBuilder.AddFormParameter("file", "[\"fileData\"]");
|
requestBuilder.AddFormParameter("file", "[\"fileData\"]");
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
@@ -107,7 +108,7 @@ namespace NzbDrone.Core.Download.Clients.Flood
|
|||||||
{
|
{
|
||||||
var verifyRequest = BuildRequest(settings).Resource("/auth/verify").Build();
|
var verifyRequest = BuildRequest(settings).Resource("/auth/verify").Build();
|
||||||
|
|
||||||
verifyRequest.Method = HttpMethod.GET;
|
verifyRequest.Method = HttpMethod.Get;
|
||||||
|
|
||||||
HandleRequest(verifyRequest, settings);
|
HandleRequest(verifyRequest, settings);
|
||||||
}
|
}
|
||||||
@@ -180,7 +181,7 @@ namespace NzbDrone.Core.Download.Clients.Flood
|
|||||||
{
|
{
|
||||||
var getTorrentsRequest = BuildRequest(settings).Resource("/torrents").Build();
|
var getTorrentsRequest = BuildRequest(settings).Resource("/torrents").Build();
|
||||||
|
|
||||||
getTorrentsRequest.Method = HttpMethod.GET;
|
getTorrentsRequest.Method = HttpMethod.Get;
|
||||||
|
|
||||||
return Json.Deserialize<TorrentListSummary>(HandleRequest(getTorrentsRequest, settings).Content).Torrents;
|
return Json.Deserialize<TorrentListSummary>(HandleRequest(getTorrentsRequest, settings).Content).Torrents;
|
||||||
}
|
}
|
||||||
@@ -189,7 +190,7 @@ namespace NzbDrone.Core.Download.Clients.Flood
|
|||||||
{
|
{
|
||||||
var contentsRequest = BuildRequest(settings).Resource($"/torrents/{hash}/contents").Build();
|
var contentsRequest = BuildRequest(settings).Resource($"/torrents/{hash}/contents").Build();
|
||||||
|
|
||||||
contentsRequest.Method = HttpMethod.GET;
|
contentsRequest.Method = HttpMethod.Get;
|
||||||
|
|
||||||
return Json.Deserialize<List<TorrentContent>>(HandleRequest(contentsRequest, settings).Content).ConvertAll(content => content.Path);
|
return Json.Deserialize<List<TorrentContent>>(HandleRequest(contentsRequest, settings).Content).ConvertAll(content => content.Path);
|
||||||
}
|
}
|
||||||
@@ -198,7 +199,7 @@ namespace NzbDrone.Core.Download.Clients.Flood
|
|||||||
{
|
{
|
||||||
var tagsRequest = BuildRequest(settings).Resource("/torrents/tags").Build();
|
var tagsRequest = BuildRequest(settings).Resource("/torrents/tags").Build();
|
||||||
|
|
||||||
tagsRequest.Method = HttpMethod.PATCH;
|
tagsRequest.Method = HttpMethod.Patch;
|
||||||
|
|
||||||
var body = new Dictionary<string, object>
|
var body = new Dictionary<string, object>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
using System.Xml.Linq;
|
using System.Xml.Linq;
|
||||||
using System.Xml.XPath;
|
using System.Xml.XPath;
|
||||||
@@ -142,7 +143,7 @@ namespace NzbDrone.Core.ImportLists.Goodreads
|
|||||||
{
|
{
|
||||||
var request = new Common.Http.HttpRequest(Settings.SigningUrl)
|
var request = new Common.Http.HttpRequest(Settings.SigningUrl)
|
||||||
{
|
{
|
||||||
Method = HttpMethod.POST,
|
Method = HttpMethod.Post,
|
||||||
};
|
};
|
||||||
request.Headers.Set("Content-Type", "application/json");
|
request.Headers.Set("Content-Type", "application/json");
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Cache;
|
using NzbDrone.Common.Cache;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -71,7 +72,7 @@ namespace NzbDrone.Core.Indexers.Gazelle
|
|||||||
};
|
};
|
||||||
|
|
||||||
indexRequestBuilder.SetCookies(cookies);
|
indexRequestBuilder.SetCookies(cookies);
|
||||||
indexRequestBuilder.Method = HttpMethod.POST;
|
indexRequestBuilder.Method = HttpMethod.Post;
|
||||||
indexRequestBuilder.Resource("ajax.php?action=index");
|
indexRequestBuilder.Resource("ajax.php?action=index");
|
||||||
|
|
||||||
var authIndexRequest = indexRequestBuilder
|
var authIndexRequest = indexRequestBuilder
|
||||||
@@ -92,7 +93,7 @@ namespace NzbDrone.Core.Indexers.Gazelle
|
|||||||
LogResponseContent = true
|
LogResponseContent = true
|
||||||
};
|
};
|
||||||
|
|
||||||
requestBuilder.Method = HttpMethod.POST;
|
requestBuilder.Method = HttpMethod.Post;
|
||||||
requestBuilder.Resource("login.php");
|
requestBuilder.Resource("login.php");
|
||||||
requestBuilder.PostProcess += r => r.RequestTimeout = TimeSpan.FromSeconds(15);
|
requestBuilder.PostProcess += r => r.RequestTimeout = TimeSpan.FromSeconds(15);
|
||||||
|
|
||||||
|
|||||||
@@ -47,11 +47,6 @@ namespace NzbDrone.Core.Messaging.Commands
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (ThreadAbortException ex)
|
|
||||||
{
|
|
||||||
_logger.Error(ex, "Thread aborted");
|
|
||||||
Thread.ResetAbort();
|
|
||||||
}
|
|
||||||
catch (OperationCanceledException)
|
catch (OperationCanceledException)
|
||||||
{
|
{
|
||||||
_logger.Trace("Stopped one command execution pipeline");
|
_logger.Trace("Stopped one command execution pipeline");
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
using NzbDrone.Common.Serializer;
|
using NzbDrone.Common.Serializer;
|
||||||
@@ -29,7 +30,7 @@ namespace NzbDrone.Core.Notifications.Discord
|
|||||||
.Accept(HttpAccept.Json)
|
.Accept(HttpAccept.Json)
|
||||||
.Build();
|
.Build();
|
||||||
|
|
||||||
request.Method = HttpMethod.POST;
|
request.Method = HttpMethod.Post;
|
||||||
request.Headers.ContentType = "application/json";
|
request.Headers.ContentType = "application/json";
|
||||||
request.SetContent(payload.ToJson());
|
request.SetContent(payload.ToJson());
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
using System.Xml.Linq;
|
using System.Xml.Linq;
|
||||||
@@ -112,11 +113,11 @@ namespace NzbDrone.Core.Notifications.Goodreads
|
|||||||
// we need the url without the query to sign
|
// we need the url without the query to sign
|
||||||
auth.RequestUrl = request.Url.SetQuery(null).FullUri;
|
auth.RequestUrl = request.Url.SetQuery(null).FullUri;
|
||||||
|
|
||||||
if (builder.Method == HttpMethod.GET)
|
if (builder.Method == HttpMethod.Get)
|
||||||
{
|
{
|
||||||
auth.Parameters = builder.QueryParams.ToDictionary(x => x.Key, x => x.Value);
|
auth.Parameters = builder.QueryParams.ToDictionary(x => x.Key, x => x.Value);
|
||||||
}
|
}
|
||||||
else if (builder.Method == HttpMethod.POST)
|
else if (builder.Method == HttpMethod.Post)
|
||||||
{
|
{
|
||||||
auth.Parameters = builder.FormData.ToDictionary(x => x.Name, x => Encoding.UTF8.GetString(x.ContentData));
|
auth.Parameters = builder.FormData.ToDictionary(x => x.Name, x => Encoding.UTF8.GetString(x.ContentData));
|
||||||
}
|
}
|
||||||
@@ -172,7 +173,7 @@ namespace NzbDrone.Core.Notifications.Goodreads
|
|||||||
{
|
{
|
||||||
var request = new Common.Http.HttpRequest(Settings.SigningUrl)
|
var request = new Common.Http.HttpRequest(Settings.SigningUrl)
|
||||||
{
|
{
|
||||||
Method = HttpMethod.POST,
|
Method = HttpMethod.Post,
|
||||||
};
|
};
|
||||||
request.Headers.Set("Content-Type", "application/json");
|
request.Headers.Set("Content-Type", "application/json");
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentValidation.Results;
|
using FluentValidation.Results;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -27,7 +28,7 @@ namespace NzbDrone.Core.Notifications.Join
|
|||||||
|
|
||||||
public void SendNotification(string title, string message, JoinSettings settings)
|
public void SendNotification(string title, string message, JoinSettings settings)
|
||||||
{
|
{
|
||||||
var method = HttpMethod.GET;
|
var method = HttpMethod.Get;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
using HttpMethod = NzbDrone.Common.Http.HttpMethod;
|
|
||||||
|
|
||||||
namespace NzbDrone.Core.Notifications.Mailgun
|
namespace NzbDrone.Core.Notifications.Mailgun
|
||||||
{
|
{
|
||||||
@@ -27,7 +27,7 @@ namespace NzbDrone.Core.Notifications.Mailgun
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var request = BuildRequest(settings, $"{settings.SenderDomain}/messages", HttpMethod.POST, title, message).Build();
|
var request = BuildRequest(settings, $"{settings.SenderDomain}/messages", HttpMethod.Post, title, message).Build();
|
||||||
_httpClient.Execute(request);
|
_httpClient.Execute(request);
|
||||||
}
|
}
|
||||||
catch (HttpException ex)
|
catch (HttpException ex)
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using FluentValidation.Results;
|
using FluentValidation.Results;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -100,7 +101,7 @@ namespace NzbDrone.Core.Notifications.PushBullet
|
|||||||
|
|
||||||
var request = requestBuilder.Build();
|
var request = requestBuilder.Build();
|
||||||
|
|
||||||
request.Method = HttpMethod.GET;
|
request.Method = HttpMethod.Get;
|
||||||
request.Credentials = new BasicNetworkCredential(settings.ApiKey, string.Empty);
|
request.Credentials = new BasicNetworkCredential(settings.ApiKey, string.Empty);
|
||||||
|
|
||||||
var response = _httpClient.Execute(request);
|
var response = _httpClient.Execute(request);
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
using NzbDrone.Common.Serializer;
|
using NzbDrone.Common.Serializer;
|
||||||
|
|
||||||
@@ -22,7 +23,7 @@ namespace NzbDrone.Core.Notifications.SendGrid
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var request = BuildRequest(settings, "mail/send", HttpMethod.POST);
|
var request = BuildRequest(settings, "mail/send", HttpMethod.Post);
|
||||||
|
|
||||||
var payload = new SendGridPayload
|
var payload = new SendGridPayload
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
using System.Net.Http;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
using NzbDrone.Common.Serializer;
|
using NzbDrone.Common.Serializer;
|
||||||
@@ -29,7 +30,7 @@ namespace NzbDrone.Core.Notifications.Slack
|
|||||||
.Accept(HttpAccept.Json)
|
.Accept(HttpAccept.Json)
|
||||||
.Build();
|
.Build();
|
||||||
|
|
||||||
request.Method = HttpMethod.POST;
|
request.Method = HttpMethod.Post;
|
||||||
request.Headers.ContentType = "application/json";
|
request.Headers.ContentType = "application/json";
|
||||||
request.SetContent(payload.ToJson());
|
request.SetContent(payload.ToJson());
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Xml.Linq;
|
using System.Xml.Linq;
|
||||||
using NLog;
|
using NLog;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
@@ -36,7 +37,7 @@ namespace NzbDrone.Core.Notifications.Subsonic
|
|||||||
public void Notify(SubsonicSettings settings, string message)
|
public void Notify(SubsonicSettings settings, string message)
|
||||||
{
|
{
|
||||||
var resource = "addChatMessage";
|
var resource = "addChatMessage";
|
||||||
var request = GetSubsonicServerRequest(resource, HttpMethod.GET, settings);
|
var request = GetSubsonicServerRequest(resource, HttpMethod.Get, settings);
|
||||||
request.AddQueryParam("message", message);
|
request.AddQueryParam("message", message);
|
||||||
|
|
||||||
var response = _httpClient.Execute(request.Build());
|
var response = _httpClient.Execute(request.Build());
|
||||||
@@ -48,7 +49,7 @@ namespace NzbDrone.Core.Notifications.Subsonic
|
|||||||
public void Update(SubsonicSettings settings)
|
public void Update(SubsonicSettings settings)
|
||||||
{
|
{
|
||||||
var resource = "startScan";
|
var resource = "startScan";
|
||||||
var request = GetSubsonicServerRequest(resource, HttpMethod.GET, settings);
|
var request = GetSubsonicServerRequest(resource, HttpMethod.Get, settings);
|
||||||
var response = _httpClient.Execute(request.Build());
|
var response = _httpClient.Execute(request.Build());
|
||||||
|
|
||||||
_logger.Trace("Update response: {0}", response.Content);
|
_logger.Trace("Update response: {0}", response.Content);
|
||||||
@@ -57,7 +58,7 @@ namespace NzbDrone.Core.Notifications.Subsonic
|
|||||||
|
|
||||||
public string Version(SubsonicSettings settings)
|
public string Version(SubsonicSettings settings)
|
||||||
{
|
{
|
||||||
var request = GetSubsonicServerRequest("ping", HttpMethod.GET, settings);
|
var request = GetSubsonicServerRequest("ping", HttpMethod.Get, settings);
|
||||||
var response = _httpClient.Execute(request.Build());
|
var response = _httpClient.Execute(request.Build());
|
||||||
|
|
||||||
_logger.Trace("Version response: {0}", response.Content);
|
_logger.Trace("Version response: {0}", response.Content);
|
||||||
|
|||||||
@@ -1,10 +1,8 @@
|
|||||||
using NzbDrone.Common.Http;
|
|
||||||
|
|
||||||
namespace NzbDrone.Core.Notifications.Webhook
|
namespace NzbDrone.Core.Notifications.Webhook
|
||||||
{
|
{
|
||||||
public enum WebhookMethod
|
public enum WebhookMethod
|
||||||
{
|
{
|
||||||
POST = HttpMethod.POST,
|
POST = 1,
|
||||||
PUT = HttpMethod.PUT
|
PUT = 2
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using System.Net;
|
using System;
|
||||||
|
using System.Net.Http;
|
||||||
using NzbDrone.Common.Extensions;
|
using NzbDrone.Common.Extensions;
|
||||||
using NzbDrone.Common.Http;
|
using NzbDrone.Common.Http;
|
||||||
using NzbDrone.Common.Serializer;
|
using NzbDrone.Common.Serializer;
|
||||||
@@ -27,7 +28,13 @@ namespace NzbDrone.Core.Notifications.Webhook
|
|||||||
.Accept(HttpAccept.Json)
|
.Accept(HttpAccept.Json)
|
||||||
.Build();
|
.Build();
|
||||||
|
|
||||||
request.Method = (HttpMethod)settings.Method;
|
request.Method = settings.Method switch
|
||||||
|
{
|
||||||
|
(int)WebhookMethod.POST => HttpMethod.Post,
|
||||||
|
(int)WebhookMethod.PUT => HttpMethod.Put,
|
||||||
|
_ => throw new ArgumentOutOfRangeException($"Invalid Webhook method {settings.Method}")
|
||||||
|
};
|
||||||
|
|
||||||
request.Headers.ContentType = "application/json";
|
request.Headers.ContentType = "application/json";
|
||||||
request.SetContent(body.ToJson());
|
request.SetContent(body.ToJson());
|
||||||
|
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ using System.Collections.Generic;
|
|||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
using System.Security.Cryptography;
|
using System.Security.Cryptography;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
@@ -37,7 +38,7 @@ namespace TinyTwitter
|
|||||||
|
|
||||||
public void UpdateStatus(string message)
|
public void UpdateStatus(string message)
|
||||||
{
|
{
|
||||||
new RequestBuilder(_oauth, "POST", "https://api.twitter.com/1.1/statuses/update.json")
|
new RequestBuilder(_oauth, HttpMethod.Post, "https://api.twitter.com/1.1/statuses/update.json")
|
||||||
.AddParameter("status", message)
|
.AddParameter("status", message)
|
||||||
.Execute();
|
.Execute();
|
||||||
}
|
}
|
||||||
@@ -51,7 +52,7 @@ namespace TinyTwitter
|
|||||||
**/
|
**/
|
||||||
public void DirectMessage(string message, string screenName)
|
public void DirectMessage(string message, string screenName)
|
||||||
{
|
{
|
||||||
new RequestBuilder(_oauth, "POST", "https://api.twitter.com/1.1/direct_messages/new.json")
|
new RequestBuilder(_oauth, HttpMethod.Post, "https://api.twitter.com/1.1/direct_messages/new.json")
|
||||||
.AddParameter("text", message)
|
.AddParameter("text", message)
|
||||||
.AddParameter("screen_name", screenName)
|
.AddParameter("screen_name", screenName)
|
||||||
.Execute();
|
.Execute();
|
||||||
@@ -63,16 +64,18 @@ namespace TinyTwitter
|
|||||||
private const string SIGNATURE_METHOD = "HMAC-SHA1";
|
private const string SIGNATURE_METHOD = "HMAC-SHA1";
|
||||||
|
|
||||||
private readonly OAuthInfo _oauth;
|
private readonly OAuthInfo _oauth;
|
||||||
private readonly string _method;
|
private readonly HttpMethod _method;
|
||||||
private readonly IDictionary<string, string> _customParameters;
|
private readonly IDictionary<string, string> _customParameters;
|
||||||
private readonly string _url;
|
private readonly string _url;
|
||||||
|
private readonly HttpClient _httpClient;
|
||||||
|
|
||||||
public RequestBuilder(OAuthInfo oauth, string method, string url)
|
public RequestBuilder(OAuthInfo oauth, HttpMethod method, string url)
|
||||||
{
|
{
|
||||||
_oauth = oauth;
|
_oauth = oauth;
|
||||||
_method = method;
|
_method = method;
|
||||||
_url = url;
|
_url = url;
|
||||||
_customParameters = new Dictionary<string, string>();
|
_customParameters = new Dictionary<string, string>();
|
||||||
|
_httpClient = new ();
|
||||||
}
|
}
|
||||||
|
|
||||||
public RequestBuilder AddParameter(string name, string value)
|
public RequestBuilder AddParameter(string name, string value)
|
||||||
@@ -92,61 +95,13 @@ namespace TinyTwitter
|
|||||||
var signature = GenerateSignature(parameters);
|
var signature = GenerateSignature(parameters);
|
||||||
var headerValue = GenerateAuthorizationHeaderValue(parameters, signature);
|
var headerValue = GenerateAuthorizationHeaderValue(parameters, signature);
|
||||||
|
|
||||||
var request = (HttpWebRequest)WebRequest.Create(GetRequestUrl());
|
var request = new HttpRequestMessage(_method, _url);
|
||||||
request.Method = _method;
|
request.Content = new FormUrlEncodedContent(_customParameters);
|
||||||
request.ContentType = "application/x-www-form-urlencoded";
|
|
||||||
|
|
||||||
request.Headers.Add("Authorization", headerValue);
|
request.Headers.Add("Authorization", headerValue);
|
||||||
|
|
||||||
WriteRequestBody(request);
|
var response = _httpClient.Send(request);
|
||||||
|
return response.Content.ReadAsStringAsync().GetAwaiter().GetResult();
|
||||||
// It looks like a bug in HttpWebRequest. It throws random TimeoutExceptions
|
|
||||||
// after some requests. Abort the request seems to work. More info:
|
|
||||||
// http://stackoverflow.com/questions/2252762/getrequeststream-throws-timeout-exception-randomly
|
|
||||||
var response = request.GetResponse();
|
|
||||||
|
|
||||||
string content;
|
|
||||||
|
|
||||||
using (var stream = response.GetResponseStream())
|
|
||||||
{
|
|
||||||
using (var reader = new StreamReader(stream))
|
|
||||||
{
|
|
||||||
content = reader.ReadToEnd();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
request.Abort();
|
|
||||||
|
|
||||||
return content;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void WriteRequestBody(HttpWebRequest request)
|
|
||||||
{
|
|
||||||
if (_method == "GET")
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var requestBody = Encoding.ASCII.GetBytes(GetCustomParametersString());
|
|
||||||
using (var stream = request.GetRequestStream())
|
|
||||||
{
|
|
||||||
stream.Write(requestBody, 0, requestBody.Length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private string GetRequestUrl()
|
|
||||||
{
|
|
||||||
if (_method != "GET" || _customParameters.Count == 0)
|
|
||||||
{
|
|
||||||
return _url;
|
|
||||||
}
|
|
||||||
|
|
||||||
return string.Format("{0}?{1}", _url, GetCustomParametersString());
|
|
||||||
}
|
|
||||||
|
|
||||||
private string GetCustomParametersString()
|
|
||||||
{
|
|
||||||
return _customParameters.Select(x => string.Format("{0}={1}", x.Key, x.Value)).Join("&");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private string GenerateAuthorizationHeaderValue(IEnumerable<KeyValuePair<string, string>> parameters, string signature)
|
private string GenerateAuthorizationHeaderValue(IEnumerable<KeyValuePair<string, string>> parameters, string signature)
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
using System.Linq;
|
using System;
|
||||||
using System.Net;
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
using FluentAssertions;
|
using FluentAssertions;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
|
|
||||||
@@ -8,25 +9,30 @@ namespace NzbDrone.Integration.Test
|
|||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class IndexHtmlFixture : IntegrationTest
|
public class IndexHtmlFixture : IntegrationTest
|
||||||
{
|
{
|
||||||
|
private HttpClient _httpClient = new HttpClient();
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void should_get_index_html()
|
public void should_get_index_html()
|
||||||
{
|
{
|
||||||
var text = new WebClient().DownloadString(RootUrl);
|
var request = new HttpRequestMessage(HttpMethod.Get, RootUrl);
|
||||||
|
var response = _httpClient.Send(request);
|
||||||
|
var text = response.Content.ReadAsStringAsync().GetAwaiter().GetResult();
|
||||||
text.Should().NotBeNullOrWhiteSpace();
|
text.Should().NotBeNullOrWhiteSpace();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void index_should_not_be_cached()
|
public void index_should_not_be_cached()
|
||||||
{
|
{
|
||||||
var client = new WebClient();
|
var request = new HttpRequestMessage(HttpMethod.Get, RootUrl);
|
||||||
_ = client.DownloadString(RootUrl);
|
var response = _httpClient.Send(request);
|
||||||
|
|
||||||
var headers = client.ResponseHeaders;
|
var headers = response.Headers;
|
||||||
|
|
||||||
headers.Get("Cache-Control").Split(',').Select(x => x.Trim())
|
headers.CacheControl.NoStore.Should().BeTrue();
|
||||||
.Should().BeEquivalentTo("no-store, no-cache".Split(',').Select(x => x.Trim()));
|
headers.CacheControl.NoCache.Should().BeTrue();
|
||||||
headers.Get("Pragma").Should().Be("no-cache");
|
headers.Pragma.Should().Contain(new NameValueHeaderValue("no-cache"));
|
||||||
headers.Get("Expires").Should().Be("-1");
|
|
||||||
|
response.Content.Headers.Expires.Should().BeBefore(DateTime.UtcNow);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user