blob: 60f9491832690e651fb9b7cce34eec6dc138af70 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/domain_reliability/uploader.h"
#include <stddef.h>
#include <memory>
#include <string>
#include "base/bind.h"
#include "base/memory/ptr_util.h"
#include "base/run_loop.h"
#include "base/test/scoped_task_environment.h"
#include "base/test/test_simple_task_runner.h"
#include "base/threading/thread_task_runner_handle.h"
#include "components/domain_reliability/test_util.h"
#include "net/base/load_flags.h"
#include "net/http/http_response_headers.h"
#include "net/http/http_response_info.h"
#include "net/log/net_log_with_source.h"
#include "net/url_request/url_request.h"
#include "net/url_request/url_request_filter.h"
#include "net/url_request/url_request_interceptor.h"
#include "net/url_request/url_request_job.h"
#include "net/url_request/url_request_test_util.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace domain_reliability {
namespace {
const char kUploadURL[] = "https://example/upload";
struct MockUploadResult {
int net_error;
scoped_refptr<net::HttpResponseHeaders> response_headers;
};
class UploadMockURLRequestJob : public net::URLRequestJob {
public:
UploadMockURLRequestJob(net::URLRequest* request,
net::NetworkDelegate* network_delegate,
MockUploadResult result)
: net::URLRequestJob(request, network_delegate),
upload_stream_(nullptr),
result_(result) {
int load_flags = request->load_flags();
EXPECT_TRUE(load_flags & net::LOAD_DO_NOT_SEND_COOKIES);
EXPECT_TRUE(load_flags & net::LOAD_DO_NOT_SAVE_COOKIES);
}
protected:
void Start() override {
int rv = upload_stream_->Init(
base::BindOnce(&UploadMockURLRequestJob::OnStreamInitialized,
base::Unretained(this)),
net::NetLogWithSource());
if (rv == net::ERR_IO_PENDING)
return;
OnStreamInitialized(rv);
}
void SetUpload(net::UploadDataStream* upload_stream) override {
upload_stream_ = upload_stream;
}
private:
~UploadMockURLRequestJob() override {}
void OnStreamInitialized(int rv) {
EXPECT_EQ(net::OK, rv);
size_t upload_size = upload_stream_->size();
upload_buffer_ = base::MakeRefCounted<net::IOBufferWithSize>(upload_size);
rv = upload_stream_->Read(
upload_buffer_.get(), upload_size,
base::BindOnce(&UploadMockURLRequestJob::OnStreamRead,
base::Unretained(this)));
if (rv == net::ERR_IO_PENDING)
return;
OnStreamRead(rv);
}
void OnStreamRead(int rv) {
EXPECT_EQ(upload_buffer_->size(), rv);
upload_data_ = std::string(upload_buffer_->data(), upload_buffer_->size());
upload_buffer_ = nullptr;
if (result_.net_error == net::OK)
NotifyHeadersComplete();
else if (result_.net_error != net::ERR_IO_PENDING)
NotifyStartError(net::URLRequestStatus::FromError(result_.net_error));
}
void GetResponseInfo(net::HttpResponseInfo* info) override {
info->headers = result_.response_headers;
}
net::UploadDataStream* upload_stream_;
scoped_refptr<net::IOBufferWithSize> upload_buffer_;
std::string upload_data_;
MockUploadResult result_;
};
class UploadInterceptor : public net::URLRequestInterceptor {
public:
UploadInterceptor() : request_count_(0), last_upload_depth_(-1) {}
~UploadInterceptor() override { EXPECT_TRUE(results_.empty()); }
net::URLRequestJob* MaybeInterceptRequest(
net::URLRequest* request,
net::NetworkDelegate* delegate) const override {
EXPECT_FALSE(results_.empty());
MockUploadResult result = results_.front();
results_.pop_front();
last_upload_depth_ =
DomainReliabilityUploader::GetURLRequestUploadDepth(*request);
++request_count_;
return new UploadMockURLRequestJob(request, delegate, result);
}
void ExpectRequestAndReturnError(int net_error) {
MockUploadResult result;
result.net_error = net_error;
results_.push_back(result);
}
void ExpectRequestAndReturnResponseHeaders(const char* headers) {
MockUploadResult result;
result.net_error = net::OK;
result.response_headers = new net::HttpResponseHeaders(
net::HttpUtil::AssembleRawHeaders(headers, strlen(headers)));
results_.push_back(result);
}
int request_count() const { return request_count_; }
int last_upload_depth() const { return last_upload_depth_; }
private:
mutable std::list<MockUploadResult> results_;
mutable int request_count_;
mutable int last_upload_depth_;
};
class TestUploadCallback {
public:
TestUploadCallback() : called_count_(0u) {}
DomainReliabilityUploader::UploadCallback callback() {
return base::Bind(&TestUploadCallback::OnCalled, base::Unretained(this));
}
unsigned called_count() const { return called_count_; }
DomainReliabilityUploader::UploadResult last_result() const {
return last_result_;
}
private:
void OnCalled(const DomainReliabilityUploader::UploadResult& result) {
called_count_++;
last_result_ = result;
}
unsigned called_count_;
DomainReliabilityUploader::UploadResult last_result_;
};
class DomainReliabilityUploaderTest : public testing::Test {
protected:
DomainReliabilityUploaderTest()
: url_request_context_getter_(new net::TestURLRequestContextGetter(
base::ThreadTaskRunnerHandle::Get())),
interceptor_(new UploadInterceptor()),
uploader_(
DomainReliabilityUploader::Create(&time_,
url_request_context_getter_)) {
net::URLRequestFilter::GetInstance()->AddUrlInterceptor(
GURL(kUploadURL), base::WrapUnique(interceptor_));
uploader_->SetDiscardUploads(false);
}
~DomainReliabilityUploaderTest() override {
net::URLRequestFilter::GetInstance()->ClearHandlers();
}
DomainReliabilityUploader* uploader() const { return uploader_.get(); }
UploadInterceptor* interceptor() const { return interceptor_; }
scoped_refptr<net::TestURLRequestContextGetter> url_request_context_getter() {
return url_request_context_getter_;
}
private:
base::test::ScopedTaskEnvironment task_environment_{
base::test::ScopedTaskEnvironment::MainThreadType::IO};
scoped_refptr<net::TestURLRequestContextGetter> url_request_context_getter_;
UploadInterceptor* interceptor_;
MockTime time_;
std::unique_ptr<DomainReliabilityUploader> uploader_;
};
TEST_F(DomainReliabilityUploaderTest, Null) {
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, SuccessfulUpload) {
interceptor()->ExpectRequestAndReturnResponseHeaders("HTTP/1.1 200\r\n\r\n");
TestUploadCallback c;
uploader()->UploadReport("{}", 0, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_TRUE(c.last_result().is_success());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, NetworkErrorUpload) {
interceptor()->ExpectRequestAndReturnError(net::ERR_CONNECTION_REFUSED);
TestUploadCallback c;
uploader()->UploadReport("{}", 0, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_TRUE(c.last_result().is_failure());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, ServerErrorUpload) {
interceptor()->ExpectRequestAndReturnResponseHeaders("HTTP/1.1 500\r\n\r\n");
TestUploadCallback c;
uploader()->UploadReport("{}", 0, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_TRUE(c.last_result().is_failure());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, RetryAfterUpload) {
interceptor()->ExpectRequestAndReturnResponseHeaders(
"HTTP/1.1 503 Ugh\nRetry-After: 3600\n\n");
TestUploadCallback c;
uploader()->UploadReport("{}", 0, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_TRUE(c.last_result().is_retry_after());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, UploadDepth1) {
interceptor()->ExpectRequestAndReturnResponseHeaders("HTTP/1.1 200\r\n\r\n");
TestUploadCallback c;
uploader()->UploadReport("{}", 0, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_EQ(1, interceptor()->last_upload_depth());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, UploadDepth2) {
interceptor()->ExpectRequestAndReturnResponseHeaders("HTTP/1.1 200\r\n\r\n");
TestUploadCallback c;
uploader()->UploadReport("{}", 1, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_EQ(2, interceptor()->last_upload_depth());
uploader()->Shutdown();
}
TEST_F(DomainReliabilityUploaderTest, UploadCanceledAtShutdown) {
interceptor()->ExpectRequestAndReturnError(net::ERR_IO_PENDING);
TestUploadCallback c;
uploader()->UploadReport("{}", 1, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1, interceptor()->request_count());
EXPECT_EQ(0u, c.called_count());
uploader()->Shutdown();
EXPECT_EQ(0u, c.called_count());
url_request_context_getter()->GetURLRequestContext()->AssertNoURLRequests();
}
TEST_F(DomainReliabilityUploaderTest, NoUploadAfterShutdown) {
uploader()->Shutdown();
TestUploadCallback c;
uploader()->UploadReport("{}", 1, GURL(kUploadURL), c.callback());
base::RunLoop().RunUntilIdle();
EXPECT_EQ(1u, c.called_count());
EXPECT_EQ(0, interceptor()->request_count());
}
} // namespace
} // namespace domain_reliability