// 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 #include #include #include "base/bind.h" #include "base/memory/ptr_util.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "base/test/test_simple_task_runner.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 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::Bind(&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_ = new net::IOBufferWithSize(upload_size); rv = upload_stream_->Read( upload_buffer_.get(), upload_size, base::Bind(&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 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 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( message_loop_.task_runner())), interceptor_(new UploadInterceptor()), uploader_(DomainReliabilityUploader::Create( &time_, url_request_context_getter_)) { net::URLRequestFilter::GetInstance()->AddUrlInterceptor( GURL(kUploadURL), base::WrapUnique(interceptor_)); uploader_->set_discard_uploads(false); } ~DomainReliabilityUploaderTest() override { net::URLRequestFilter::GetInstance()->ClearHandlers(); } DomainReliabilityUploader* uploader() const { return uploader_.get(); } UploadInterceptor* interceptor() const { return interceptor_; } scoped_refptr url_request_context_getter() { return url_request_context_getter_; } private: base::MessageLoopForIO message_loop_; scoped_refptr url_request_context_getter_; UploadInterceptor* interceptor_; MockTime time_; std::unique_ptr 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