You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
661 lines
24 KiB
661 lines
24 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Net;
|
|
using System.Threading;
|
|
using FluentAssertions;
|
|
using Moq;
|
|
using NLog;
|
|
using NUnit.Framework;
|
|
using NzbDrone.Common.Cache;
|
|
using NzbDrone.Common.EnvironmentInfo;
|
|
using NzbDrone.Common.Http;
|
|
using NzbDrone.Common.Http.Dispatchers;
|
|
using NzbDrone.Common.Http.Proxy;
|
|
using NzbDrone.Common.TPL;
|
|
using NzbDrone.Test.Common;
|
|
using NzbDrone.Test.Common.Categories;
|
|
|
|
namespace NzbDrone.Common.Test.Http
|
|
{
|
|
[IntegrationTest]
|
|
[TestFixture(typeof(ManagedHttpDispatcher))]
|
|
public class HttpClientFixture<TDispatcher> : TestBase<HttpClient> where TDispatcher : IHttpDispatcher
|
|
{
|
|
private static string[] _httpBinHosts = new[] { "eu.httpbin.org", "httpbin.org" };
|
|
private static int _httpBinRandom;
|
|
private string _httpBinHost;
|
|
|
|
[SetUp]
|
|
public void SetUp()
|
|
{
|
|
Mocker.GetMock<IPlatformInfo>().Setup(c => c.Version).Returns(new Version("1.0.0"));
|
|
Mocker.GetMock<IOsInfo>().Setup(c => c.Name).Returns("TestOS");
|
|
Mocker.GetMock<IOsInfo>().Setup(c => c.Version).Returns("9.0.0");
|
|
|
|
Mocker.SetConstant<IUserAgentBuilder>(Mocker.Resolve<UserAgentBuilder>());
|
|
|
|
Mocker.SetConstant<ICacheManager>(Mocker.Resolve<CacheManager>());
|
|
Mocker.SetConstant<ICreateManagedWebProxy>(Mocker.Resolve<ManagedWebProxyFactory>());
|
|
Mocker.SetConstant<IRateLimitService>(Mocker.Resolve<RateLimitService>());
|
|
Mocker.SetConstant<IEnumerable<IHttpRequestInterceptor>>(new IHttpRequestInterceptor[0]);
|
|
Mocker.SetConstant<IHttpDispatcher>(Mocker.Resolve<TDispatcher>());
|
|
|
|
// Used for manual testing of socks proxies.
|
|
//Mocker.GetMock<IHttpProxySettingsProvider>()
|
|
// .Setup(v => v.GetProxySettings(It.IsAny<HttpRequest>()))
|
|
// .Returns(new HttpProxySettings(ProxyType.Socks5, "127.0.0.1", 5476, "", false));
|
|
|
|
// Roundrobin over the two servers, to reduce the chance of hitting the ratelimiter.
|
|
_httpBinHost = _httpBinHosts[_httpBinRandom++ % _httpBinHosts.Length];
|
|
}
|
|
|
|
[Test]
|
|
public void should_execute_simple_get()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
var response = Subject.Execute(request);
|
|
|
|
response.Content.Should().NotBeNullOrWhiteSpace();
|
|
}
|
|
|
|
[Test]
|
|
public void should_execute_https_get()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
var response = Subject.Execute(request);
|
|
|
|
response.Content.Should().NotBeNullOrWhiteSpace();
|
|
}
|
|
|
|
[Test]
|
|
public void should_execute_typed_get()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get?test=1");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Url.EndsWith("/get?test=1");
|
|
response.Resource.Args.Should().Contain("test", "1");
|
|
}
|
|
|
|
[Test]
|
|
public void should_execute_simple_post()
|
|
{
|
|
var message = "{ my: 1 }";
|
|
|
|
var request = new HttpRequest($"https://{_httpBinHost}/post");
|
|
request.SetContent(message);
|
|
|
|
var response = Subject.Post<HttpBinResource>(request);
|
|
|
|
response.Resource.Data.Should().Be(message);
|
|
}
|
|
|
|
[TestCase("gzip")]
|
|
public void should_execute_get_using_gzip(string compression)
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/{compression}");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers["Accept-Encoding"].ToString().Should().Be(compression);
|
|
response.Headers.ContentLength.Should().BeLessOrEqualTo(response.Content.Length);
|
|
}
|
|
|
|
[TestCase(HttpStatusCode.Unauthorized)]
|
|
[TestCase(HttpStatusCode.Forbidden)]
|
|
[TestCase(HttpStatusCode.NotFound)]
|
|
[TestCase(HttpStatusCode.InternalServerError)]
|
|
[TestCase(HttpStatusCode.ServiceUnavailable)]
|
|
[TestCase(HttpStatusCode.BadGateway)]
|
|
public void should_throw_on_unsuccessful_status_codes(int statusCode)
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/status/{statusCode}");
|
|
|
|
var exception = Assert.Throws<HttpException>(() => Subject.Get<HttpBinResource>(request));
|
|
|
|
((int)exception.Response.StatusCode).Should().Be(statusCode);
|
|
|
|
ExceptionVerification.IgnoreWarns();
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_follow_redirects_when_not_in_production()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/redirect/1");
|
|
|
|
Subject.Get(request);
|
|
|
|
ExceptionVerification.ExpectedErrors(1);
|
|
}
|
|
|
|
[Test]
|
|
public void should_follow_redirects()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/redirect/1");
|
|
request.AllowAutoRedirect = true;
|
|
|
|
var response = Subject.Get(request);
|
|
|
|
response.StatusCode.Should().Be(HttpStatusCode.OK);
|
|
|
|
ExceptionVerification.ExpectedErrors(0);
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_follow_redirects()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/redirect/1");
|
|
request.AllowAutoRedirect = false;
|
|
|
|
var response = Subject.Get(request);
|
|
|
|
response.StatusCode.Should().Be(HttpStatusCode.Found);
|
|
|
|
ExceptionVerification.ExpectedErrors(1);
|
|
}
|
|
|
|
[Test]
|
|
public void should_follow_redirects_to_https()
|
|
{
|
|
if (typeof(TDispatcher) == typeof(ManagedHttpDispatcher) && PlatformInfo.IsMono)
|
|
{
|
|
Assert.Ignore("Will fail on tls1.2 via managed dispatcher, ignore.");
|
|
}
|
|
|
|
var request = new HttpRequestBuilder($"https://{_httpBinHost}/redirect-to")
|
|
.AddQueryParam("url", $"https://lidarr.audio/")
|
|
.Build();
|
|
request.AllowAutoRedirect = true;
|
|
|
|
var response = Subject.Get(request);
|
|
|
|
response.StatusCode.Should().Be(HttpStatusCode.OK);
|
|
response.Content.Should().Contain("Lidarr");
|
|
|
|
ExceptionVerification.ExpectedErrors(0);
|
|
}
|
|
|
|
[Test]
|
|
public void should_throw_on_too_many_redirects()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/redirect/4");
|
|
request.AllowAutoRedirect = true;
|
|
|
|
Assert.Throws<WebException>(() => Subject.Get(request));
|
|
|
|
ExceptionVerification.ExpectedErrors(0);
|
|
}
|
|
|
|
[Test]
|
|
public void should_send_user_agent()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().ContainKey("User-Agent");
|
|
|
|
var userAgent = response.Resource.Headers["User-Agent"].ToString();
|
|
|
|
userAgent.Should().Contain("Lidarr");
|
|
}
|
|
|
|
[TestCase("Accept", "text/xml, text/rss+xml, application/rss+xml")]
|
|
public void should_send_headers(string header, string value)
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
request.Headers.Add(header, value);
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers[header].ToString().Should().Be(value);
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_download_file_with_error()
|
|
{
|
|
var file = GetTempFilePath();
|
|
|
|
Assert.Throws<WebException>(() => Subject.DownloadFile("https://download.lidarr.audio/wrongpath", file));
|
|
|
|
File.Exists(file).Should().BeFalse();
|
|
|
|
ExceptionVerification.ExpectedWarns(1);
|
|
}
|
|
|
|
[Test]
|
|
public void should_send_cookie()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
request.Cookies["my"] = "cookie";
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().ContainKey("Cookie");
|
|
|
|
var cookie = response.Resource.Headers["Cookie"].ToString();
|
|
|
|
cookie.Should().Contain("my=cookie");
|
|
}
|
|
|
|
public void GivenOldCookie()
|
|
{
|
|
var oldRequest = new HttpRequest("https://eu.httpbin.org/get");
|
|
oldRequest.Cookies["my"] = "cookie";
|
|
|
|
var oldClient = new HttpClient(new IHttpRequestInterceptor[0], Mocker.Resolve<ICacheManager>(), Mocker.Resolve<IRateLimitService>(), Mocker.Resolve<IHttpDispatcher>(), Mocker.GetMock<IUserAgentBuilder>().Object, Mocker.Resolve<Logger>());
|
|
|
|
oldClient.Should().NotBeSameAs(Subject);
|
|
|
|
var oldResponse = oldClient.Get<HttpBinResource>(oldRequest);
|
|
|
|
oldResponse.Resource.Headers.Should().ContainKey("Cookie");
|
|
}
|
|
|
|
[Test]
|
|
public void should_preserve_cookie_during_session()
|
|
{
|
|
GivenOldCookie();
|
|
|
|
var request = new HttpRequest("https://eu.httpbin.org/get");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().ContainKey("Cookie");
|
|
|
|
var cookie = response.Resource.Headers["Cookie"].ToString();
|
|
|
|
cookie.Should().Contain("my=cookie");
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_send_cookie_to_other_host()
|
|
{
|
|
GivenOldCookie();
|
|
|
|
var request = new HttpRequest("https://httpbin.org/get");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().NotContainKey("Cookie");
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_store_request_cookie()
|
|
{
|
|
var requestGet = new HttpRequest($"https://{_httpBinHost}/get");
|
|
requestGet.Cookies.Add("my", "cookie");
|
|
requestGet.AllowAutoRedirect = false;
|
|
requestGet.StoreRequestCookie = false;
|
|
requestGet.StoreResponseCookie = false;
|
|
var responseGet = Subject.Get<HttpBinResource>(requestGet);
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.AllowAutoRedirect = false;
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().BeEmpty();
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_store_request_cookie()
|
|
{
|
|
var requestGet = new HttpRequest($"https://{_httpBinHost}/get");
|
|
requestGet.Cookies.Add("my", "cookie");
|
|
requestGet.AllowAutoRedirect = false;
|
|
requestGet.StoreRequestCookie.Should().BeTrue();
|
|
requestGet.StoreResponseCookie = false;
|
|
var responseGet = Subject.Get<HttpBinResource>(requestGet);
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.AllowAutoRedirect = false;
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_delete_request_cookie()
|
|
{
|
|
var requestDelete = new HttpRequest($"https://{_httpBinHost}/cookies/delete?my");
|
|
requestDelete.Cookies.Add("my", "cookie");
|
|
requestDelete.AllowAutoRedirect = true;
|
|
requestDelete.StoreRequestCookie = false;
|
|
requestDelete.StoreResponseCookie = false;
|
|
|
|
// Delete and redirect since that's the only way to check the internal temporary cookie container
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestDelete);
|
|
|
|
responseCookies.Resource.Cookies.Should().BeEmpty();
|
|
}
|
|
|
|
[Test]
|
|
public void should_clear_request_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestSet.Cookies.Add("my", "cookie");
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreRequestCookie = true;
|
|
requestSet.StoreResponseCookie = false;
|
|
|
|
var responseSet = Subject.Get<HttpCookieResource>(requestSet);
|
|
|
|
var requestClear = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestClear.Cookies.Add("my", null);
|
|
requestClear.AllowAutoRedirect = false;
|
|
requestClear.StoreRequestCookie = true;
|
|
requestClear.StoreResponseCookie = false;
|
|
|
|
var responseClear = Subject.Get<HttpCookieResource>(requestClear);
|
|
|
|
responseClear.Resource.Cookies.Should().BeEmpty();
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_store_response_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies/set?my=cookie");
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreRequestCookie = false;
|
|
requestSet.StoreResponseCookie.Should().BeFalse();
|
|
|
|
var responseSet = Subject.Get(requestSet);
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().BeEmpty();
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_store_response_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies/set?my=cookie");
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreRequestCookie = false;
|
|
requestSet.StoreResponseCookie = true;
|
|
|
|
var responseSet = Subject.Get(requestSet);
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_temp_store_response_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies/set?my=cookie");
|
|
requestSet.AllowAutoRedirect = true;
|
|
requestSet.StoreRequestCookie = false;
|
|
requestSet.StoreResponseCookie.Should().BeFalse();
|
|
var responseSet = Subject.Get<HttpCookieResource>(requestSet);
|
|
|
|
// Set and redirect since that's the only way to check the internal temporary cookie container
|
|
responseSet.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_overwrite_response_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies/set?my=cookie");
|
|
requestSet.Cookies.Add("my", "oldcookie");
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreRequestCookie = false;
|
|
requestSet.StoreResponseCookie = true;
|
|
|
|
var responseSet = Subject.Get(requestSet);
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_overwrite_temp_response_cookie()
|
|
{
|
|
var requestSet = new HttpRequest($"https://{_httpBinHost}/cookies/set?my=cookie");
|
|
requestSet.Cookies.Add("my", "oldcookie");
|
|
requestSet.AllowAutoRedirect = true;
|
|
requestSet.StoreRequestCookie = true;
|
|
requestSet.StoreResponseCookie = false;
|
|
|
|
var responseSet = Subject.Get<HttpCookieResource>(requestSet);
|
|
|
|
responseSet.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "oldcookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_not_delete_response_cookie()
|
|
{
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.Cookies.Add("my", "cookie");
|
|
requestCookies.AllowAutoRedirect = false;
|
|
requestCookies.StoreRequestCookie = true;
|
|
requestCookies.StoreResponseCookie = false;
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
var requestDelete = new HttpRequest($"https://{_httpBinHost}/cookies/delete?my");
|
|
requestDelete.AllowAutoRedirect = false;
|
|
requestDelete.StoreRequestCookie = false;
|
|
requestDelete.StoreResponseCookie = false;
|
|
|
|
var responseDelete = Subject.Get(requestDelete);
|
|
|
|
requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.StoreRequestCookie = false;
|
|
requestCookies.StoreResponseCookie = false;
|
|
|
|
responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_delete_response_cookie()
|
|
{
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.Cookies.Add("my", "cookie");
|
|
requestCookies.AllowAutoRedirect = false;
|
|
requestCookies.StoreRequestCookie = true;
|
|
requestCookies.StoreResponseCookie = false;
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
var requestDelete = new HttpRequest($"https://{_httpBinHost}/cookies/delete?my");
|
|
requestDelete.AllowAutoRedirect = false;
|
|
requestDelete.StoreRequestCookie = false;
|
|
requestDelete.StoreResponseCookie = true;
|
|
|
|
var responseDelete = Subject.Get(requestDelete);
|
|
|
|
requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.StoreRequestCookie = false;
|
|
requestCookies.StoreResponseCookie = false;
|
|
|
|
responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().BeEmpty();
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_delete_temp_response_cookie()
|
|
{
|
|
var requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.Cookies.Add("my", "cookie");
|
|
requestCookies.AllowAutoRedirect = false;
|
|
requestCookies.StoreRequestCookie = true;
|
|
requestCookies.StoreResponseCookie = false;
|
|
var responseCookies = Subject.Get<HttpCookieResource>(requestCookies);
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
var requestDelete = new HttpRequest($"https://{_httpBinHost}/cookies/delete?my");
|
|
requestDelete.AllowAutoRedirect = true;
|
|
requestDelete.StoreRequestCookie = false;
|
|
requestDelete.StoreResponseCookie = false;
|
|
var responseDelete = Subject.Get<HttpCookieResource>(requestDelete);
|
|
|
|
responseDelete.Resource.Cookies.Should().BeEmpty();
|
|
|
|
requestCookies = new HttpRequest($"https://{_httpBinHost}/cookies");
|
|
requestCookies.StoreRequestCookie = false;
|
|
requestCookies.StoreResponseCookie = false;
|
|
|
|
responseCookies.Resource.Cookies.Should().HaveCount(1).And.Contain("my", "cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
|
|
[Test]
|
|
public void should_throw_on_http429_too_many_requests()
|
|
{
|
|
var request = new HttpRequest($"https://{_httpBinHost}/status/429");
|
|
|
|
Assert.Throws<TooManyRequestsException>(() => Subject.Get(request));
|
|
|
|
ExceptionVerification.IgnoreWarns();
|
|
}
|
|
|
|
[Test]
|
|
public void should_call_interceptor()
|
|
{
|
|
Mocker.SetConstant<IEnumerable<IHttpRequestInterceptor>>(new [] { Mocker.GetMock<IHttpRequestInterceptor>().Object });
|
|
|
|
Mocker.GetMock<IHttpRequestInterceptor>()
|
|
.Setup(v => v.PreRequest(It.IsAny<HttpRequest>()))
|
|
.Returns<HttpRequest>(r => r);
|
|
|
|
Mocker.GetMock<IHttpRequestInterceptor>()
|
|
.Setup(v => v.PostResponse(It.IsAny<HttpResponse>()))
|
|
.Returns<HttpResponse>(r => r);
|
|
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
Subject.Get(request);
|
|
|
|
Mocker.GetMock<IHttpRequestInterceptor>()
|
|
.Verify(v => v.PreRequest(It.IsAny<HttpRequest>()), Times.Once());
|
|
|
|
Mocker.GetMock<IHttpRequestInterceptor>()
|
|
.Verify(v => v.PostResponse(It.IsAny<HttpResponse>()), Times.Once());
|
|
}
|
|
|
|
[TestCase("en-US")]
|
|
[TestCase("es-ES")]
|
|
public void should_parse_malformed_cloudflare_cookie(string culture)
|
|
{
|
|
var origCulture = Thread.CurrentThread.CurrentCulture;
|
|
Thread.CurrentThread.CurrentCulture = CultureInfo.GetCultureInfo(culture);
|
|
Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo(culture);
|
|
try
|
|
{
|
|
// the date is bad in the below - should be 13-Jul-2026
|
|
string malformedCookie = @"__cfduid=d29e686a9d65800021c66faca0a29b4261436890790; expires=Mon, 13-Jul-26 16:19:50 GMT; path=/; HttpOnly";
|
|
var requestSet = new HttpRequestBuilder($"https://{_httpBinHost}/response-headers")
|
|
.AddQueryParam("Set-Cookie", malformedCookie)
|
|
.Build();
|
|
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreResponseCookie = true;
|
|
|
|
var responseSet = Subject.Get(requestSet);
|
|
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().ContainKey("Cookie");
|
|
|
|
var cookie = response.Resource.Headers["Cookie"].ToString();
|
|
|
|
cookie.Should().Contain("__cfduid=d29e686a9d65800021c66faca0a29b4261436890790");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
finally
|
|
{
|
|
Thread.CurrentThread.CurrentCulture = origCulture;
|
|
Thread.CurrentThread.CurrentUICulture = origCulture;
|
|
}
|
|
}
|
|
|
|
[TestCase("lang_code=en; expires=Wed, 23-Dec-2026 18:09:14 GMT; Max-Age=31536000; path=/; domain=.abc.com")]
|
|
public void should_reject_malformed_domain_cookie(string malformedCookie)
|
|
{
|
|
try
|
|
{
|
|
string url = $"https://{_httpBinHost}/response-headers?Set-Cookie={Uri.EscapeUriString(malformedCookie)}";
|
|
|
|
var requestSet = new HttpRequest(url);
|
|
requestSet.AllowAutoRedirect = false;
|
|
requestSet.StoreResponseCookie = true;
|
|
|
|
var responseSet = Subject.Get(requestSet);
|
|
|
|
var request = new HttpRequest($"https://{_httpBinHost}/get");
|
|
|
|
var response = Subject.Get<HttpBinResource>(request);
|
|
|
|
response.Resource.Headers.Should().NotContainKey("Cookie");
|
|
|
|
ExceptionVerification.IgnoreErrors();
|
|
}
|
|
finally
|
|
{
|
|
}
|
|
}
|
|
}
|
|
|
|
public class HttpBinResource
|
|
{
|
|
public Dictionary<string, object> Args { get; set; }
|
|
public Dictionary<string, object> Headers { get; set; }
|
|
public string Origin { get; set; }
|
|
public string Url { get; set; }
|
|
public string Data { get; set; }
|
|
}
|
|
|
|
public class HttpCookieResource
|
|
{
|
|
public Dictionary<string, string> Cookies { get; set; }
|
|
}
|
|
}
|